4

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

我从 android 位图中找到了给我 NV21 的代码,而且似乎该代码有效。(将位图数组转换为 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

4

3 回答 3

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;
于 2013-12-20T07:03:24.437 回答
1

你可以使用

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

您的其余代码效果很好。

于 2014-10-02T22:08:40.590 回答
0

我使用相同的代码从相机图像中使用 mediaEncoder 创建视频。生成的视频中存在颜色问题(例如图片看这里)。似乎 mediaEncoder 支持的格式很少(参考- Q5)

所以不得不将其转换为N12(或I420格式),我根据维基百科文章修改了上面的代码

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));
...

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

于 2021-04-01T09:55:10.970 回答