人生苦短,不说废话,不做无用功      

      当我们在用android 开发视频录制时候,会遇到录制出来的视频 花屏、绿屏等现象,不用怀疑,90%是因为视频编码时候的颜色格式和编码器配置的编码格式不匹配。

相机预览的数据一般是两种 NV21和YV12,以下代码可以查出手机支持的预览格式:

List<Integer> previewFormats = mCamera.getParameters().getSupportedPreviewFormats();

手机MediaCodec编解码颜色格式一般为:

1、COLOR_FormatYUV420Planar
2、COLOR_FormatYUV420SemiPlanar

YUV420Planar支持的颜色格式一般也有两种:I420、YV12

YUV420SemiPlanne支持的颜色格式一般也有两种:NV12、NV21

对应关系如下:

I420: YYYYYYYY UU VV =>标准的YUV420P (COLOR_FormatYUV420Plannar编码格式)

YV12: YYYYYYYY VV UU =>属于YUV420P的一种
NV12: YYYYYYYY UVUV =>标准的YUV420SP (COLOR_FormatYUV420SemiPlannar编码格式)
NV21: YYYYYYYY VUVU =>属于YUV420SP的一种

 

用Camera摄像头预览的时候:

这里第一个参数data的数据格式,如果没有特殊配置,android默认返回的是NV21格式,查看源码可以看到:

所以如果相机支持YUV420SemiPlanar编码格式,那么NV21数据时没问题的,如果不支持那么需要将NV21的数据格式转为标准的YUV420Planar所支持的

例如:

 public final static int NV21_TO_yuv420P(byte[] dst, byte[] src, int w, int h){
        int ysize = w * h;
        int usize = w * h * 1 / 4;

        byte[] dsttmp = dst;

        // y
        System.arraycopy(src, 0, dst, 0, ysize);

        // u, 1/4
        int srcPointer = ysize;
        int dstPointer = ysize;
        int count = usize;
        while (count > 0)
        {
            srcPointer++;
            dst[dstPointer] = src[srcPointer];
            dstPointer++;
            srcPointer++;
            count--;
        }

        // v, 1/4
        srcPointer = ysize;

        count = usize;
        while (count > 0)
        {
            dst[dstPointer] = src[srcPointer];
            dstPointer++;
            srcPointer += 2;
            count--;
        }

        dst = dsttmp;

        // _EF_TIME_DEBUG_END(0x000414141);

        return 0;
    }

  同时设置mediaCodec的颜色编码格式:

 

mediaFormat_camera.setInteger(MediaFormat.KEY_COLOR_FORMAT,mColorFormat);
//mColorFormat可以是 MediaCodecInfo.CodecCapabilities.COLOR_FormatYUV420Planar 或者 MediaCodecInfo.CodecCapabilities.COLOR_FormatYUV420SemiPlanar

 

  

 

    如果相机支持YUV420SemiPlanar编码格式,那么NV21需要转为NV12 否则会报错

 

    public byte[] nv21ToNv12(byte[] nv21) {
        if (nv21 == null) return null;
        int size = nv21.length;
        byte[] nv12 = new byte[size];
        int yLen = (int) (size * 2.0 / 3.0);
        System.arraycopy(nv21, 0, nv12, 0, yLen);
        int i = yLen;
        while (i < size - 1) {
            nv12[i + 1] = nv21[i];
            nv12[i] = nv21[i + 1];
            i += 2;
        }
        return nv12;
    } 

 

 

 

 

 

 

 

 

posted on 2020-12-03 21:04  毕哥  阅读(2771)  评论(1编辑  收藏  举报