Невозможно сохранить изменения после сохранения или загрузки измененного изображения.
Я использую стеганографический код из этой ссылки для встраивания информации во все каналы ARGB изображения. Код был написан для Java, но я изменил его для Android, используя Bitmap для работы с изображением.
Я преобразую длину сообщения в 32-разрядное число, которое храню в первых 8 пикселях. 4 бита на пиксель, с одним битом в младшем бите каждого из каналов A, R, G и B. Однако, когда я извлекаю биты для длины сообщения и преобразую их в целое число в extractInteger
функция, я получаю 2147483647, что позже приводит к исключению OutOfMemory.
Я попытался следующее, чтобы найти источник ошибки.
- Я использую изображение M для встраивания своей информации, таким образом изменяя ее на M'.
- Я проверяю значение измененных пикселей M'.
- Я спасаю М '.
- Я перезагружаю M 'для извлечения, но значение пикселей не всегда такое же, как указано выше.
Поэтому я подозреваю, что существует проблема либо с сохранением M ', либо с загрузкой M'. Последнее может возникнуть из-за того, что я изменяю значение альфа-канала как часть моего процесса встраивания, но при загрузке M 'по умолчанию устанавливается значение 255.
Если в коде нет ничего плохого, могу ли я что-нибудь решить, чтобы решить проблему? Благодарю.
Функция сохранения изображения:
private void saveImage() {
path = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES + "/Testing/");
date = new Date() ;
dateFormat = new SimpleDateFormat("yyyyMMdd_HHmmss", Locale.getDefault()) ;
filename = new File(dateFormat.format(date) + ".png");
fileName = filename.toString();
file = new File(path, fileName);
filePath = file.toString();
FileOutputStream out = null;
try {
out = new FileOutputStream(filePath);
embeddedImg.compress(Bitmap.CompressFormat.PNG, 100, out);
out.close();
Toast.makeText(this, "Saved!", Toast.LENGTH_SHORT).show();
sendBroadcast(new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE, Uri.parse("file://"+file.getAbsolutePath())));
} catch (Exception e) {
e.printStackTrace();
}
}
Отрезано от функции загрузки изображения:
Bundle getImage = this.getIntent().getExtras();
gotImage = getImage.getString("decode");
BitmapFactory.Options op = new BitmapFactory.Options();
op.inPreferredConfig = Bitmap.Config.ARGB_8888;
try {
bitmapImg = BitmapFactory.decodeFile(gotImage, op);
newBitmapImg = bitmapImg.copy(Bitmap.Config.ARGB_8888, true);
decodeMessage();
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
Функции декодирования:
private void decodeMessage() {
int len = extractInteger(newBitmapImg, 0);
byte b[] = new byte[len];
for(int i=0; i<len; i++)
b[i] = extractByte(newBitmapImg, i*2+8);
viewMessage.setText(new String(b));
}
private int extractInteger(Bitmap img, int start) {
int maxX = img.getWidth(), maxY = img.getHeight(),
startX = start/maxY, startY = start - startX*maxY, count=0;
int length = 0;
for(int i=startX; i<maxX && count<32; i++) {
for(int j=startY; j<maxY && count<32; j++) {
int rgb = img.getPixel(i, j), bit = getBitValue(rgb, 0);
length = setBitValue(length, count, bit);
bit = getBitValue(rgb, 8); length = setBitValue(length, count+1, bit);
bit = getBitValue(rgb, 16); length = setBitValue(length, count+2, bit);
bit = getBitValue(rgb, 24); length = setBitValue(length, count+3, bit);
count = count+4;
}
}
return length;
}
private byte extractByte(Bitmap img, int start) {
int maxX = img.getWidth(), maxY = img.getHeight(),
startX = start/maxY, startY = start - startX*maxY, count=0;
byte b = 0;
for(int i=startX; i<maxX && count<8; i++) {
for(int j=startY; j<maxY && count<8; j++) {
if(j==maxY-1){
startY = 0;
}
int rgb = img.getPixel(i, j), bit = getBitValue(rgb, 0);
b = (byte)setBitValue(b, count, bit);
bit = getBitValue(rgb, 8); b = (byte)setBitValue(b, count+1, bit);
bit = getBitValue(rgb, 16); b = (byte)setBitValue(b, count+2, bit);
bit = getBitValue(rgb, 24); b = (byte)setBitValue(b, count+3, bit);
count = count+4;
}
}
return b;
}
private int getBitValue(int n, int location) { //n=messageLength, location=count
int v = n & (int) Math.round(Math.pow(2, location));
return v==0?0:1;
}
private int setBitValue(int n, int location, int bit) {
int toggle = (int) Math.pow(2, location), bv = getBitValue(n, location);
if(bv == bit)
return n;
if(bv == 0 && bit == 1){
n |= toggle;
System.out.println("n{toggle: "+n);
}else if(bv == 1 && bit == 0){
n ^= toggle;
}
return n;
}