FFmpeg教程:libswscale对图像进行简单处理

我们在FFmpeg简单总结对FFmpeg 组成模块,编码进行了简单介绍。

FFmpeg组成部分:
libavcodec: 提供了音视频编解码器的库。
libavformat: 处理多媒体容器格式的库,包括封装和解封装。
libavutil: 包含一些公共的实用工具函数。
libswscale: 提供图像缩放和颜色转换功能的库。
libavfilter: 实现音视频过滤器的库,用于进行各种音视频处理操作。
ffmpeg: 命令行工具,用于进行音视频处理和转码。
ffprobe: 用于检测多媒体文件信息的命令行工具。
ffplay: 简单的播放器,支持音视频播放。

libswscale

libswscale(Software Scaling and Conversion Library)是FFmpeg中的一个库,用于执行图像缩放、颜色空间转换以及图像格式转换等操作。它主要提供了一些函数和工具,使得在视频处理过程中可以方便地进行图像大小和颜色空间的调整。

常用的API介绍

sws_getContext: 创建并返回一个SwsContext对象,用于设置缩放和颜色空间转换的参数。

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

sws_scale: 将输入图像进行缩放和颜色空间转换,并将结果输出到目标图像。

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[]);

sws_freeContext: 释放SwsContext对象占用的资源。

void sws_freeContext(struct SwsContext *swsContext);

灰度化

下面给出一个例子,使用FFmpeg对图像进行处理,该程序解码视频流,选取视频流中的输入图片,进行灰度化处理,然后保存为新的图片

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <libavcodec/avcodec.h>
#include <libavformat/avformat.h>
#include <libswscale/swscale.h>void process_image(const char *input_filename, const char *output_filename) {AVFormatContext *input_format_context = NULL;AVCodecContext *codec_context = NULL;AVCodec *codec = NULL;AVFrame *frame = NULL;AVFrame *gray_frame = NULL;AVPacket packet;struct SwsContext *sws_context = NULL;// Register FFmpeg componentsav_register_all();// Open input fileif (avformat_open_input(&input_format_context, input_filename, NULL, NULL) != 0) {fprintf(stderr, "Error opening input file\n");exit(EXIT_FAILURE);}// Retrieve stream informationif (avformat_find_stream_info(input_format_context, NULL) < 0) {fprintf(stderr, "Error finding stream information\n");exit(EXIT_FAILURE);}// Find the first video streamint video_stream_index = -1;for (int i = 0; i < input_format_context->nb_streams; i++) {if (input_format_context->streams[i]->codecpar->codec_type == AVMEDIA_TYPE_VIDEO) {video_stream_index = i;break;}}if (video_stream_index == -1) {fprintf(stderr, "Error: No video stream found\n");exit(EXIT_FAILURE);}// Find the decoder for the video streamcodec = avcodec_find_decoder(input_format_context->streams[video_stream_index]->codecpar->codec_id);if (!codec) {fprintf(stderr, "Error: No decoder found\n");exit(EXIT_FAILURE);}// Allocate a codec context for the decodercodec_context = avcodec_alloc_context3(codec);if (!codec_context) {fprintf(stderr, "Error allocating codec context\n");exit(EXIT_FAILURE);}// Copy codec parameters from input stream to codec contextif (avcodec_parameters_to_context(codec_context, input_format_context->streams[video_stream_index]->codecpar) < 0) {fprintf(stderr, "Error setting codec parameters\n");exit(EXIT_FAILURE);}// Open codecif (avcodec_open2(codec_context, codec, NULL) < 0) {fprintf(stderr, "Error opening codec\n");exit(EXIT_FAILURE);}// Allocate video framesframe = av_frame_alloc();gray_frame = av_frame_alloc();if (!frame || !gray_frame) {fprintf(stderr, "Error allocating frames\n");exit(EXIT_FAILURE);}// Determine required buffer size and allocate bufferint num_bytes = av_image_get_buffer_size(AV_PIX_FMT_GRAY8, codec_context->width, codec_context->height, 1);uint8_t *buffer = (uint8_t *)av_malloc(num_bytes * sizeof(uint8_t));// Assign appropriate parts of buffer to image planes in gray_frameav_image_fill_arrays(gray_frame->data, gray_frame->linesize, buffer, AV_PIX_FMT_GRAY8, codec_context->width, codec_context->height, 1);// Initialize SwsContext for converting between color spacessws_context = sws_getContext(codec_context->width, codec_context->height, codec_context->pix_fmt,codec_context->width, codec_context->height, AV_PIX_FMT_GRAY8,SWS_BILINEAR, NULL, NULL, NULL);// Open output fileFILE *output_file = fopen(output_filename, "wb");if (!output_file) {fprintf(stderr, "Error opening output file\n");exit(EXIT_FAILURE);}// Read frames from the input file, convert to grayscale, and write to the output filewhile (av_read_frame(input_format_context, &packet) == 0) {if (packet.stream_index == video_stream_index) {// Decode video frameif (avcodec_receive_frame(codec_context, frame) == 0) {// Convert frame to grayscalesws_scale(sws_context, frame->data, frame->linesize, 0, frame->height, gray_frame->data, gray_frame->linesize);// Write grayscale frame to output filefwrite(gray_frame->data[0], 1, num_bytes, output_file);}}av_packet_unref(&packet);}// Cleanupfclose(output_file);av_frame_free(&frame);av_frame_free(&gray_frame);av_free(buffer);sws_freeContext(sws_context);avcodec_close(codec_context);avformat_close_input(&input_format_context);
}int main() {const char *input_filename = "input.jpg";const char *output_filename = "output.gray";process_image(input_filename, output_filename);return 0;
}

放大缩小

也可以直接用接口函数,对image操作


#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <libswscale/swscale.h>int main() {// Input image parametersint src_width = 1920;int src_height = 1080;enum AVPixelFormat src_format = AV_PIX_FMT_RGB24;// Output image parametersint dst_width = 640;int dst_height = 360;enum AVPixelFormat dst_format = AV_PIX_FMT_YUV420P;// Allocate source and destination buffersuint8_t *src_data[4];int src_linesize[4];uint8_t *dst_data[4];int dst_linesize[4];// Initialize SwsContextstruct SwsContext *sws_context = sws_getContext(src_width, src_height, src_format,dst_width, dst_height, dst_format,SWS_BILINEAR, NULL, NULL, NULL);// Perform scaling and color space conversionsws_scale(sws_context, src_data, src_linesize, 0, src_height,dst_data, dst_linesize);// Free resourcessws_freeContext(sws_context);return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/641977.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

MySQL 8.3 发布,具体有哪些新增和删减?

MySQL 8.3 主要更新&#xff1a;用于标记事务分组的 GTID、JSON EXPLAIN 格式增强、一些功能删除等。 MySQL 是一款广泛使用的开源的关系型数据库管理系统&#xff0c;已推出其最新版本 MySQL 8.3。它带来了新功能和一些删除&#xff0c;有望简化数据库操作。让我们来看看有哪些…

基本的 Socket 模型

什么是Socket Socket 的中文名叫作插口&#xff0c;咋一看还挺迷惑的。事实上&#xff0c;双方要进行网络通信前&#xff0c;各自得创建一个 Socket&#xff0c;这相当于客户端和服务器都开了一个“口子”&#xff0c;双方读取和发送数据的时候&#xff0c;都通过这个“口子”…

用通俗易懂的方式讲解:太棒了!构建大模型 Advanced RAG(检索增强生成)的速查表和实战技巧最全总结来了!

新的一年开始了&#xff0c;也许您正打算通过构建自己的第一个RAG系统进入RAG领域。或者&#xff0c;您可能已经构建了基本的RAG系统&#xff0c;现在希望将它们改进为更高级的系统&#xff0c;以更好地处理用户的查询和数据结构。 无论哪种情况&#xff0c;了解从何处或如何开…

c++类与对象(五):友元、内部类、临时对象、匿名对象

上次重新再次补全了构造函数的内容&#xff0c;以及static成员&#xff1a;C类与对象&#xff08;四&#xff09;&#xff1a;再谈构造函数&#xff08;详解初始化列表&#xff09;、Static成员 今天就来进行类与对象最后一部分的内容 文章目录 1.友元1.1友元函数1.2友元类 2.内…

JVM系列-4.类加载器

&#x1f44f;作者简介&#xff1a;大家好&#xff0c;我是爱吃芝士的土豆倪&#xff0c;24届校招生Java选手&#xff0c;很高兴认识大家&#x1f4d5;系列专栏&#xff1a;Spring原理、JUC原理、Kafka原理、分布式技术原理、数据库技术、JVM原理&#x1f525;如果感觉博主的文…

LeetCode670.最大交换

我真的怀疑他是不是难度等级评错了&#xff0c;因为感觉没到中级&#xff0c;总之先看题吧 给定一个非负整数&#xff0c;你至多可以交换一次数字中的任意两位。返回你能得到的最大值。 示例 1 : 输入: 2736 输出: 7236 解释: 交换数字2和数字7。示例 2 : 输入: 9973 输出:…

常用的三维尺寸公差分析软件有哪些?各有什么特点?

公差分析软件主要用于产品设计和制造过程中&#xff0c;帮助工程师们评估和控制产品的尺寸和公差。以下是一些常用的公差分析软件&#xff1a; 1.DTAS3D是一种用于三维尺寸公差分析的软件系统。 DTAS软件可以帮助工程师和设计师对零件和装配体的尺寸公差进行分析&#xff0c;…

Python os模块

简介 Python的os模块是一个标准库模块&#xff0c;用于提供与操作系统相关的功能&#xff08;相当于接口&#xff09;。os模块允许Python程序与文件系统、目录结构、进程管理等操作系统级别的功能进行交互。 主要功能 文件和目录操作 创建、删除、重命名文件和目录&#xf…

centos7安装pip

centos7 没有python-pip包就执行命令 yum -y install epel-release 执行成功之后&#xff0c;再次执行 yum install python-pip 对安装好的pip进行升级 pip install --upgrade pip 至此&#xff0c;pip工具就安装好了。 所有的问题都是最后一刻解决&#xff0c;如果没有解决…

智慧工业园区建设方案-智慧化工园区物联网管理系统平台---豌豆云

将化工园区海量信息互联、互通、互融&#xff0c;结合化工园区建设管理经验&#xff0c;通过“动态感知、主动监测、政企联动、综合管理”。 将事件管理从事后处置变为事前防控&#xff0c;保障生产安全&#xff0c;提升环保水平&#xff0c;结合可视化大屏&#xff0c;帮助管…

静态路由实验

一&#xff1a;实验内容 二&#xff1a;实验分析 &#xff08;一&#xff09;&#xff1a;实验要求 1、R6为ISP&#xff0c;接口IP地址均为公有地址&#xff1b;该设备只能配置IP地址&#xff0c;之后不能再对其进行其他任何配置&#xff1b; 2、R1-R5为局域网&#xff0c…

Vue.js代码检查

一、CSS 1、属性可安全的替换为速记形式 .el-dialog .el-dialog__header .el-dialog__body {padding: 0 40px;padding-top: 4px;}.el-dialog .el-dialog__header .el-dialog__body {padding: 4px 40px 0;} 2、度量单位冗余 padding: 0px 20px;// 修改后 padding: 0 20px; 3、…

过滤器监听器拦截器AOP

过滤器、监听器、拦截器、AOP的实现 一、过滤器 Filter ​ 在传统的Servlet容器中&#xff0c;可以使用过滤器和监听器&#xff0c;在Java框架中还可以使用拦截器。 ​ 过滤器&#xff0c;这里指的是Servlet过滤器&#xff0c;它是在Java Servlet中定义的&#xff0c;能够对…

EasyExcelFactory 导入导出功能的实战使用

EasyExcelFactory 导入导出功能的实战使用分享&#xff1a; 1、jar包引入 <!-- 阿里巴巴Excel处理--><dependency><groupId>com.alibaba</groupId><artifactId>easyexcel</artifactId><version>3.0.6</version></dependen…

1、Seaborn可视化库

你的数据可视化编程初体验! Seaborn是一个基于matplotlib的图形可视化Python库,它提供了一种高级的API接口,使得制作统计图形更加容易。 Seaborn的目标是使可视化成为探索和理解数据的核心部分,它面向数据集的绘图功能对整个数据集进行操作,并在内部执行必要的语义映射和统…

Python - 【Socket】消息粘包处理demo(一)

一. 前言 在网络编程中&#xff0c;粘包是指TCP协议在传输过程中&#xff0c;多条数据被合并成一条数据发送或者一条数据被拆分成多个数据发送的现象。 二. 粘包问题的常规处理方法&#xff1a; 使用固定长度的包头 可以在发送数据前先发送一个固定长度的包头&#xff0c;包…

个人云服务器docker搭建部署前后端应用-myos

var code "87c5235c-b551-45bb-a5e4-9593cb104663" mysql、redis、nginx、java应用、前端应用部署 本文以单台云服务器为例&#xff1a; 1. 使用腾讯云服务器 阿里或其他云服务器皆可&#xff0c;类似 安装系统&#xff0c;现在服务器系统都集成安装了docker镜像&a…

Vue中的模式和环境变量

文章目录 一、介绍二、配置1、环境文件2、变量使用 三、读取环境文件 一、介绍 vue官网&#xff1a;https://cli.vuejs.org/zh/guide/mode-and-env.html模式是 Vue CLI 项目中一个重要的概念。默认情况下&#xff0c;一个 Vue CLI 项目有三个模式 开发环境&#xff1a;develop…

[ACM学习]自上而下树形dp

问题引入 设置dp状态&#xff0c;相比于更容易出错的贪心更...不易出错。 状态设计 如果选择父结点&#xff0c;就会使孩子结点不能被选择&#xff0c;我们会多开一维的dp&#xff0c;用来标记该点是否被标记过。 以1点举例&#xff0c;f[1][0]为不选它的状态&#xff0c;那么…

微机原理 常考实验 第(二)弹~

1&#xff0c;已知在AX&#xff0c;BX中放32位有符号的二进制数&#xff0c;求其绝对值送入CX&#xff0c;DX中&#xff0c;其中AX&#xff0c;CX放高位。 分析&#xff1a;就是 AX: BX双字的绝对值放到CX: DX的双字中 &#xff08;1&#xff09;先判断AX,BX是不是为正&#…