我正在从相机中检索原始图像,图像的规范如下:
- 80 x 60 分辨率
- 4 位灰度
我将图像检索为字节数组,并且有一个长度为 2400 (1/2 * 80 * 60) 字节的数组。下一步是将字节数组转换为位图。我已经用过了
BitmapFactory.decodeByteArray(bytes, 0, bytes.length)
但这并没有返回可显示的图像。我看了this post并将下面的代码复制到我的 Android 应用程序中,但出现“缓冲区不够大,无法容纳像素”运行时错误。
byte [] Src; //Comes from somewhere...
byte [] Bits = new byte[Src.length*4]; //That's where the RGBA array goes.
int i;
for(i=0;i<Src.length;i++)
{
Bits[i*4] =
Bits[i*4+1] =
Bits[i*4+2] = ~Src[i]; //Invert the source bits
Bits[i*4+3] = -1;//0xff, that's the alpha.
}
//Now put these nice RGBA pixels into a Bitmap object
Bitmap bm = Bitmap.createBitmap(Width, Height, Bitmap.Config.ARGB_8888);
bm.copyPixelsFromBuffer(ByteBuffer.wrap(Bits));
在线程的底部,原始发布者有与我目前相同的错误。但是,他的问题已通过上面粘贴的代码解决。有没有人对我应该如何将原始图像或 RGBA 数组转换为位图有任何建议?
非常感谢!
更新:
我听从了 Geobits 的建议,这是我的新代码
byte[] seperatedBytes = new byte[jpegBytes.length * 8];
for (int i = 0; i < jpegBytes.length; i++) {
seperatedBytes[i * 8] = seperatedBytes[i * 8 + 1] = seperatedBytes[i * 8 + 2] = (byte) ((jpegBytes[i] >> 4) & (byte) 0x0F);
seperatedBytes[i * 8 + 4] = seperatedBytes[i * 8 + 5] = seperatedBytes[i * 8 + 6] = (byte) (jpegBytes[i] & 0x0F);
seperatedBytes[i * 8 + 3] = seperatedBytes[i * 8 + 7] = -1; //0xFF
}
现在,我可以使用此命令获取位图
Bitmap bm = BitmapFactory.decodeByteArray(seperatedBytes, 0, seperatedBytes.length);
但位图的大小为 0KB。
我得到的图像是来自 this camera 的原始图像.不幸的是,检索预压缩的 JPEG 图像不是一种选择,因为我需要 4 位灰度。
最佳答案
如果传入的图像只有 2400 字节,这意味着每个字节有两个像素(每个像素 4 位)。当 ARGB_8888 每个像素需要 4 个字节或 60 * 80 * 4 = 19200
时,您只需要给字节缓冲区 2400 * 4 = 9600
字节。
您需要将每个传入字节拆分为高/低半字节值,然后将其应用于以下 8 个字节(不包括 alpha)。可以看到this answer有关如何拆分字节的示例。
基本上:
- 将传入字节
i
分成两个半字节,ia
和ib
- 将
ia
应用于输出字节i*8
到(i*8)+2
- 将
ib
应用于输出字节(i*8)+4
到(i*8)+6
- 字节
(i*8)+3
和(i*8)+7
是 alpha (0xFF
)
一旦您拥有合适大小的字节缓冲区,您应该能够毫无问题地使用decodeByteArry()
。
关于java - 将 4 位灰度字节数组转换为位图 Android,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/17735097/