将4位灰度字节数组转换为Android位图

5

我正在从相机中检索一张原始图像,该图像的规格如下:

  • 80 x 60 分辨率
  • 4位灰度

我将图像作为字节数组检索,并拥有一个长度为2400 (1/2 * 80 * 60) 字节的数组。下一步是将字节数组转换为位图。我已经使用了

BitmapFactory.decodeByteArray(bytes, 0, bytes.length)

但是这并没有返回一个可显示的图片。我看了这篇文章,并将下面的代码复制到我的安卓应用程序中,但我得到了“缓冲区不足以容纳像素”的运行时错误。

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。 我获取的图像是来自这个相机的原始图像。不幸的是,检索预压缩的JPEG图像不是一个选项,因为我需要4位灰度。
2个回答

2
如果输入的图像只有2400字节,那么每个字节中有两个像素(每个像素4位)。当使用ARGB_8888格式时,每个像素需要4个字节,即60 * 80 * 4 = 19200字节。因此,你需要将每个输入字节拆分为上/下半字节值,然后应用于接下来的8个字节(不包括alpha通道)。可以参考这个答案中的示例来了解如何拆分字节。
  • 将传入的字节i分成两个半字节,iaib
  • ia应用于传出字节i*8(i*8)+2
  • ib应用于传出字节(i*8)+4(i*8)+6
  • 字节(i*8)+3(i*8)+7为alpha(0xFF

一旦您拥有正确大小的字节缓冲区,您应该能够毫无问题地使用decodeByteArry()


感谢您的回复。我尝试了这个方法,但不幸的是它没有起作用。以下是我的代码: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] = seperatedBytes[i * 8 + 3] = (byte) ((jpegBytes[i] >> 4) & (byte) 0x0F); seperatedBytes[i * 8 + 4] = seperatedBytes[i * 8 + 5] = seperatedBytes[i * 8 + 6] = seperatedBytes[i * 8 + 7] = (byte) (jpegBytes[i] & 0x0F); } - Ankit Goyal
然后这样: `Bitmap bm = BitmapFactory.decodeByteArray(seperatedBytes, 0, seperatedBytes.length);`我将 .jpg 文件保存到了 Android 上,但是它显示的文件大小为 0KB。数组 jpegBytes 的长度为 2400,而 seperatedBytes 数组的长度为 19200。 - Ankit Goyal
这是我从中检索图像的相机的数据表。虽然它可以发送JPEG图像,但我需要4位灰度图像。此外,我更改了我的代码,使每个组的第3和第7个字节等于Alpha。更新后的代码在“UPDATE”中,但我仍然得到大小为0KB的位图。 - Ankit Goyal
哦,那可能就是问题所在了。看起来我们正在创建一个 RGBA_8888,但应该是 ARGB_8888 才能匹配格式。尝试交换每个像素的第一个/最后一个字节。不过这仍然不应该导致大小为0。你是如何保存图像的? - Geobits
同时 - 我交换了每个像素的第一个/最后一个字节,但图像大小仍然为0。 - Ankit Goyal
显示剩余7条评论

1
如果我理解正确,您有一个字节数组,其中的字节包含2个灰度值。我认为可以将灰度值视为简单的强度值,不是吗?因此,您需要做的第一件事是将灰度值分隔成单个字节,因为BitmapFactory.decodeByteArray可能无法处理您的“半字节”。这可以通过位运算轻松完成。要获取字节`0bxxxxxxxx`的前4位,您需要右移4次:`0bxxxxxxxx >> 4`,这将导致`0b0000xxxx`。第二个值可以通过与模式`0b00001111`进行按位或运算来获得:`0b00001111 ^ 0bxxxxxxxx`,这将导致`0b0000xxxx`。(有关位运算的详细信息,请参见此处: Bit-Operations
这两个值现在可以存储到一个新的字节数组中。如果对每一对半字节都执行此操作,则最终将得到一个完整字节和加倍大小的数组。

我不确定这是否已足够用于“BitmapFactory.decodeByteArray”,或者是否需要为每个RGBA通道重复每个字节4次,这将再次将您的字节数组大小增加原始大小的四倍。 我希望我没有误解任何内容,并且我的建议能够帮助您 ;)

我遇到了与上述描述相同的问题 - 每个图像都是0KB。我还按照此帖子将图像解码为RGBA数组。即使如此,图像仍然是空白的。 - Ankit Goyal

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接