利用FFmpge进行视频压缩(从图像到H264视频流)
对于FFmpeg相信做视频或图像处理这一块的都不会陌生,在网上也能找到非常多相关的代码。但因为版本号不同等原因。往往找到的代码都是须要自行改动才干够用,为此本人希望能尽绵薄之力,将开发包和自行编写的代码都放出来,假设刚開始学习的人想要能够直接执行的代码做參考的话。能够下载我放出的FFmpeg开发包进行配置(配置的教程地址例如以下:点击打开链接)。然后參考我写的编解码代码来进行程序的开发。
以下贴出的是我自己封装的FFmpeg视频压缩代码。如有更好的建议请告诉我,转载请注明出处。
首先我们设计一个视频压缩相关的类,定义例如以下
class Ffmpeg_Encoder
{
public:
AVFrame *m_pYUVFrame; //帧对象
AVCodec *pCodecH264; //编码器
AVCodecContext *c; //编码器数据结构对象
uint8_t *yuv_buff; //yuv图像数据区
uint8_t *rgb_buff; //rgb图像数据区
SwsContext *scxt; //图像格式转换对象
uint8_t *outbuf; //编码出来视频数据缓存
int outbuf_size; //编码输出数据去大小
int nDataLen; //rgb图像数据区长度
int width; //输出视频宽度
int height; //输出视频高度
AVPacket pkt; //数据包对象
int imgwidthlen; //图像宽度占用空间
public:
void Ffmpeg_Encoder_Init();//初始化
void Ffmpeg_Encoder_Setpara(AVCodecID mycodeid, int vwidth, int vheight);//设置參数,第一个參数为编码器,第二个參数为压缩出来的视频的宽度,第三个视频则为其高度
void Ffmpeg_Encoder_Encode(FILE *file, uint8_t *data);//编码并写入数据到文件
void Ffmpeg_Encoder_Close();//关闭
};
对类中声明的四个函数进行定义
void Ffmpeg_Encoder::Ffmpeg_Encoder_Init()
{
av_register_all();
avcodec_register_all();
m_pYUVFrame = new AVFrame[1];//YUV帧数据赋值
c = NULL;//解码器指针对象赋初值
}
void Ffmpeg_Encoder::Ffmpeg_Encoder_Setpara(AVCodecID mycodeid, int vwidth, int vheight)
{
pCodecH264 = avcodec_find_encoder(mycodeid);//查找h264编码器
if (!pCodecH264)
{
fprintf(stderr, "h264 codec not found\n");
exit(1);
}
width = vwidth;
height = vheight; c = avcodec_alloc_context3(pCodecH264);//函数用于分配一个AVCodecContext并设置默认值。假设失败返回NULL。并可用av_free()进行释放
c->bit_rate = 1024*1024; //码率,越高视频质量越好
c->width = vwidth;//设置编码视频宽度
c->height = vheight; //设置编码视频高度
c->time_base.den = 25;//设置帧率,num为分子和den为分母,假设是1/25则表示25帧/s
c->time_base.num = 1;
c->gop_size = 10; //设置GOP大小,该值表示每10帧会插入一个I帧
c->max_b_frames = 1;//设置B帧最大数,该值表示在两个非B帧之间,所同意插入的B帧的最大帧数
c->pix_fmt = AV_PIX_FMT_YUV420P;//设置像素格式 imgwidthlen = c->width * 3;
av_opt_set(c->priv_data, "tune", "zerolatency", 0);//设置编码器的延时,解决前面的几十帧不出数据的情况 if (avcodec_open2(c, pCodecH264, NULL) < 0)//打开编码器
return; nDataLen = vwidth*vheight * 3;//计算图像rgb数据区长度 yuv_buff = new uint8_t[nDataLen/2];//初始化数据区,为yuv图像帧准备填充缓存
rgb_buff = new uint8_t[nDataLen];//初始化数据区,为rgb图像帧准备填充缓存
outbuf_size = 100000;////初始化编码输出数据区
outbuf = new uint8_t[outbuf_size]; scxt = sws_getContext(c->width, c->height, AV_PIX_FMT_BGR24, c->width, c->height, AV_PIX_FMT_YUV420P, SWS_POINT, NULL, NULL, NULL);//初始化格式转换函数
av_image_fill_arrays(m_pYUVFrame->data, m_pYUVFrame->linesize, yuv_buff, AV_PIX_FMT_YUV420P, width, height, 1);
}
void Ffmpeg_Encoder::Ffmpeg_Encoder_Encode(FILE *file, uint8_t *data)
{
av_init_packet(&pkt);
memcpy(rgb_buff, data, nDataLen);//拷贝图像数据到rgb图像帧缓存中准备处理
sws_scale(scxt, &rgb_buff, &imgwidthlen, 0, c->height, m_pYUVFrame->data, m_pYUVFrame->linesize);// 将RGB转化为YUV int myoutputlen = 0;
int returnvalue = avcodec_encode_video2(c, &pkt, m_pYUVFrame, &myoutputlen);
if (returnvalue == 0)
{
fwrite(pkt.data, 1, pkt.size, file);
}
}
void Ffmpeg_Encoder::Ffmpeg_Encoder_Close()
{
delete[]m_pYUVFrame;
delete[]rgb_buff;
delete[]yuv_buff;
delete[]outbuf;
sws_freeContext(scxt);
avcodec_close(c);//关闭编码器
av_free(c);
}
最后我们仅仅须要在主函数对这几个函数进行调用就能够了。因为本人为了方便直接用OpencCV来打开图像并取得图像的数据区。所以假设希望直接执行本人后面所发的project的话还须要自行去配置OpenCV。只是这个在网上实在是说烂了,随便找找就能配置出来。假设不希望用OpenCV来打开图像的话,本人在代码中也注明了应该改动的位置。可自行想办法得到图像数据区并放入视频压缩函数就可以。
void main()
{
Ffmpeg_Encoder ffmpegobj;
//图象编码
FILE *f = NULL;
char * filename = "myData.h264";
fopen_s(&f, filename, "wb");//打开文件存储编码完毕数据 IplImage* img = NULL;//OpenCV图像数据结构指针 int picturecount = 1;
bool firstencode = true;
/**此部分用的是OpenCV读入图像对象并取得图像的数据区,也能够用别的方法获得图像数据区**/
img = cvLoadImage("1.png", 1);//打开图像
while (picturecount != 60)
{
if (firstencode)
{
firstencode = false;
ffmpegobj.Ffmpeg_Encoder_Init();//初始化编码器
ffmpegobj.Ffmpeg_Encoder_Setpara(AV_CODEC_ID_H264, img->width, img->height);//设置编码器參数
}
ffmpegobj.Ffmpeg_Encoder_Encode(f, (uchar*)img->imageData);//编码
picturecount++;
}
fclose(f);
ffmpegobj.Ffmpeg_Encoder_Close();
cvReleaseImage(&img);//释放图像数据结构指针对像所指内容
}
OK。到此用FFmpeg进行视频压缩的博客就告一段落了,以下是整个project的下载地址:点击打开链接。只是假设想要执行起来配置OpenCV还是须要自行动手的哈哈。
另外在做项目的过程中在编码这块还遇到一个非常奇怪的问题。就是在编码调用到avcodec_encode_video2特别是project也调用到解码函数的时候,在VS里面点击执行是没有问题的,但一执行release目录以下编译出来的exe文件,程序偶尔会报内存错误,错误的位置在avcodec_encode_video2这个函数里面。
这个让人百思不得其解。后来想想或者VS在执行的时候兼容性比較好,所以没有这个问题。是故例如以下图对exe文件的兼容性进行设置
这个问题就攻克了,是故怀疑ffmpeg库的avcodec_encode_video2这个函数所在的dll对于windows或者是windows64位系统的兼容性不大好。
利用FFmpge进行视频压缩(从图像到H264视频流)的更多相关文章
- Matlab利用subplot绘制多个图像
利用subplot绘制多个图像 subplot(m,n,p) subplot是将多个图画到一个平面上的函数,m是行,n是列,p是所要绘制图所在的位置 x = 0:0.1:100; sinY = sin ...
- OpenCV实时美颜摄像并生成H264视频流
为什么美颜摄像这么简单的功能,OpenCV这个开源项目网上很少有代码呢?对于在windows平台下,生成h264视频流也比价麻烦,没有现成的api可以使用,需要借助MinGw编译libx264,或者f ...
- 从wireshark数据中分析rtmp协议,并提取出H264视频流
我写的小工具 rtmp_parse.exe 使用用法如先介绍下: -sps [文件路径] 解析 sps 数据 文件当中的内容就是纯方本的hexstring: 如 42 E0 33 8D 68 05 ...
- 在iOS平台使用ffmpeg解码h264视频流(转)
在iOS平台使用ffmpeg解码h264视频流,有需要的朋友可以参考下. 对于视频文件和rtsp之类的主流视频传输协议,ffmpeg提供avformat_open_input接口,直接将文件路径或UR ...
- 在iOS平台使用ffmpeg解码h264视频流
来源:http://www.aichengxu.com/view/37145 在iOS平台使用ffmpeg解码h264视频流,有需要的朋友可以参考下. 对于视频文件和rtsp之类的主流视频传输协议,f ...
- (数据科学学习手札55)利用ggthemr来美化ggplot2图像
一.简介 R中的ggplot2是一个非常强大灵活的数据可视化包,熟悉其绘图规则后便可以自由地生成各种可视化图像,但其默认的色彩和样式在很多时候难免有些过于朴素,本文将要介绍的ggthemr包专门针对原 ...
- 利用PIL库创建空白图像
背景 最近,想自己生成带位置坐标的文字数据集来训练文本位置探测网络. 理想情况是,给文字加盐噪声,背景不需要加噪声,所以需要创建一个空白的背景.将文字放在空白背景上,然后利用opencv加噪声. 解决 ...
- python利用sift和surf进行图像配准
1.SIFT特征点和特征描述提取(注意opencv版本) 高斯金字塔:O组L层不同尺度的图像(每一组中各层尺寸相同,高斯函数的参数不同,不同组尺寸递减2倍) 特征点定位:极值点 特征点描述:根据不同b ...
- 利用onMouseOver和onMouseOut实现图像翻滚
代码: <img src="images/001.jpg" alt="pic" onmouseover="this.src='images/00 ...
随机推荐
- hihocoder offer收割编程练习赛13 D 骑士游历
思路: 矩阵快速幂. 实现: #include <iostream> #include <cstdio> #include <vector> using names ...
- Django--4、认证系统
cookie与session 概念 因http协议无法保存状态,但是又需要保存状态,所以有了cookie.它不属于http协议范畴 工作原理:相当于一段标识数据.在访问服务器产生标识内容(cookie ...
- (转)全文检索技术学习(一)——Lucene的介绍
http://blog.csdn.net/yerenyuan_pku/article/details/72582979 本文我将为大家讲解全文检索技术——Lucene,现在这个技术用到的比较多,我觉得 ...
- Exception Information
https://developer.apple.com/library/content/technotes/tn2004/tn2123.html Exception Information The t ...
- CSS 命名规范 BEM 思想
Part.1 何为 BEM? BEM :Block ( 块 ) 丶Element ( 元素 ) 丶Modifier ( 修饰符 ) 出 处:是由 Yandex 团队提出的一种前端命名方法论 优 点:命 ...
- vue基础---条件渲染
(1)v-if条件渲染 v-if 指令用于条件性地渲染一块内容.这块内容只会在指令的表达式返回 truthy 值的时候被渲染. 可以用 v-else 添加一个“else 块”: ①表达式 <di ...
- jquery 实现点评标签 类似淘宝大众点评的 快速准时 货品完好等
111 <!DOCTYPE html> <html> <head> <meta charset="utf-8" /> <tit ...
- Bullet:ORACLE Using SQL Plan Management(一)
SQL Plan Management如何工作? 当一个SQL硬解析时,基于成本的优化器CBO会生成多个执行计划,并从这些执行计划中选择一个优化器认为最低成本的执行计划. 如果SQL plan bas ...
- 洛谷——P1229 遍历问题
P1229 遍历问题 题目描述 我们都很熟悉二叉树的前序.中序.后序遍历,在数据结构中常提出这样的问题:已知一棵二叉树的前序和中序遍历,求它的后序遍历,相应的,已知一棵二叉树的后序遍历和中序遍历序列你 ...
- springboot+idea+jsp 404问题
我是这么解决的 对于单一项目,加入以下jar包即可. <!--前台页面的支持--> <dependency> <groupId>javax.servlet</ ...