zoukankan      html  css  js  c++  java
  • YUV图像格式

    YUV
    YUV

      做视频采集与处理,自然少不了要学会分析YUV数据。因为从采集的角度来说,一般的视频采集芯片输出的码流一般都是YUV数据流的形式,而从视频处理(例如H.264、MPEG视频编解码)的角度来说,也是在原始YUV码流进行编码和解析,所以,了解如何分析YUV数据流对于做视频领域的人而言,至关重要。YUV是指亮度参量和色度参量分开表示的像素格式,而这样分开的好处就是不但可以避免相互干扰,还可以降低色度的采样率而不会对图像质量影响太大。

          人眼对色度的敏感程度要低于对亮度的敏感程度。

          YUV,分为三个分量,“Y”表示明亮度(Luminance或Luma),也就是灰度值;而“U”和“V” 表示的则是色度(Chrominance或Chroma),作用是描述影像色彩及饱和度,用于指定像素的颜色。与我们熟知的RGB类似,YUV也是一种颜色编码方法,主要用于电视系统以及模拟视频领域,它将亮度信息(Y)与色彩信息(UV)分离,没有UV信息一样可以显示完整的图像,只不过是黑白的,这样的设计很好地解决了彩色电视机与黑白电视的兼容问题。并且,YUV不像RGB那样要求三个独立的视频信号同时传输,所以用YUV方式传送占用极少的频宽。

          YUV码流有多种不同的格式,要分析YUV码流,就必须搞清楚你面对的到底是哪一种格式,并且必须搞清楚这种格式的YUV采样和分布情况。

      YUV格式有两大类:planar和packed。
      对于planar的YUV格式,先连续存储所有像素点的Y,紧接着存储所有像素点的U,随后是所有像素点的V。
      对于packed的YUV格式,每个像素点的Y,U,V是连续交叉存储的。

    1.  采样方式   

        YUV码流的存储格式其实与其采样的方式密切相关,主流的采样方式有三种,YUV4:4:4,YUV4:2:2,YUV4:2:0,如何根据其采样格式来从码流中还原每个像素点的YUV值,因为只有正确地还原了每个像素点的YUV值,才能通过YUV与RGB的转换公式提取出每个像素点的RGB值,然后显示出来。

        用三个图来直观地表示采集的方式吧,以黑点表示采样该像素点的Y分量,以空心圆圈表示采用该像素点的UV分量。

    先记住下面这段话,以后提取每个像素的YUV分量会用到。

        YUV 4:4:4采样,每一个Y对应一组UV分量,每像素32位

        YUV 4:2:2采样,每两个Y共用一组UV分量,每像素16位

        YUV 4:2:0采样,每四个Y共用一组UV分量,每像素16位

    平常所讲的YUV A:B:C的意思一般是指基于4个象素来讲,其中Y采样了A次,U采样了B次,V采样了C次.

    YUV 格式可以分为打包格式packed format和平面格式planar format。打包格式将YUV分量存放在同一个数组中,通常是几个相邻的像素组成一个宏像素(macro-pixel);而平面格使用三个数组分开存放YUV三个分量,就像是一个三维平面一样。Packed format和planner format的区别在于,packed format中的YUV是混合在一起的,因此就有了UYVY、YUYV等等,他们在码流中排列的方式有所不同。而对于planner format每一个Y分量,U分量和V分量都是以独立的平面组织的,也就是说所有的U分量都在Y分量之后出现,而V分量在所有的U分量之后。就像三个大色块一样。

    2.  存储方式

        下面用图的形式给出常见的YUV码流的存储方式,并在存储方式后面附有取样每个像素点的YUV数据的方法,其中,Cb、Cr的含义等同于U、V。因为我们在实验中芷使用到YUV422的格式,这里只介绍这个,其他的可以去其他博文了找。

    (1) YUYV 格式 (属于YUV422)

    YUYV(YUY2)为YUV422采样的存储格式中的一种,相邻的两个Y共用其相邻的两个Cb(U)、Cr(V),分析,对于像素点Y'00、Y'01 而言,其Cb、Cr的值均为 Cb00、Cr00,其他的像素点的YUV取值依次类推。YVYU(YVY2)也一样,只是UV的位置调换了一下,先V后U。

    (2) UYVY 格式 (属于YUV422)

     UYVY格式也是YUV422采样的存储格式中的一种,只不过与YUYV不同的是UV的排列顺序不一样而已,还原其每个像素点的YUV值的方法与上面一样。

     

    (3) YUV422P(属于YUV422)

    YUV422P也属于YUV422的一种,它是一种Plane模式,即打包模式,并不是将YUV数据交错存储,而是先存放所有的Y分量,然后存储所有的U(Cb)分量,最后存储所有的V(Cr)分量,如上图所示。其每一个像素点的YUV值提取方法也是遵循YUV422格式的最基本提取方法,即两个Y共用一个UV。比如,对于像素点Y'00、Y'01 而言,其Cb、Cr的值均为 Cb00、Cr00。

    以YUV420 planar数据为例, 以720×480大小图象YUV420 planar为例,

    其存储格式是: 共大小为(720×480×3>>1)字节,

    分为三个部分:Y,U和V

    Y分量:    (720×480)个字节  

    U(Cb)分量:(720×480>>2)个字节

    V(Cr)分量:(720×480>>2)个字节

    三个部分内部均是行优先存储,三个部分之间是Y,U,V 顺序存储。

    即YUV数据的0--720×480字节是Y分量值,         

    720×480--720×480×5/4字节是U分量    

    720×480×5/4 --720×480×3/2字节是V分量。

    这里Y分量其实就是我们常说的灰度值,所以需要对图片进行灰度处理的话,可以直接提取出图片的Y分量。这点对我们后面的处理很重要。

    YUV4:4:4  

    下面的四个像素为: [Y0 U0 V0] [Y1 U1 V1] [Y2 U2 V2] [Y3 U3 V3]

    存放的码流为: Y0 U0 V0 Y1 U1 V1 Y2 U2 V2 Y3 U3 V3

    映射出像素点保持原样

    YUV4:2:2

    每个色差信道的抽样率是亮度信道的一半,所以水平方向的色度抽样率只是4:4:4的一半。对非压缩的8比特量化的图像来说,每个由两个水平方向相邻的像素组成的宏像素需要占用4字节内存。  
    下面的四个像素为:[Y0 U0 V0] [Y1 U1 V1] [Y2 U2 V2] [Y3 U3 V3]

    存放的码流为:Y0 U0 Y1 V1 Y2 U2 Y3 V3

    映射出像素点为:[Y0 U0 V1] [Y1 U0 V1] [Y2 U2 V3] [Y3 U2 V3]

    YUV 4:1:1

    4:1:1的色度抽样,是在水平方向上对色度进行4:1抽样。对于低端用户和消费类产品这仍然是可以接受的。对非压缩的8比特量化的视频来说,每个由4个水平方向相邻的像素组成的宏像素需要占用6字节内存。  
    原来四个像素为: [Y0 U0 V0] [Y1 U1 V1] [Y2 U2 V2] [Y3 U3 V3]  

    存放的码流为: Y0 U0 Y1 Y2 V2 Y3

    还原出像素点为:[Y0 U0 V2] [Y1 U0 V2] [Y2 U0 V2] [Y3 U0 V2]

    用6个YUV分量描述了原来的12个YUV分量,因此压缩比为1/2,平均来讲,就是用了12bit表示了一个象素点,原来YUV(8bit*3)是24bit。

     

    YUV4:2:0
    4:2:0并不意味着只有Y,Cb而没有Cr分量。它指得是对每行扫描线来说,只有一种色度分量以2:1的抽样率存储。相邻的扫描行存储不同的色度分 量,也就是说,如果一行是4:2:0的话,下一行就是4:0:2,再下一行是4:2:0...以此类推。对每个色度分量来说,水平方向和竖直方向的抽样率 都是2:1,所以可以说色度的抽样率是4:1。对非压缩的8比特量化的视频来说,每个由2x2个2行2列相邻的像素组成的宏像素需要占用6字节内存。  
    下面八个像素为:[Y0 U0 V0] [Y1 U1 V1] [Y2 U2 V2] [Y3 U3 V3] [Y5 U5 V5] [Y6 U6 V6] [Y7U7 V7] [Y8 U8 V8]
    存放的码流为:Y0 U0 Y1 Y2 U2 Y3 Y5 V5 Y6 Y7 V7 Y8
    映射出的像素点为:[Y0 U0 V5] [Y1 U0 V5] [Y2 U2 V7] [Y3 U2 V7] [Y5 U0 V5] [Y6 U0 V5] [Y7U2 V7] [Y8 U2 V7]

     
    RGB

      计算机彩色显示器显示色彩的原理与彩色电视机一样,都是采用R(Red)、G(Green)、B(Blue)相加混色的原理:通过发射出三种不同强度的电子束,使屏幕内侧覆盖的红、绿、蓝磷光材料发光而产生色彩。这种色彩的表示方法称为RGB色彩空间表示(它也是多媒体计算机技术中用得最 多的一种色彩空间表示方法)。根据色度学的介绍,不同波长的单色光会引起不同的彩色感觉,但相同的彩色感觉却可以来源于不同的光谱成分组合。自然界中几乎所有的颜色都能用三种基本彩色混合配出,在彩色电视技术中选择红色、绿色、和蓝色作为三基色。其他的颜色都可以用红色、绿色和蓝色按照不同的比例混合而成。所选取的红色、绿色和蓝色三基色空间。简称为RGB颜色空间。

    RGB565    每个像素用16位表示,RGB分量分别使用5位、6位、5位

    RGB555    每个像素用16位表示,RGB分量都使用5位(剩下1位不用)

    RGB24    每个像素用24位表示,RGB分量各使用8位

    RGB32    每个像素用32位表示,RGB分量各使用8位(剩下8位不用)

    ARGB32    每个像素用32位表示,RGB分量各使用8位(剩下的8位用于表示Alpha通道值)

    RGB565(我们使用的格式)

    使用16位表示一个像素,这16位中的5位用于R,6位用于G,5位用于B。

    程序中通常使用一个字(WORD,一个字等于两个字节)来操作一个像素。当读出一个像素后,这个字的各个位意义如下:      

    高字节              低字节

    R R R R R G G G     G G G B B B B B

    可以组合使用屏蔽字和移位操作来得到RGB各分量的值:  

    #define RGB565_MASK_RED    0xF800

    #define RGB565_MASK_GREEN  0x07E0

    #define RGB565_MASK_BLUE   0x001F

    R = (wPixel & RGB565_MASK_RED) >> 11;   // 取值范围0-31

    G = (wPixel & RGB565_MASK_GREEN) >> 5;  // 取值范围0-63

    B =  wPixel & RGB565_MASK_BLUE;         // 取值范围0-31

    #define RGB(r,g,b) (unsigned int)( (r|0x08 << 11) | (g|0x08 << 6) | b|0x08 )

    #define RGB(r,g,b) (unsigned int)( (r|0x08 << 10) | (g|0x08 << 5) | b|0x08 )

    该代码可以解决24位与16位相互转换的问题

    RGB555

    是另一种16位的RGB格式,RGB分量都用5位表示(剩下的1位不用)。

    使用一个字读出一个像素后,这个字的各个位意义如下:     

     高字节             低字节

    X R R R R G G       G G G B B B B B       (X表示不用,可以忽略)  

    RGB24使用24位来表示一个像素,RGB分量都用8位表示,取值范围为0-255

    RGB32使用32位来表示一个像素,RGB分量各用去8位,剩下的8位不用
    RGB24

    RGB24使用24位来表示一个像素,RGB分量都用8位表示,取值范围为0-255。注意在内存中RGB各分量的排列顺序为:BGR BGR BGR…。通常可以使用RGBTRIPLE数据结构来操作一个像素,它的定义为:

    typedef struct tagRGBTRIPLE {

    BYTE rgbtBlue; // 蓝色分量

    BYTE rgbtGreen; // 绿色分量

    BYTE rgbtRed; // 红色分量

    } RGBTRIPLE;
    RGB32

    RGB32使用32位来表示一个像素,RGB分量各用去8位,剩下的8位用作Alpha通道或者不用。(ARGB32就是带Alpha通道的RGB24。)注意在内存中RGB各分量的排列顺序为:BGRA BGRA BGRA…。通常可以使用RGBQUAD数据结构来操作一个像素,它的定义为:

    typedef struct tagRGBQUAD {

    BYTE rgbBlue; // 蓝色分量

    BYTE rgbGreen; // 绿色分量

    BYTE rgbRed; // 红色分量

    BYTE rgbReserved; // 保留字节(用作Alpha通道或忽略)

    } RGBQUAD。


    使用YUV的优点有两个:

    一.YUV主要用于优化彩色视频信号的传输,使其向后兼容老式黑白电视,这一特性用在于电视信号上。

    二.YUV是数据总尺寸小于RGB格式(但用YUV444的话,和RGB888一样都是24bits)


    YUV细分的话有Y'UV,YUV,YCbCr,YPbPr等类型,其中YCbCr主要用于数字信号.

    YCbCr 是在世界数字组织视频标准研制过程中作为ITU - R BT1601 建议的一部分, 其实是YUV经过Gamma的翻版。其中Y与YUV 中的Y含义一致, Cb , Cr 同样都指色彩, 只是在表示方法上不同而已。在YUV 家族中, YCbCr 是在计算机系统中应用最多的成员, 其应用领域很广泛,JPEG、MPEG,H264均采用此格式。

    其中,Cr反映了RGB输入信号红色部分与RGB信号亮度值之间的差异,而Cb反映的是RGB输入信号蓝色部分与RGB信号亮度值之间的差异,此即所谓的色差信号

    一般所讲的YUV大多是指YCbCr。以下用YUV指代YCbCr。

    所谓采样格式,本质便是在每一个2x2的像素块中,选取几个Y分量,几个U分量,几个V分量来表示该像素块,不同的选取方式便是不同的采样格式喽。

    YUV444:每一个像素用Y, U, V共同表示
    YUV422:Y每个像素采样,U,V每两个像素采一次样,第一个像素的U,V的值用在第0个像素处采样的U0,VO的值;第三个像素的U,V的值用在第2个元素处采样的U1,V1的值。
    YUV420:对于每个2x2像素块,对于每个像素Y值采样,整个2*2块中分别对U和V只做一次采样.

  • 相关阅读:
    vue动态设置页面title方法
    laravel 只有/login路由403,如何解决
    vue-cli中使用rem,vue自适应
    Vuex的全面用法总结
    template or render function not defined.
    Laravel Mix编译前端资源
    laravel学习:模块化caffeinated
    laravel学习:php写一个简单的ioc服务管理容器
    Aspose Cells dll 实现数据简单下载
    sessionStorage 的使用
  • 原文地址:https://www.cnblogs.com/liuzhiwei0925/p/12082751.html
Copyright © 2011-2022 走看看