一、重采样流程

  重采样(解码音频数据之后格式不可以直接播放,需要重采样,类似图像的像素转换)
    1.分配上下文
    2.设置参数(分为(前几个是)输出格式和(后几个)输入格式,两个相对应的) 可以通过改变样本率来改变音频的播放速度,但是会失帧。
    3.初始化
    4.开始转换

二、相关函数说明

   1、 SwrContext *actx = swr_alloc(); //进行分配和初始化
   2、 actx = swr_alloc_set_opts(actx,
              av_get_default_channel_layout(2), //输出标准(左右声道)
              AV_SAMPLE_FMT_S16, //样本格式:float 、s16、s24
              ac->sample_rate,//样本率,一般使用原始的
              av_get_default_channel_layout(ac->channels),
              ac->sample_fmt,
              ac->sample_rate,
              0,0 );
   3、 re = swr_init(actx);
   4、 //将一帧一帧的音频进行重采样 (参数对应输入输出)
       int len = swr_convert(actx, //重采样上下文
              out, //输出的指针
              frame->nb_samples, //样本数量:单声道的样本数量
              (const uint8_t**)frame->data,//解码出来的data
              frame->nb_samples);//单通的样本数量

代码说明:

  1. #include <jni.h>
  2. #include <string>
  3. #include <android/log.h>
  4. #include <android/native_window.h>
  5. #include <android/native_window_jni.h>
  6. #define LOGW(...) __android_log_print(ANDROID_LOG_WARN,"testff",__VA_ARGS__)
  7.  
  8. extern "C"{
  9. #include <libavcodec/avcodec.h>
  10. #include <libavformat/avformat.h>
  11. #include <libavcodec/jni.h>
  12. #include <libswscale/swscale.h>
  13. #include <libswresample/swresample.h>
  14. }
  15. #include<iostream>
  16. using namespace std;
  17.  
  18. static double r2d(AVRational r)
  19. {
  20. return r.num==||r.den == ? :(double)r.num/(double)r.den;
  21. }
  22.  
  23. //当前时间戳 clock
  24. long long GetNowMs()
  25. {
  26. struct timeval tv;
  27. gettimeofday(&tv,NULL);
  28. int sec = tv.tv_sec%;
  29. long long t = sec*+tv.tv_usec/;
  30. return t;
  31. }
  32. extern "C"
  33. JNIEXPORT
  34. jint JNI_OnLoad(JavaVM *vm,void *res)
  35. {
  36. av_jni_set_java_vm(vm,);
  37. return JNI_VERSION_1_4;
  38. }
  39.  
  40. extern "C"
  41. JNIEXPORT jstring
  42. JNICALL
  43. Java_aplay_testffmpeg_MainActivity_stringFromJNI(
  44. JNIEnv *env,
  45. jobject /* this */) {
  46. std::string hello = "Hello from C++ ";
  47. // TODO
  48. hello += avcodec_configuration();
  49. return env->NewStringUTF(hello.c_str());
  50. }
  51.  
  52. extern "C"
  53. JNIEXPORT void JNICALL
  54. Java_aplay_testffmpeg_XPlay_Open(JNIEnv *env, jobject instance, jstring url_, jobject surface) {
  55. const char *path = env->GetStringUTFChars(url_, );
  56.  
  57. //初始化解封装
  58. av_register_all();
  59. //初始化网络
  60. avformat_network_init();
  61.  
  62. avcodec_register_all();
  63.  
  64. //打开文件
  65. AVFormatContext *ic = NULL;
  66. //char path[] = "/sdcard/video.flv";
  67. int re = avformat_open_input(&ic,path,,);
  68. if(re != )
  69. {
  70. LOGW("avformat_open_input failed!:%s",av_err2str(re));
  71. return;
  72. }
  73. LOGW("avformat_open_input %s success!",path);
  74. //获取流信息
  75. re = avformat_find_stream_info(ic,);
  76. if(re != )
  77. {
  78. LOGW("avformat_find_stream_info failed!");
  79. }
  80. LOGW("duration = %lld nb_streams = %d",ic->duration,ic->nb_streams);
  81.  
  82. int fps = ;
  83. int videoStream = ;
  84. int audioStream = ;
  85.  
  86. for(int i = ; i < ic->nb_streams; i++)
  87. {
  88. AVStream *as = ic->streams[i];
  89. if(as->codecpar->codec_type == AVMEDIA_TYPE_VIDEO)
  90. {
  91. LOGW("视频数据");
  92. videoStream = i;
  93. fps = r2d(as->avg_frame_rate);
  94.  
  95. LOGW("fps = %d,width=%d height=%d codeid=%d pixformat=%d",fps,
  96. as->codecpar->width,
  97. as->codecpar->height,
  98. as->codecpar->codec_id,
  99. as->codecpar->format
  100. );
  101. }
  102. else if(as->codecpar->codec_type ==AVMEDIA_TYPE_AUDIO )
  103. {
  104. LOGW("音频数据");
  105. audioStream = i;
  106. LOGW("sample_rate=%d channels=%d sample_format=%d",
  107. as->codecpar->sample_rate,
  108. as->codecpar->channels,
  109. as->codecpar->format
  110. );
  111. }
  112. }
  113. //ic->streams[videoStream];
  114. //获取音频流信息
  115. audioStream = av_find_best_stream(ic,AVMEDIA_TYPE_AUDIO,-,-,NULL,);
  116. LOGW("av_find_best_stream audioStream = %d",audioStream);
  117. //////////////////////////////////////////////////////////
  118. //打开视频解码器
  119. //软解码器
  120. AVCodec *codec = avcodec_find_decoder(ic->streams[videoStream]->codecpar->codec_id);
  121. //硬解码
  122. codec = avcodec_find_decoder_by_name("h264_mediacodec");
  123. if(!codec)
  124. {
  125. LOGW("avcodec_find failed!");
  126. return;
  127. }
  128. //解码器初始化
  129. AVCodecContext *vc = avcodec_alloc_context3(codec);
  130. avcodec_parameters_to_context(vc,ic->streams[videoStream]->codecpar);
  131.  
  132. vc->thread_count = ;
  133. //打开解码器
  134. re = avcodec_open2(vc,,);
  135. //vc->time_base = ic->streams[videoStream]->time_base;
  136. LOGW("vc timebase = %d/ %d",vc->time_base.num,vc->time_base.den);
  137. if(re != )
  138. {
  139. LOGW("avcodec_open2 video failed!");
  140. return;
  141. }
  142.  
  143. //////////////////////////////////////////////////////////
  144. //打开音频解码器
  145. //软解码器
  146. AVCodec *acodec = avcodec_find_decoder(ic->streams[audioStream]->codecpar->codec_id);
  147. //硬解码
  148. //codec = avcodec_find_decoder_by_name("h264_mediacodec");
  149. if(!acodec)
  150. {
  151. LOGW("avcodec_find failed!");
  152. return;
  153. }
  154. //音频解码器初始化
  155. AVCodecContext *ac = avcodec_alloc_context3(acodec);
  156. avcodec_parameters_to_context(ac,ic->streams[audioStream]->codecpar);
  157. ac->thread_count = ;
  158. //打开解码器
  159. re = avcodec_open2(ac,,);
  160. if(re != )
  161. {
  162. LOGW("avcodec_open2 audio failed!");
  163. return;
  164. }
  165. //读取帧数据
  166. AVPacket *pkt = av_packet_alloc();
  167. AVFrame *frame = av_frame_alloc();
  168. long long start = GetNowMs();
  169. int frameCount = ;
  170.  
  171. //初始化像素格式转换的上下文
  172. SwsContext *vctx = NULL;
  173. int outWidth = ;
  174. int outHeight = ;
  175. char *rgb = new char[**];
  176. char *pcm = new char[**];
  177.  
  178. //音频重采样上下文初始化
  179. SwrContext *actx = swr_alloc();
  180. actx = swr_alloc_set_opts(actx,
  181. av_get_default_channel_layout(),
  182. AV_SAMPLE_FMT_S16,ac->sample_rate,
  183. av_get_default_channel_layout(ac->channels),
  184. ac->sample_fmt,ac->sample_rate,
  185. , );
  186. re = swr_init(actx);
  187. if(re != )
  188. {
  189. LOGW("swr_init failed!");
  190. }
  191. else
  192. {
  193. LOGW("swr_init success!");
  194. }
  195.  
  196. //显示窗口初始化
  197. ANativeWindow *nwin = ANativeWindow_fromSurface(env,surface);
  198. ANativeWindow_setBuffersGeometry(nwin,outWidth,outHeight,WINDOW_FORMAT_RGBA_8888);
  199. ANativeWindow_Buffer wbuf;
  200.  
  201. for(;;)
  202. {
  203. //超过三秒
  204. if(GetNowMs() - start >= )
  205. {
  206. LOGW("now decode fps is %d",frameCount/);
  207. start = GetNowMs();
  208. frameCount = ;
  209. }
  210.  
  211. int re = av_read_frame(ic,pkt);
  212. if(re != )
  213. {
  214.  
  215. LOGW("读取到结尾处!");
  216. int pos = * r2d(ic->streams[videoStream]->time_base);
  217. av_seek_frame(ic,videoStream,pos,AVSEEK_FLAG_BACKWARD|AVSEEK_FLAG_FRAME );
  218. continue;
  219. }
  220.  
  221. AVCodecContext *cc = vc;
  222. if(pkt->stream_index == audioStream)
  223. cc=ac;
  224.  
  225. //发送到线程中解码
  226. re = avcodec_send_packet(cc,pkt);
  227. //清理
  228. int p = pkt->pts;
  229. av_packet_unref(pkt);
  230.  
  231. if(re != )
  232. {
  233. LOGW("avcodec_send_packet failed!");
  234. continue;
  235. }
  236.  
  237. for(;;)
  238. {
  239. re = avcodec_receive_frame(cc,frame);
  240. if(re !=)
  241. {
  242. //LOGW("avcodec_receive_frame failed!");
  243. break;
  244. }
  245. //LOGW("avcodec_receive_frame %lld",frame->pts);
  246. //如果是视频帧
  247. if(cc == vc)
  248. {
  249. frameCount++;
  250. vctx = sws_getCachedContext(vctx,
  251. frame->width,
  252. frame->height,
  253. (AVPixelFormat)frame->format,
  254. outWidth,
  255. outHeight,
  256. AV_PIX_FMT_RGBA,
  257. SWS_FAST_BILINEAR,
  258. ,,
  259. );
  260. if(!vctx)
  261. {
  262. LOGW("sws_getCachedContext failed!");
  263. }
  264. else
  265. {
  266. uint8_t *data[AV_NUM_DATA_POINTERS] = {};
  267. data[] =(uint8_t *)rgb;
  268. int lines[AV_NUM_DATA_POINTERS] = {};
  269. lines[] = outWidth * ;
  270. int h = sws_scale(vctx,
  271. (const uint8_t **)frame->data,
  272. frame->linesize,,
  273. frame->height,
  274. data,lines);
  275. LOGW("sws_scale = %d",h);
  276. if(h > )
  277. {
  278. ANativeWindow_lock(nwin,&wbuf,);
  279. uint8_t *dst = (uint8_t*)wbuf.bits;
  280. memcpy(dst,rgb,outWidth*outHeight*);
  281. ANativeWindow_unlockAndPost(nwin);
  282. }
  283. }
  284.  
  285. }
  286. else //音频
  287. {
  288. uint8_t *out[] = {};
  289. out[] = (uint8_t*) pcm;
  290.  
  291. //音频重采样
  292. int len = swr_convert(actx,out,
  293. frame->nb_samples,
  294. (const uint8_t**)frame->data,
  295. frame->nb_samples);
  296. LOGW("swr_convert = %d",len);
  297. }
  298.  
  299. }
  300.  
  301. //////////////////////
  302.  
  303. }
  304. delete rgb;
  305. delete pcm;
  306.  
  307. //关闭上下文
  308. avformat_close_input(&ic);
  309.  
  310. env->ReleaseStringUTFChars(url_, path);
  311. }

FFmpeg(五) 重采样相关函数理解的更多相关文章

  1. FFmpeg(四) 像素转换相关函数理解

    一.基本流程 1.sws_getCachedContext();//得到像素转换的上下文 2.sws_scale()://进行转换 二.函数说明 1.SwsContext *vctx = NULL;  ...

  2. FFmpeg(三) 编解码相关函数理解

    一.编解码基本流程 主要流程: 打开视频解码器(音频一样) 软解码.硬解码 进行编解码 下面先来看打开视频解码器 ①avcodec_register_all()//初始化解码 ②先找到解码器. 找解码 ...

  3. FFmpeg(二) 解封装相关函数理解

    一.解封装基本流程 ①av_register_All()////初始化解封装,注册解析和封装的格式. ②avformat_netword_init()//初始化网络,解析rtsp协议 ③avforma ...

  4. Java提高班(五)深入理解BIO、NIO、AIO

    导读:本文你将获取到:同/异步 + 阻/非阻塞的性能区别:BIO.NIO.AIO 的区别:理解和实现 NIO 操作 Socket 时的多路复用:同时掌握 IO 最底层最核心的操作技巧. BIO.NIO ...

  5. (原)ffmpeg过滤器开发和理解

    最近学习了ffmpeg关于filter过滤器的开发,关于中间的几个相关概念,我们先放在简单介绍一下: AVFilterGraph:几乎完全等同与directShow中的fitlerGraph,代表一串 ...

  6. 《深入理解Android 卷III》第五章 深入理解Android输入系统

    <深入理解Android 卷III>即将公布.作者是张大伟.此书填补了深入理解Android Framework卷中的一个主要空白.即Android Framework中和UI相关的部分. ...

  7. Golang 入门系列(十五)如何理解go的并发?

    前面已经讲过很多Golang系列知识,感兴趣的可以看看以前的文章,https://www.cnblogs.com/zhangweizhong/category/1275863.html, 接下来要说的 ...

  8. Deep learning:四十五(maxout简单理解)

    maxout出现在ICML2013上,作者Goodfellow将maxout和dropout结合后,号称在MNIST, CIFAR-10, CIFAR-100, SVHN这4个数据上都取得了start ...

  9. FFMpeg音频重采样和视频格式转

    一.视频像素和尺寸转换函数 1.sws_getContext : 像素格式上下文  --------------->多副图像(多路视频)进行转换同时显示 2.struct SwsContext  ...

随机推荐

  1. codeforce#483div2D-XOR-pyramid+DP

    题意:求给定区间中最大的连续异或和: 思路:DP的思想,先dp求出每个区间的异或和,再dp更新成当前这个dp[i][j]和dp[i-1][j].dp[i-1][j+1]中的最大值: 这样可以保证是同一 ...

  2. codeforces 814 C. An impassioned circulation of affection(二分+思维)

    题目链接:http://codeforces.com/contest/814/problem/C 题意:给出一串字符串然后q个询问,问替换掉将m个字符替换为字符c,能得到的最长的连续的字符c是多长 题 ...

  3. Codeforces 734D. Anton and Chess(模拟)

    Anton likes to play chess. Also, he likes to do programming. That is why he decided to write the pro ...

  4. poj 2777 Count Color(线段树(有点意思))

    题目链接 http://poj.org/problem?id=2777 题意:题意是有L个单位长的画板,T种颜色,O个操作.画板初始化为颜色1.操作C讲l到r单位之间的颜色变为c,操作P查询l到r单位 ...

  5. d3.js 制作简单的贪吃蛇

    d3.js是一个不错的可视化框架,同时对于操作dom也是十分方便的.今天我们使用d3.js配合es6的类来制作一个童年小游戏–贪吃蛇.话不多说先上图片. 1. js snaker类 class Sna ...

  6. 【Offer】[36] 【二叉搜索树与双向链表】

    题目描述 思路分析 测试用例 Java代码 代码链接 题目描述 输入一棵二叉搜索树,将该二叉搜索树转换成一个排序的双向链表.要求不能创建任何新的节点,只能调整树中节点指针的指向.比如,输入下图中左边的 ...

  7. 5.cookie每个参数的意义和作用以及工作原理?

    cookie主要参数有: (1)expires 过期时间 (2)path cookie存放路径 (3)domain 域名 同域名下可访问 (4)Set-Cookie: name cookie名称

  8. Dijkstra算法详细(单源最短路径算法)

    介绍 对于dijkstra算法,很多人可能感觉熟悉而又陌生,可能大部分人比较了解bfs和dfs,而对dijkstra和floyd算法可能知道大概是图论中的某个算法,但是可能不清楚其中的作用和原理,又或 ...

  9. Net基础篇_学习笔记_第十天_方法_方法的练习

    using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.T ...

  10. Spring Boot应用上传文件时报错

    问题描述 Spring Boot应用(使用默认的嵌入式Tomcat)在上传文件时,偶尔会出现上传失败的情况,后台报错日志信息如下:"The temporary upload location ...