使用ffmpeg进行图像格式转换以及图像缩放/sws_scale/linux/c++/c/rgb-yuv420

本文介绍如何使用FFmpeg的swscale库进行图像格式转换及缩放操作。重点介绍了sws_getContext、sws_scale和sws_freeContext三个关键函数,并通过两个示例代码详细展示了RGB到RGB、RGB到YUV420P的具体实现。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

  利用ffmpeg进行图像数据格式的转换以及图片的缩放应用中,主要用到了swscale.h文件中的三个函数,分别是:

      struct SwsContext *sws_getContext(int srcW, int srcH, enum AVPixelFormat srcFormat,
                               int dstW, int dstH, enum AVPixelFormat dstFormat,
                               int flags, SwsFilter *srcFilter,
                               SwsFilter *dstFilter, const double *param);

      int sws_scale(struct SwsContext *c, const uint8_t *const srcSlice[],

                     const int srcStride[], int srcSliceY, int srcSliceH,
                   uint8_t *const dst[], const int dstStride[]);

      void sws_freeContext(struct SwsContext *swsContext);

  sws_getContext函数可以看做是初始化函数,它的参数定义分别为:

      int srcW,int srcH 为原始图像数据的高和宽;

      int dstW,int dstH 为输出图像数据的高和宽;

      enum AVPixelFormat srcFormat 为输入和输出图片数据的类型;eg:AV_PIX_FMT_YUV420、PAV_PIX_FMT_RGB24;

      int flags 为scale算法种类;eg:SWS_BICUBIC、SWS_BICUBLIN、SWS_POINT、SWS_SINC;

      SwsFilter *srcFilter ,SwsFilter *dstFilter,const double *param 可以不用管,全为NULL即可;

  sws_scale函数则为执行函数,它的参数定义分别为:

      struct SwsContext *c 为sws_getContext函数返回的值;

      const uint8_t *const srcSlice[],uint8_t *const dst[] 为输入输出图像数据各颜色通道的buffer指针数组;

      const int srcStride[],const int dstStride[] 为输入输出图像数据各颜色通道每行存储的字节数数组;     

      int srcSliceY 为从输入图像数据的第多少列开始逐行扫描,通常设为0;

      int srcSliceH 为需要扫描多少行,通常为输入图像数据的高度;

  sws_freeContext函数为结束函数,它的参数即为sws_getContext函数返回的值;

  示例代码:rgb24_2_rgb24/1600*1200—352*288

//.h

#ifndef INT64_C
#define INT64_C
#define UINT64_C
#endif

extern "C"
{
#include "libswscale/swscale.h"
}
struct ImgInfo
{
    unsigned int height;
    unsigned int width;
    unsigned long bufferSize;
    unsigned char *bufferPtr;
}rawDate;

//.cpp

    int nSrcH = rawData.height;  //1200
    int nSrcW = rawData.width;   //1600
		
    int nDstH = RESIZED_HIGHT;   //258
    int nDstW = RESIZED_WIDTH;   //352
		
    uint8_t *pSrcBuff[3] = {rawData.bufferPtr, rawData.bufferPtr  + nSrcW * nSrcH, rawData.bufferPtr  + nSrcW * nSrcH * 2};
    uint8_t *pDstBuff[3] = {pResizedData->bufferPtr, pResizedData->bufferPtr + nDstW * nDstH, pResizedData->bufferPtr + nDstW * nDstH * 2};
       
    int nSrcStride[3];
    int nDstStride[3];
    
    for (int i=0; i<3; i++)
    {
    	nSrcStride[i] = nSrcW * 3;
    	nDstStride[i] = nDstW * 3;
    }
    
    SwsContext*	m_pSwsContext;
    
    m_pSwsContext = sws_getContext(nSrcW, nSrcH, AV_PIX_FMT_RGB24,
                                  nDstW, nDstH, AV_PIX_FMT_RGB24,
                                  SWS_SINC, 
                                  NULL, NULL, NULL);
                                  
    if (NULL == m_pSwsContext) 
    {
    	printf("ffmpeg get context error!\n");
    	return false;
    }
    
    sws_scale(m_pSwsContext, pSrcBuff,
              nSrcStride, 0, nSrcH,
              pDstBuff, nDstStride);
              
    sws_freeContext(m_pSwsContext);

示例代码:rgb24_2_yuv420/1600*1200—352*288

//.cpp

    int nSrcH = rawData->height;   //1200
    int nSrcW = rawData->width;    //1600
		
    int nDstH = RESIZED_HIGHT;    //288
    int nDstW = RESIZED_WIDTH;    //352
		
    uint8_t *pSrcBuff[3] = {rawData->bufferPtr, rawData->bufferPtr  + nSrcW * nSrcH, rawData->bufferPtr  + nSrcW * nSrcH * 2};
    uint8_t *pDstBuff[3] = {resizedYuvData->bufferPtr, resizedYuvData->bufferPtr + nDstW * nDstH, resizedYuvData->bufferPtr + nDstW * nDstH * 5 / 4};
       
    int nSrcStride[3];
    int nDstStride[3];
    
    for (int i=0; i<3; i++)
    {
    	nSrcStride[i] = nSrcW * 3;
    }
    
    nDstStride[0] = nDstW;
    nDstStride[1] = nDstW / 2;
    nDstStride[2] = nDstW / 2;
    
    SwsContext*	m_pSwsContext;
    
    m_pSwsContext = sws_getContext(nSrcW, nSrcH, AV_PIX_FMT_RGB24,
                                  nDstW, nDstH, AV_PIX_FMT_YUV420P,
                                  SWS_SINC, 
                                  NULL, NULL, NULL);
                                  
    if (NULL == m_pSwsContext) 
    {
    	printf("ffmpeg get context error!\n");
    	return false;
    }
    
    sws_scale(m_pSwsContext, pSrcBuff,
              nSrcStride, 0, nSrcH,
              pDstBuff, nDstStride);
              
    sws_freeContext(m_pSwsContext);	




### 图像处理中的YUVRGB转换图像处理领域,颜色空间的转换是一项常见的操作。特别是从YUVRGB的颜色空间转换,在视频编码、解码以及显示过程中扮演着重要角色。 #### 方法一:基于公式的逐像素计算 最基础的方法是依据标准定义的线性方程组来完成这种转变。对于每个单独的像素点而言,会应用如下所示的一套公式来进行色彩分量间的映射[^1]: \[ R = Y + 1.402(V - 128) \] \[ G = Y - 0.344136(U - 128) - 0.714136(V - 128) \] \[ B = Y + 1.772(U - 128) \] 这种方法虽然直观易懂,但在实际应用场景下可能因为性能原因而不被优先考虑。当面对大规模的数据集或是实时性的需求时,简单的循环遍历每一个像素并执行上述运算可能会显得效率低下。 为了提高这一过程的速度,开发者们探索出了多种优化策略,比如利用SIMD指令集加速矩阵乘法的操作,或者是采用查表法减少重复计算等手段。 #### 方法二:借助第三方库实现高效转换 除了手动编写代码外,还可以依赖成熟的多媒体框架简化开发流程的同时获得更好的运行表现。例如FFmpeg就是一个广泛使用的开源项目,它内部封装了一系列高效的算法用于不同格式间相互转化的任务。 具体来说,在C/C++环境中可以通过调用`libswscale`库里的API——即`sws_scale()`函数轻松达成目的。此接口允许指定源图象(通常是已经过解码得到的YUV数据)及其属性参数,并指明期望获取的目标格式(此处为RGB),从而自动完成整个变换工作流[^2]。 下面给出了一段示范性质的小程序片段展示如何运用该功能模块进行一次完整的由YUVRGB的空间跃迁: ```c #include <libavutil/imgutils.h> #include <libswscale/swscale.h> // 初始化缩放器上下文对象 struct SwsContext *sws_ctx; int width, height; // 输入图片尺寸 uint8_t* yuv_data[AV_NUM_DATA_POINTERS]; // 存储输入yuv平面数组 int linesize_yuv[AV_NUM_DATA_POINTERS]; enum AVPixelFormat src_pix_fmt = AV_PIX_FMT_YUV420P; // 创建输出缓冲区保存rgb结果 uint8_t* rgb_buffer; int line_size_rgb; enum AVPixelFormat dst_pix_fmt = AV_PIX_FMT_RGB24; // 设置Swscale配置项... sws_ctx = sws_getContext(width, height, src_pix_fmt, width, height, dst_pix_fmt, SWS_BILINEAR, NULL, NULL, NULL); if (!sws_ctx){ fprintf(stderr,"Could not initialize the conversion context!\n"); } // 执行转换操作 sws_scale(sws_ctx, (const uint8_t* const*)yuv_data, linesize_yuv, 0, height, &rgb_buffer, &line_size_rgb); ``` 这段示例展示了创建一个软件尺度调整(Software Scaling Context),设置好必要的参数之后就可以直接调用`sws_scale`来进行批量化的像素级转换了。需要注意的是这里假设输入是一张遵循YUV420P采样模式的标准画面;而对于其他类型的媒体文件,则需相应修改读取方式及初始化部分的相关设定。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值