版权声明:本文为博主原创文章,未经博主允许不得转载。

使用ffmpeg解码h264数据其实相对使用x264进行视频编码是简单了许多的,因为ffmpeg提供了一个decoding_encoding.c的文件,这个文件里面有简单的使用ffmpeg进行视频、音频编解码的例子,不过可能有的人也会找不到这个示例,我就讲我改造过的这个示例放在这里,同时加一些解释。

其中需要注意的的一点我需要在此说明,就是ffmpeg在进行解码的时候是会考虑要解码的数据包是否有0x00 00 001这样的头的,如果没有的话,ffmpeg会认为是错误的数据包。下面是使用opencv对解码后的图像进行显示,所以还要配置opencv的环境,如果没有的话,可以注释掉ShowImage这个函数,然后使用pgm_save这个函数将解码后的图像保存。

下面将我的代码放在下面,同样,过程参见代码注释,相对来说比较简单,不在此过多叙述:

[cpp] view
plain
 copy

  1. static void pgm_save(unsigned char *buf, int wrap, int xsize, int ysize,char*filename)
  2. {
  3. FILE *f;
  4. int i;
  5. f=fopen(filename,"wb");
  6. fprintf(f,"P5\n%d%d\n%d\n",xsize,ysize,255);
  7. for(i=0;i<ysize;i++)
  8. fwrite(buf + i * wrap,1,xsize,f);
  9. fclose(f);
  10. }
  11. //通过查找0x000001或者0x00000001找到下一个数据包的头部
  12. static int _find_head(unsigned char*buffer, int len)
  13. {
  14. int i;
  15. for(i=512;i<len;i++)
  16. {
  17. if(buffer[i] == 0 && buffer[i+1] == 0 && buffer[i+2] == 0&& buffer[i+3] == 1)
  18. break;
  19. if(buffer[i]== 0 && buffer[i+1] == 0 && buffer[i+2] == 1)
  20. break;
  21. }
  22. if (i ==len)
  23. return0;
  24. if (i ==512)
  25. return0;
  26. return i;
  27. }
  28. //将文件中的一个数据包转换成AVPacket类型以便ffmpeg进行解码
  29. #define FILE_READING_BUFFER (1*1024*1024)
  30. static void build_avpkt(AVPacket *avpkt, FILE *fp)
  31. {
  32. static unsigned charbuffer[1*1024*1024];
  33. static int readptr = 0;
  34. static int writeptr = 0;
  35. intlen,toread;
  36. intnexthead;
  37. if (writeptr- readptr < 200 * 1024)
  38. {
  39. memmove(buffer, &buffer[readptr],writeptr - readptr);
  40. writeptr -= readptr;
  41. readptr = 0;
  42. toread = FILE_READING_BUFFER - writeptr;
  43. len = fread(&buffer[writeptr], 1,toread, fp);
  44. writeptr += len;
  45. }
  46. nexthead = _find_head(&buffer[readptr], writeptr-readptr);
  47. if (nexthead== 0)
  48. {
  49. printf("failedfind next head...\n");
  50. nexthead = writeptr - readptr;
  51. }
  52. avpkt->size = nexthead;
  53. avpkt->data = &buffer[readptr];
  54. readptr += nexthead;
  55. }
  56. static voidvideo_decode_example(const char *outfilename, constchar *filename)
  57. {
  58. AVCodec *codec;
  59. AVCodecContext *c= NULL;
  60. int frame,got_picture, len;
  61. FILE *f, *fout;
  62. AVFrame *picture;
  63. uint8_t inbuf[INBUF_SIZE +FF_INPUT_BUFFER_PADDING_SIZE];
  64. charbuf[1024];
  65. AVPacket avpkt;
  66. av_init_packet(&avpkt);
  67. /* set end ofbuffer to 0 (this ensures that no overreading happens for damaged mpeg streams)*/
  68. memset(inbuf + INBUF_SIZE, 0,FF_INPUT_BUFFER_PADDING_SIZE);
  69. printf("Videodecoding\n");
  70. opts = NULL;
  71. //av_dict_set(&opts,"b", "2.5M", 0);
  72. /* find the h264video decoder */
  73. codec = avcodec_find_decoder(CODEC_ID_H264);
  74. if (!codec){
  75. fprintf(stderr, "codecnot found\n");
  76. return ;
  77. }
  78. c = avcodec_alloc_context3(codec);
  79. picture= avcodec_alloc_frame();
  80. if(codec->capabilities&CODEC_CAP_TRUNCATED)
  81. c->flags|= CODEC_FLAG_TRUNCATED; /* we do not send complete frames */
  82. /* For somecodecs, such as msmpeg4 and mpeg4, width and height
  83. MUST be initialized there because thisinformation is not
  84. available in the bitstream. */
  85. /* open it */
  86. if(avcodec_open2(c, codec, NULL) < 0) {
  87. fprintf(stderr, "couldnot open codec\n");
  88. exit(1);
  89. }
  90. //  fout=fopen(outfilename,"wb");
  91. /* the codec givesus the frame size, in samples */
  92. f = fopen(filename, "rb");
  93. if (!f) {
  94. fprintf(stderr, "couldnot open %s\n", filename);
  95. exit(1);
  96. }
  97. //解码与显示需要的辅助的数据结构,需要注意的是,AVFrame必须经过alloc才能使用,不然其内存的缓存空间指针是空的,程序会崩溃
  98. AVFrame frameRGB;
  99. IplImage *showImage =cvCreateImage(cvSize(352,288),8,3);
  100. avpicture_alloc((AVPicture*)&frameRGB,PIX_FMT_RGB24,352,288);
  101. cvNamedWindow("decode");
  102. frame = 0;
  103. for(;;) {
  104. build_avpkt(&avpkt, f);
  105. if(avpkt.size == 0)
  106. break;
  107. while(avpkt.size > 0) {
  108. len = avcodec_decode_video2(c,picture, &got_picture, &avpkt);//解码每一帧
  109. if(len < 0) {
  110. fprintf(stderr, "Error while decoding frame %d\n",frame);
  111. break;
  112. }
  113. if(got_picture) {
  114. printf("savingframe %3d\n", frame);
  115. fflush(stdout);
  116. /* thepicture is allocated by the decoder. no need to free it */
  117. //将YUV420格式的图像转换成RGB格式所需要的转换上下文
  118. SwsContext* scxt =sws_getContext(picture->width,picture->height,PIX_FMT_YUV420P,
  119. picture->width,picture->height,PIX_FMT_RGB24,
  120. 2,NULL,NULL,NULL);
  121. if(scxt != NULL)
  122. {
  123. sws_scale(scxt,picture->data,picture->linesize,0,c->height,frameRGB.data,frameRGB.linesize);//图像格式转换
  124. showImage->imageSize =frameRGB.linesize[0];//指针赋值给要显示的图像
  125. showImage->imageData = (char *)frameRGB.data[0];
  126. cvShowImage("decode",showImage);//显示
  127. cvWaitKey(0.5);//设置0.5s显示一帧,如果不设置由于这是个循环,会导致看不到显示出来的图像
  128. }
  129. //sprintf(buf,outfilename,frame);
  130. //pgm_save(picture->data[0],picture->linesize[0],
  131. //c->width,c->height, buf);
  132. //pgm_save(picture->data[1],picture->linesize[1],
  133. //c->width/2,c->height/2, fout);
  134. //pgm_save(picture->data[2],picture->linesize[2],
  135. //c->width/2,c->height/2, fout);
  136. frame++;
  137. }
  138. avpkt.size -= len;
  139. avpkt.data += len;
  140. }
  141. }
  142. /* some codecs,such as MPEG, transmit the I and P frame with a
  143. latency of one frame. You must do thefollowing to have a
  144. chance to get the last frame of the video */
  145. avpkt.data = NULL;
  146. avpkt.size = 0;
  147. len = avcodec_decode_video2(c, picture,&got_picture, &avpkt);
  148. if(got_picture) {
  149. printf("savinglast frame %3d\n", frame);
  150. fflush(stdout);
  151. /* the pictureis allocated by the decoder. no need to
  152. free it */
  153. sprintf(buf, outfilename, frame);
  154. //pgm_save(picture->data[0],picture->linesize[0],
  155. //       c->width, c->height, fout);
  156. pgm_save(picture->data[0],picture->linesize[0],c->width, c->height, fout);
  157. pgm_save(picture->data[1],picture->linesize[1],c->width/2, c->height/2, fout);
  158. pgm_save(picture->data[2],picture->linesize[2],c->width/2, c->height/2, fout);
  159. frame++;
  160. }
  161. fclose(f);
  162. //  fclose(fout);
  163. avcodec_close(c);
  164. av_free(c);
  165. av_free(picture);
  166. printf("\n");
  167. }
  168. int main(int argc, char* argv[])
  169. {
  170. avcodec_register_all();//注册所有的编解码器,一定要注意,如果没有这行代码则会出错,提示没有找不到编解码器
  171. video_decode_example("%3d.pgm","test.264");//可以使用x264编码出来的264文件
  172. system("pause");
  173. return 0;
  174. }

【FFMPEG】从内存中获取H264数据并进行decode的更多相关文章

  1. FFmpeg 如何探测网络流格式/如何从内存中获取数据

    文章转自:http://blog.csdn.net/rootusers/article/details/42551935 一般ffmpeg都是直接从文件中读取或者从网络流中读取,比如rtp://xx. ...

  2. C# 操作地址 从内存中读取写入数据(初级)

    本示例以植物大战僵尸为例, 实现功能为 每1秒让阳光刷新为 9999.本示例使用的游戏版本为 [植物大战僵尸2010年度版], 使用的辅助查看内存地址的工具是  CE. 由于每次启动游戏, 游戏中阳光 ...

  3. <转>libjpeg解码内存中的jpeg数据

    转自http://my.unix-center.net/~Simon_fu/?p=565 熟悉libjpeg的朋友都知道libjpeg是一个开源的库.Linux和Android都是用libjpeg来 ...

  4. 通过DialogFragment从DatePicker或TimePicker中获取日期数据

    通过DialogFragment从DatePicker或TimePicker中获取日期数据 一个activity类,里面存有date和time的变量,想通过dialogfragment的方式获取用户输 ...

  5. ffmpeg 从内存中读取数据(或将数据输出到内存)

    更新记录(2014.7.24): 1.为了使本文更通俗易懂,更新了部分内容,将例子改为从内存中打开. 2.增加了将数据输出到内存的方法. 从内存中读取数据 ffmpeg一般情况下支持打开一个本地文件, ...

  6. ffmpeg 从内存中读取数据(或将数据输出到内存)(转)

    更新记录(2014.7.24): 1.为了使本文更通俗易懂,更新了部分内容,将例子改为从内存中打开. 2.增加了将数据输出到内存的方法. 从内存中读取数据 ffmpeg一般情况下支持打开一个本地文件, ...

  7. ffmpeg 从内存中读取数据 .

    http://blog.csdn.net/leixiaohua1020/article/details/12980423 ——————————————————————————————————————— ...

  8. 【在网页中获取截图数据】Chrome和Firefox下的实战经验

    [转载自我在segmentfault的专栏:https://segmentfault.com/a/1190000004584071] 最近在实现一个功能,需求如下: 前提:当前页面无弹窗 页面任意位置 ...

  9. Android中获取网络数据时的分页加载

    //此实在Fragment中实现的,黄色部分为自动加载,红色部分是需要注意的和手动加载,    蓝色部分是睡眠时间,自我感觉不用写  ,还有就是手动加载时,不知道为什么进去后显示的就是最后一行,求大神 ...

随机推荐

  1. python中的pyc和pyo文件和__pyc__文件夹

    一.命令生成 pyc 文件 *.py:源码文件,由 Python 程序解释. *.pyc:源码经编译后生成的二进制字节码(Bytecode)文件. *.pyo:优化编译后的程序,也是二进制字节码文件. ...

  2. RecursiveTask和RecursiveAction的使用 以及java 8 并行流和顺序流(转)

    什么是Fork/Join框架        Fork/Join框架是Java7提供了的一个用于并行执行任务的框架, 是一个把大任务分割成若干个小任务,最终汇总每个小任务结果后得到大任务结果的框架. 我 ...

  3. HDU 5852 Intersection is not allowed! ( 2016多校9、不相交路径的方案、LGV定理、行列式计算 )

    题目链接 题意 : 给定方格中第一行的各个起点.再给定最后一行与起点相对应的终点.问你从这些起点出发到各自的终点.不相交的路径有多少条.移动方向只能向下或向右 分析 : 首先对于多起点和多终点的不相交 ...

  4. 下载PDF格式的Html

    下载PDF格式的Html 首先准备需要的两个js jsPdf.debug.js html2canvas.js 直接上代码: function download() { html2canvas(docu ...

  5. Zookeeper原理 二

    Zookeeper到底是什么!? 学一个东西,不搞明白他是什么东西,哪还有心情学啊!! 首先,Zookeeper是Apache的一个java项目,属于Hadoop系统,扮演管理员的角色. 然后看到官网 ...

  6. [Vue] : 键盘修饰符

    键盘修饰符以及自定义键盘修饰符 为文本框回车键绑定事件 <input type="text" class="form-control" v-model=& ...

  7. linux 搭建 jenkins 前端自动构建时,老是提示 sh: vue-cli-service: command not found

    如题. 在 shell 里面执行  env 发现一个   NODE_ENV=production  的环境变量,是这个东西的原因. 通过  unset NODE_ENV  命令去掉这个环境变量就可以了 ...

  8. poj1737

    Connected Graph POJ - 1737 An undirected graph is a set V of vertices and a set of E∈{V*V} edges.An ...

  9. deepin linux安装为知笔记

    sudo apt-get install wiznote

  10. node和npm版本引起的安装依赖和运行项目失败问题

    问题:node版本不同导致的安装依赖版本不同而无法启动 https://www.jianshu.com/p/c07293c8c6d4 实际上问题分为两个部分: 1,npm包管理器安装依赖不成功,此时需 ...