搞视频处理的朋友肯定比较熟悉YUV视频序列,很多测试库提供的视频数据都是YUV视频序列,我们这里就用用YUV视频序列来做视频。关于YUV视频序列,我就不多讲了,可以看书学习,通常的视频序列都是YUV420格式的。

步骤也就那几步,添加视频流,打开编码器,开辟相应的内存空间,然后就可以打开YUV序列逐帧写入数据了,so easy!记得最后要做好文件的关闭和内存的释放,因为FFmpeg是c风格的(不知道新版本是否是c++风格的),这些工作都需要自己做好啊。过多的说明是没用的,直接上代码:

这里我补充一下,大多数的视频格式好像只支持YUV格式的视频帧AVFrame,我试图直接把RGB的视频序列直接编码到视频这条路好像走不通,都需要把RGB的视频帧再转成YUV视频帧才行,不知道高手有没有其他高见。

[cpp] view plaincopy
  1. #include <stdio.h>
  2. #include <string.h>
  3. extern "C"
  4. {
  5. #include <libavcodec\avcodec.h>
  6. #include <libavformat\avformat.h>
  7. #include <libswscale\swscale.h>
  8. };
  9. void main(int argc, char ** argv)
  10. {
  11. AVFormatContext* oc;
  12. AVOutputFormat* fmt;
  13. AVStream* video_st;
  14. double video_pts;
  15. uint8_t* video_outbuf;
  16. uint8_t* picture_buf;
  17. AVFrame* picture;
  18. //  AVFrame* pictureRGB;
  19. int size;
  20. int ret;
  21. int video_outbuf_size;
  22. FILE *fin = fopen("akiyo_qcif.yuv", "rb"); //视频源文件
  23. const char* filename = "test.mpg";
  24. //  const char* filename;
  25. //  filename = argv[1];
  26. av_register_all();
  27. //  avcodec_init(); // 初始化codec库
  28. //  avcodec_register_all(); // 注册编码器
  29. fmt = guess_format(NULL, filename, NULL);
  30. oc = av_alloc_format_context();
  31. oc->oformat = fmt;
  32. snprintf(oc->filename, sizeof(oc->filename), "%s", filename);
  33. video_st = NULL;
  34. if (fmt->video_codec != CODEC_ID_NONE)
  35. {
  36. AVCodecContext* c;
  37. video_st = av_new_stream(oc, 0);
  38. c = video_st->codec;
  39. c->codec_id = fmt->video_codec;
  40. c->codec_type = CODEC_TYPE_VIDEO;
  41. c->bit_rate = 400000;
  42. c->width = 176;
  43. c->height = 144;
  44. c->time_base.num = 1;
  45. c->time_base.den = 25;
  46. c->gop_size = 12;
  47. c->pix_fmt = PIX_FMT_YUV420P;
  48. if (c->codec_id == CODEC_ID_MPEG2VIDEO)
  49. {
  50. c->max_b_frames = 2;
  51. }
  52. if (c->codec_id == CODEC_ID_MPEG1VIDEO)
  53. {
  54. c->mb_decision = 2;
  55. }
  56. if (!strcmp(oc->oformat->name, "mp4") || !strcmp(oc->oformat->name, "mov") || !strcmp(oc->oformat->name, "3gp"))
  57. {
  58. c->flags |= CODEC_FLAG_GLOBAL_HEADER;
  59. }
  60. }
  61. if (av_set_parameters(oc, NULL)<0)
  62. {
  63. return;
  64. }
  65. dump_format(oc, 0, filename, 1);
  66. if (video_st)
  67. {
  68. AVCodecContext* c;
  69. AVCodec* codec;
  70. c = video_st->codec;
  71. codec = avcodec_find_encoder(c->codec_id);
  72. if (!codec)
  73. {
  74. return;
  75. }
  76. if (avcodec_open(c, codec) < 0)
  77. {
  78. return;
  79. }
  80. if (!(oc->oformat->flags & AVFMT_RAWPICTURE))
  81. {
  82. video_outbuf_size = 200000;
  83. video_outbuf = (uint8_t*)av_malloc(video_outbuf_size);
  84. }
  85. picture = avcodec_alloc_frame();
  86. size = avpicture_get_size(c->pix_fmt, c->width, c->height);
  87. picture_buf = (uint8_t*)av_malloc(size);
  88. if (!picture_buf)
  89. {
  90. av_free(picture);
  91. }
  92. avpicture_fill((AVPicture*)picture, picture_buf, c->pix_fmt, c->width, c->height);
  93. }
  94. if (!(fmt->flags & AVFMT_NOFILE))
  95. {
  96. if (url_fopen(&oc->pb, filename, URL_WRONLY) < 0)
  97. {
  98. return;
  99. }
  100. }
  101. av_write_header(oc);
  102. for (int i=0; i<300; i++)
  103. {
  104. if (video_st)
  105. {
  106. video_pts = (double)(video_st->pts.val * video_st->time_base.num / video_st->time_base.den);
  107. }
  108. else
  109. {
  110. video_pts = 0.0;
  111. }
  112. if (!video_st/* || video_pts >= 5.0*/)
  113. {
  114. break;
  115. }
  116. AVCodecContext* c;
  117. c = video_st->codec;
  118. size = c->width * c->height;
  119. if (fread(picture_buf, 1, size*3/2, fin) < 0)
  120. {
  121. break;
  122. }
  123. picture->data[0] = picture_buf;  // 亮度
  124. picture->data[1] = picture_buf+ size;  // 色度
  125. picture->data[2] = picture_buf+ size*5/4; // 色度
  126. // 如果是rgb序列,可能需要如下代码
  127. //      SwsContext* img_convert_ctx;
  128. //      img_convert_ctx = sws_getContext(c->width, c->height, PIX_FMT_RGB24, c->width, c->height, c->pix_fmt, SWS_BICUBIC, NULL, NULL, NULL);
  129. //      sws_scale(img_convert_ctx, pictureRGB->data, pictureRGB->linesize, 0, c->height, picture->data, picture->linesize);
  130. if (oc->oformat->flags & AVFMT_RAWPICTURE)
  131. {
  132. AVPacket pkt;
  133. av_init_packet(&pkt);
  134. pkt.flags |= PKT_FLAG_KEY;
  135. pkt.stream_index = video_st->index;
  136. pkt.data = (uint8_t*)picture;
  137. pkt.size = sizeof(AVPicture);
  138. ret = av_write_frame(oc, &pkt);
  139. }
  140. else
  141. {
  142. int out_size = avcodec_encode_video(c, video_outbuf, video_outbuf_size, picture);
  143. if (out_size > 0)
  144. {
  145. AVPacket pkt;
  146. av_init_packet(&pkt);
  147. pkt.pts = av_rescale_q(c->coded_frame->pts, c->time_base, video_st->time_base);
  148. if (c->coded_frame->key_frame)
  149. {
  150. pkt.flags |= PKT_FLAG_KEY;
  151. }
  152. pkt.stream_index = video_st->index;
  153. pkt.data = video_outbuf;
  154. pkt.size = out_size;
  155. ret = av_write_frame(oc, &pkt);
  156. }
  157. }
  158. }
  159. if (video_st)
  160. {
  161. avcodec_close(video_st->codec);
  162. //      av_free(picture->data[0]);
  163. av_free(picture);
  164. av_free(video_outbuf);
  165. //      av_free(picture_buf);
  166. }
  167. av_write_trailer(oc);
  168. for (int i=0; i<oc->nb_streams; i++)
  169. {
  170. av_freep(&oc->streams[i]->codec);
  171. av_freep(&oc->streams[i]);
  172. }
  173. if (!(fmt->flags & AVFMT_NOFILE))
  174. {
  175. url_fclose(oc->pb);
  176. }
  177. av_free(oc);
  178. }

原文地址:http://blog.csdn.net/yang_xian521/article/details/7698742

使用FFmpeg类库实现YUV视频序列编码为视频相关推荐

  1. FFmpeg浅尝辄止(二)——YUV视频序列编码为视频 ~~ 新版修改版-调通

    文章来源: FFmpeg浅尝辄止(二)--YUV视频序列编码为视频 针对新版FFmpeg进行修改后,VC2010调通 AVFormatContext* oc; AVOutputFormat* fmt; ...

  2. YUV测试序列下载及视频序列的单帧播放工具使用

    YUV测试序列下载 http://www.cipr.rpi.edu/resource/sequences/ 视频序列的单帧播放 http://www.usr.cc/thread-52335-1-1.h ...

  3. 音视频方案,音视频扩展内容(RTMP,FFMpeg/H.26*/mpeg*/AVC等标准与协议)(笔记)1,视频格式

    视频方案,雷霄骅的专栏- http://blog.csdn.net/leixiaohua1020  > SI, TI   ITU-R BT.1788建议使用时间信息(TI,Temporal pe ...

  4. 微信小程序 RTMP 音视频 通话 ffmpeg_音视频面试基础题

    编码原理 为什么巨大的原始视频可以编码成很小的视频呢?这其中的技术是什么呢?核心思想就是去除冗余信息: 1)空间冗余:图像相邻像素之间有较强的相关性 2)时间冗余:视频序列的相邻图像之间内容相似 3) ...

  5. 利用FFmpeg制作视频序列

    前言 本文介绍了利用FFmpeg进行视频测试序列制作的方法.所谓测试序列就是将若干个不同场景的视频片段,按照不同的分辨率.码率.编码方法等进行处理,然后拼接成一条长视频,供测试者打分.一般包括以下几个 ...

  6. FFMPEG采集摄像头图像SDL渲染+MP4格式视频编码

    FFMPEG采集摄像头图像SDL渲染+MP4格式视频编码 FFmpeg是一套可以用来记录.转换数字音频.视频,并能将其转化为流的开源计算机程序.采用LGPL或GPL许可证.它提供了录制.转换以及流化音 ...

  7. 利用VS2015与FFmpeg实现视频解编码的基本过程

    利用VS2015与FFmpeg实现视频解编码的基本过程 最近接到一个任务,是有关于视频的网络传输的技术应用,因为在传输的过程中用到了视频的编解码技术,所以近期对其进行了一系列学习.现将最近的学习成果即 ...

  8. VC++实现视频聊天:VFW视频采集+FFmpeg编码

    上次介绍了视频聊天软件的界面.文字聊天.文件传输部分,这此介绍视频聊天功能,这算是音视频领域一个很广的应用.首先视频聊天的双方需要有一个USB摄像头(或者笔记本摄像头),在windows系统下,一个完 ...

  9. iOS 使用FFmpeg实现视频H264编码

    本文借鉴:https://www.jianshu.com/p/70b0af4d0ec7   以及 https://www.jianshu.com/p/31d1ca4999c6 ffmpeg 相关命令行 ...

最新文章

  1. 【青少年编程】【四级】从小到大排序
  2. mysql command line闪退_MySQL Command Line Client 打不开(闪退)的解决
  3. K12教育整体业务流程+涉及各类产品分析
  4. Windows Socket五种I/O模型
  5. Android studio中不同颜色代表什么意思
  6. [Swift]LeetCode921.使括号有效的最少添加 | Minimum Add to Make Parentheses Valid
  7. SpringBoot启动报jdbc连接池错误
  8. win10电脑虚拟网络设置方法
  9. lingo的目标规划
  10. 中国的北斗,世界的北斗|hightopo卫星发射demo
  11. 声纹技术(四):声纹识别的工程部署
  12. HDMI的DDC是什么
  13. uniapp获取当前经纬度 地图 支持搜索
  14. 基于redis实现抢红包功能(包括余额退回处理)
  15. Oracle取某个时间点前后的整半小时的时间点
  16. 快速调整 图片的 像素大小
  17. C fread()函数
  18. ping www.baidu.com虚拟机中ping百度
  19. Django博客开发(十三)—一个简单的站内搜索
  20. 构建安全可靠的微服务 | Nacos 在颜铺 SaaS 平台的应用实践

热门文章

  1. HDU1557 权利指数【状态压缩】
  2. CCF201503-5 最小花费(30分)
  3. Ubuntu下BOINC服务器安装(step8失败)
  4. RSA的JavaScript程序
  5. linux 下的文件搜索、可执行文件搜索
  6. 分布式计算的基本概念
  7. leetcode(5)—— Longest Palindromic Substring(最长回文子串)
  8. 设p他主修计算机科学,华南农业大学 离散数学 期末考试2013试卷及答案
  9. python画3d图-Python绘制3D图形
  10. python软件下载-python软件下载