如何将 android 位图转换为 NV12 颜色格式?

Posted

技术标签:

【中文标题】如何将 android 位图转换为 NV12 颜色格式?【英文标题】:How can I convert android bitmap to NV12 color format? 【发布时间】:2013-08-01 18:04:28 【问题描述】:

我正在编写一些将android位图转换为NV12格式的代码。

我从 android 位图中找到了可以为我提供 NV21 的代码,并且该代码似乎有效。 (Convert bitmap array to YUV (YCbCr NV21))

我发现的唯一区别是根据参考在 NV12 和 NV21 之间切换 U 和 V 字节。 (http://www.fourcc.org/yuv.php)

于是我把原代码中U和V的位置改了,结果如下。

byte [] getNV12(int inputWidth, int inputHeight, Bitmap scaled) 
    // Reference (Variation) : https://gist.github.com/wobbals/5725412

    int [] argb = new int[inputWidth * inputHeight];

    //Log.i(TAG, "scaled : " + scaled);
    scaled.getPixels(argb, 0, inputWidth, 0, 0, inputWidth, inputHeight);

    byte [] yuv = new byte[inputWidth*inputHeight*3/2];
    encodeYUV420SP(yuv, argb, inputWidth, inputHeight);

    scaled.recycle();

    return yuv;


void encodeYUV420SP(byte[] yuv420sp, int[] argb, int width, int height) 
    final int frameSize = width * height;

    int yIndex = 0;
    int uvIndex = frameSize;

    int a, R, G, B, Y, U, V;
    int index = 0;
    for (int j = 0; j < height; j++) 
        for (int i = 0; i < width; i++) 

            a = (argb[index] & 0xff000000) >> 24; // a is not used obviously
        R = (argb[index] & 0xff0000) >> 16;
            G = (argb[index] & 0xff00) >> 8;
            B = (argb[index] & 0xff) >> 0;

        // well known RGB to YUV algorithm
        Y = ( (  66 * R + 129 * G +  25 * B + 128) >> 8) +  16;
        V = ( ( -38 * R -  74 * G + 112 * B + 128) >> 8) + 128; // Previously U
        U = ( ( 112 * R -  94 * G -  18 * B + 128) >> 8) + 128; // Previously V

        yuv420sp[yIndex++] = (byte) ((Y < 0) ? 0 : ((Y > 255) ? 255 : Y));
        if (j % 2 == 0 && index % 2 == 0)  
            yuv420sp[uvIndex++] = (byte)((V<0) ? 0 : ((V > 255) ? 255 : V));
            yuv420sp[uvIndex++] = (byte)((U<0) ? 0 : ((U > 255) ? 255 : U));
        

        index ++;
        
    

我在转换图像时错了吗? (我很确定编码器没有问题。)

破图截图:https://www.dropbox.com/s/vho14831fgnh1kl/Thu%20Aug%2001%2008_56_14%20GMT%2B09_00%202013%20%281%29.mp4_000002000.jpg

【问题讨论】:

【参考方案1】:

替换

a = (argb[index] & 0xff000000) >> 24; // a is not used obviously
 R = (argb[index] & 0xff0000) >> 16;
 G = (argb[index] & 0xff00) >> 8;
 B = (argb[index] & 0xff) >> 0;

与,

R = (argb[index] & 0xff000000) >>> 24;
G = (argb[index] & 0xff0000) >> 16;
B = (argb[index] & 0xff00) >> 8;

【讨论】:

【参考方案2】:

你可以使用

R = Color.red(argb[index]);
G = Color.green(argb[index]);
B = Color.blue(argb[index]);

您的其他代码运行良好。

【讨论】:

【参考方案3】:

我使用相同的代码从相机图像中使用 mediaEncoder 创建视频。生成的视频中存在颜色问题(example images look here)。似乎 mediaEncoder 支持的格式很少(Ref - Q5)

所以只好转成N12(或者I420格式),我根据wikipedia article修改了上面的代码这样

int uIndex = frameSize;
int vIndex = frameSize + frameSize/4;

....
     yuv420sp[uIndex++] = (byte)((U<0) ? 0 : ((U > 255) ? 255 : U));
     yuv420sp[vIndex++] = (byte)((V<0) ? 0 : ((V > 255) ? 255 : V));
...

现在生成的视频似乎工作正常。

【讨论】:

以上是关于如何将 android 位图转换为 NV12 颜色格式?的主要内容,如果未能解决你的问题,请参考以下文章

使用FFMPEG从NV12原始数据转换为H264时出现颜色不匹配

MLKit Firebase android - 如何将 FirebaseVisionFace 转换为图像对象(如位图)?

如何将位图照片转换为十六进制颜色代码?

如何使用 IPP 将 8 位灰度图像转换为 NV12(有限范围)色彩空间

Android:如何将整个 ImageView 转换为位图?

如何在android中更改位图图像颜色?