在做一些安卓偏底层开发时候,不得不面对图像数据和字节数组转换工作,但是往往理解不深,产生比较多的疑惑,今天专门做了一些实验、越多一些资料汇总

研究一:理解-1是怎么转换为FF颜色值的

注意:负数在计算中以原码的补码形式表达,例如-1对应的二进制是11111111111111111111111111

①: byte color=-1,a的取值范围是-128~127

②: -1转换为二进制表示 String b = Integer.toBinaryString(color);输出111..111(32个)

③: 转换为16进制字符串表示(注意不应理解为16进制值,只是每个字节转换为16进制表达而已,FFFFFFFF 16进制值实际对应了一个非常大的正常,不要混淆) String b =Integer.toHexString(color);输FFFFFFFF,这是二进制的16进制表达形式,不要理解为16进值了

④:byte颜色值转换为16进制 Integer.toHexString(color & 0xff),与0xff,消除了前面的24位,这样

11111111就对应16进制的FF,10进制的255(-1对应了颜色值的FF,也就是255)

研究二:ARGB_8888、ARGB_4444、 RGB_565、 ALPHA_8概念理解

最后8888表示每个像素的单通道占用了8位,想必4444,占用的位数越多,表达的越清晰

研究三:ARGB_8888顺序(Google给安卓兄弟留的坑)

通过实践,我们可以得知,ARGB_8888 格式图片的各通道顺序其实不是 ARGB,而是 RGBA

public static Bitmap getBitmap(){
        String TAG = "TTTTTTTTTT";
        Bitmap bitmap = Bitmap.createBitmap(1, 1, Bitmap.Config.ARGB_8888);
        bitmap.eraseColor(0xff336699); // AARRGGBB
//        byte[] bytes = new byte[bitmap.getWidth() * bitmap.getHeight() * 4];
//        Buffer dst = ByteBuffer.wrap(bytes);

        ByteBuffer buffer = ByteBuffer.allocate(bitmap.getWidth() * bitmap.getHeight() * 4); //  使用allocate()静态方法创建字节缓冲区
        bitmap.copyPixelsToBuffer(buffer); // 将位图的像素复制到指定的缓冲区
        byte[] bytes = buffer.array();

        // ARGB_8888 真实的存储顺序是 R-G-B-A
        Log.d(TAG, "R: " + Integer.toHexString(bytes[0] & 0xff));
        Log.d(TAG, "G: " + Integer.toHexString(bytes[1] & 0xff));
        Log.d(TAG, "B: " + Integer.toHexString(bytes[2] & 0xff));
        Log.d(TAG, "A: " + Integer.toHexString(bytes[3] & 0xff));
        return bitmap;

    }
    
TTTTTTTTTT: R: 33
 TTTTTTTTTT: G: 66
TTTTTTTTTT: B: 99
 TTTTTTTTTT: A: ff

研究四:bitmap和字节的转换

A.主要看下ARGB_8888格式的bitmap转字节(平时这种格式差不多足够了)

//        byte[] bytes = new byte[bitmap.getWidth() * bitmap.getHeight() * 4];
//        Buffer dst = ByteBuffer.wrap(bytes);
        ByteBuffer buffer = ByteBuffer.allocate(bitmap.getWidth() * bitmap.getHeight() * 4); //  使用allocate()静态方法创建字节缓冲区
        bitmap.copyPixelsToBuffer(buffer); // 将位图的像素复制到指定的缓冲区
        byte[] bytes = buffer.array();

a.对于ARGB_8888,应该申请的字节缓存区大小是ByteBuffer.allocate(w*h*4)

b.对于ARGB_4444,经过实验和阅读资料,可能默认被当作ARGB_8888处理了

c.RGB_565 w*h*2大小就足够了,但是怎么从字节数组中转换会16进色值未研究,因为R只占用5bit

经过实验,搞成w*h*3也是不行的,看样子可能需要解析5bit,6bit,5bit这样操作

B.字节转换为Bitmap

注意:字节实际内容和ARGB_8888要一致

public static Bitmap createBitmapByByte(){
    String TAG = "TTTTTTTTTT";
    Bitmap bitmap= Bitmap.createBitmap(1, 1, Bitmap.Config.ARGB_8888);
    // 缓存区是R、G、B、A存储顺序
    // 根据之前的研究 byte -1就是代表FF颜色,3代表03色值,
    //这里存储的就是R=FF G=03 B=FF A=FF
    bitmap.copyPixelsFromBuffer(ByteBuffer.wrap(new byte[]{-1, 3, -1, -1}));

    // 验证bitmap转换为字节,再转换为16进制颜色形式
    ByteBuffer buffer = ByteBuffer.allocate(bitmap.getWidth() * bitmap.getHeight() * 4); //  使用allocate()静态方法创建字节缓冲区
    bitmap.copyPixelsToBuffer(buffer); // 将位图的像素复制到指定的缓冲区
    byte[] bytes = buffer.array();

    ByteBuffer buffer2 = ByteBuffer.allocate(bitmap.getWidth() * bitmap.getHeight() * 4);
    byte[] bytes2 = buffer2.array();
    System.arraycopy(bytes, 0, bytes2, 0, buffer2.array().length);


    // ARGB_8888 真实的存储顺序是 R-G-B-A
    Log.d(TAG, "R: " + Integer.toHexString(bytes2[0] & 0xff));
    Log.d(TAG, "G: " + Integer.toHexString(bytes2[1] & 0xff));
    Log.d(TAG, "B: " + Integer.toHexString(bytes2[2] & 0xff));
    Log.d(TAG, "A: " + Integer.toHexString(bytes2[3] & 0xff));
    return bitmap;
}

// 打印结果,和预测的一样,说明理解正确
D/TTTTTTTTTT: R: ff
D/TTTTTTTTTT: G: 3
D/TTTTTTTTTT: B: ff
D/TTTTTTTTTT: A: ff

研究五:字节数组的拷贝

先申请一个字节数组,再通过arraycopy方法拷贝过去

ByteBuffer buffer2 = ByteBuffer.allocate(bitmap.getWidth() * bitmap.getHeight() * 4); byte[] bytes2 = buffer2.array(); System.arraycopy(bytes, 0, bytes2, 0, buffer2.array().length);

研究六:修改bitmap的大小

fun imageScale(bitmap: Bitmap, dst_w: Int, dst_h: Int): Bitmap? {
    val src_w = bitmap.width
    val src_h = bitmap.height
    val scale_w = dst_w.toFloat() / src_w
    val scale_h = dst_h.toFloat() / src_h
    val matrix = Matrix()
    matrix.postScale(scale_w, scale_h)
    return Bitmap.createBitmap(
        bitmap, 0, 0, src_w, src_h, matrix,
        true
    )
}

研究七:相机回调字节数组的格式

public void onPreviewFrame(byte[] data, Camera camera)

理解: NV21==YCbCr_420_SP 属于YUV420sp范围

官方注释:

Camera.Parameters.setPreviewFormat(int) is never called, the default will be the YCbCr_420_SP (NV21) format.

默认是 YCbCr_420_SP 格式,也就是NV21,不过可以通过PreviewFormat修改格式

YUV420sp分为NV12和NV21

 参考资料:

1.你真的知道 ARGB_8888 格式图片的 A、R、G、B 每个通道的排列顺序吗?