第四节 C++多线程解码音频数据

要实现的功能:FFmpeg在C++子线程中解码音频数据,得到数据包AVPacket。

AVPacket:存放原始音频/视频的压缩包


image.png

image.png

image.png

代码实现:我们定义两个类HFFmpeg和HAudio分别用于实现解码相关操作和存放解码后的音频的一些相关信息

先看个问题,在写代码的过程中发现的

仔细比较这两种使用多线程的代码的区别

主要就是pthread_t *pthread这种写法只是定义了一个存放pthread_t的地址的指针,并没有分配存放pthread_t的内存空间

image.png

image.png

代码实现

目录结构


image.png

Log.h
日志相关,方便后面多个cpp使用

//
// Created by 霍振鹏 on 2018/10/19.
//

#ifndef VOICEPLAYER_LOG_H
#define VOICEPLAYER_LOG_H

#endif //VOICEPLAYER_LOG_H
#include <android/log.h>
#define LOGI(FORMAT,...) __android_log_print(ANDROID_LOG_INFO,"VoicePlayer",FORMAT,##__VA_ARGS__);
//打印日志方便
#define LOG_DEBUG true

HAudio.h

//
// Created by 霍振鹏 on 2018/10/19.
//

#ifndef VOICEPLAYER_HAUDIO_H
#define VOICEPLAYER_HAUDIO_H

extern "C"
{
#include "libavcodec/avcodec.h"
};

class HAudio{
public:
    int streamIndex=-1;
    AVCodecParameters *avCodecParameters=NULL;
    AVCodecContext *avCodecContext=NULL;

public:
    HAudio();
    ~HAudio();
};

#endif //VOICEPLAYER_HAUDIO_H

HAudio.cpp

//
// Created by 霍振鹏 on 2018/10/19.
//

#include "HAudio.h"

HAudio::HAudio() {

}

HAudio::~HAudio() {

}

HFFmpeg.h

//
// Created by 霍振鹏 on 2018/10/19.
//

#include "CallBackJava.h"
#include <pthread.h>
// sleep 的头文件
#include <unistd.h>
#include "Log.h"
#include "HAudio.h"


extern "C"
{
#include <libavformat/avformat.h>
}

#ifndef VOICEPLAYER_HFFMPEG_H
#define VOICEPLAYER_HFFMPEG_H

#endif //VOICEPLAYER_HFFMPEG_H

class HFFmpeg{
public:
    //记得全部置为NULL
    //需要解码的音频文件的地址
    const char* url=NULL;
    //一般可能都需要回调Java层代码
    CallBackJava *callBackJava=NULL;

    //解码线程
    pthread_t pthread_decode=NULL;

    AVFormatContext *avFormatContext=NULL;

    HAudio *hAudio=NULL;



public:
    HFFmpeg(const char* url,CallBackJava *callBackJava);

    /**
     * 准备解码
     */
    void prepare();
    /**
     * 开始解码
     */
    void start();

    /**
     * 解码线程回调方法
     */
    void decode();

    ~HFFmpeg();
};

HFFmpeg.cpp

//
// Created by 霍振鹏 on 2018/10/19.
//

#include "HFFmpeg.h"
#include "HAudio.h"

HFFmpeg::~HFFmpeg() {

}

HFFmpeg::HFFmpeg(const char *url, CallBackJava *callBackJava) {

    this->url=url;
    this->callBackJava=callBackJava;
}


void * decodeFFmpeg(void * data)
{
    HFFmpeg *hfFmpeg= (HFFmpeg *) data;
    hfFmpeg->decode();
    pthread_exit(&hfFmpeg->pthread_decode);


}

void HFFmpeg::prepare() {

    //初始化线程,开始解码
    pthread_create(&pthread_decode,NULL,decodeFFmpeg,this);

}

void HFFmpeg::start() {

}

void HFFmpeg::decode() {
    av_register_all();
    avformat_network_init();
    //打开本地文件或者网络流
    avFormatContext=avformat_alloc_context();
    //0 on success
    int result=0;
    result=avformat_open_input(&avFormatContext,url,NULL,NULL);
    if(result!=0)
    {
        if(LOG_DEBUG)
        {
            LOGI("打开媒体文件失败");
        }
        return ;
    }
    //查找流信息return >=0 if OK
    result=avformat_find_stream_info(avFormatContext,NULL);
    if(result<0)
    {
        if(LOG_DEBUG)
        {
            LOGI("can not find streams from %s", url);
        }
        return ;
    }
    //查找音频流的索引
    for(int i=0;i<avFormatContext->nb_streams;i++)
    {
        if(avFormatContext->streams[i]->codecpar->codec_type==AVMEDIA_TYPE_AUDIO)
        {
            if(hAudio==NULL)
            {
                hAudio=new HAudio();
                hAudio->streamIndex=i;
                hAudio->avCodecParameters=avFormatContext->streams[i]->codecpar;
            }

        }
    }
    //获取音频解码器
    AVCodec *avCodec=avcodec_find_decoder(hAudio->avCodecParameters->codec_id);
    if(avCodec==NULL)
    {
        if(LOG_DEBUG)
        {
            LOGI("获取音频解码器失败")
        }
        return ;
    }

    //创建解码器上下文
    hAudio->avCodecContext=avcodec_alloc_context3(avCodec);
    if(hAudio->avCodecContext==NULL)
    {
        if(LOG_DEBUG)
        {
            LOGI("创建解码器上下文失败");
        }
        return ;
    }
    //将音频流信息拷贝到新的AVCodecContext结构体中  return >= 0 on success
    if(avcodec_parameters_to_context(hAudio->avCodecContext,hAudio->avCodecParameters)<0)
    {
        if(LOG_DEBUG)
        {
            LOGI("拷贝失败");
        }
        return ;
    }
    //打开解码器  zero on success
    if(avcodec_open2(hAudio->avCodecContext,avCodec,NULL)!=0)
    {
        if(LOG_DEBUG)
        {
            LOGI("打开解码器失败");
        }
        return ;
    }
    const char* msg="解码初始化完成";
    callBackJava->onPrepared(0,msg);
}

CallBackJava.h

//
// Created by 霍振鹏 on 2018/10/19.
//
#include "jni.h"
#ifndef VOICEPLAYER_CALLBACKJAVA_H
#define VOICEPLAYER_CALLBACKJAVA_H

class CallBackJava
{
public:
    JavaVM *javaVM;
    JNIEnv *jniEnv;
    jobject instance;
    jmethodID jmd;
    //解码初始化完成的回调
    jmethodID jmd_prepared;

public:
    CallBackJava(JavaVM *vm,JNIEnv *env,jobject job);
    ~CallBackJava();
    /**
     * @param type 1主线程,0子线程
     * @param code
     * @param msg
     */
    void onError(int type,int code, const char *msg);

    void onPrepared(int type,const char *msg);

};

#endif //VOICEPLAYER_CALLBACKJAVA_H


CallBackJava.cpp

//
// Created by 霍振鹏 on 2018/10/19.
//

#include "CallBackJava.h"
#include "Log.h"

CallBackJava::CallBackJava(JavaVM *vm, JNIEnv *env, jobject job) {

    javaVM=vm;
    jniEnv=env;
    instance=job;
    jclass jcl=env->GetObjectClass(job);
    this->jmd=env->GetMethodID(jcl,"onError","(ILjava/lang/String;)V");
    this->jmd_prepared=env->GetMethodID(jcl,"onPrepared","(Ljava/lang/String;)V");

}

CallBackJava::~CallBackJava() {

    LOGI("析构函数执行了");
}

void CallBackJava::onError(int type, int code, const char *msg) {
    if(type==0)
    {
        //子线程
        //这儿会重新给JNIEnv赋值
        javaVM->AttachCurrentThread(&jniEnv,0);
        jstring jsr=jniEnv->NewStringUTF(msg);
        jniEnv->CallVoidMethod(instance,jmd,code,jsr);
        jniEnv->DeleteLocalRef(jsr);
        javaVM->DetachCurrentThread();

    }
    else if(type==1)
    {
        jstring jsr=jniEnv->NewStringUTF(msg);
        //主线程
        jniEnv->CallVoidMethod(instance,jmd,code,jsr);
        jniEnv->DeleteLocalRef(jsr);

    }

}

void CallBackJava::onPrepared(int type,const char *msg) {
    if(type==0)
    {
        //子线程
        //这儿会重新给JNIEnv赋值
        javaVM->AttachCurrentThread(&jniEnv,0);
        jstring jsr=jniEnv->NewStringUTF(msg);
        jniEnv->CallVoidMethod(instance,jmd_prepared,jsr);
        jniEnv->DeleteLocalRef(jsr);
        javaVM->DetachCurrentThread();

    }
    else if(type==1)
    {
        jstring jsr=jniEnv->NewStringUTF(msg);
        //主线程
        jniEnv->CallVoidMethod(instance,jmd_prepared,jsr);
        jniEnv->DeleteLocalRef(jsr);

    }
}

native-lib.cpp

#include <jni.h>
#include <string>
#include "CallBackJava.h"
#include "Log.h"
#include "HFFmpeg.h"



JavaVM *javaVM;
JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM *vm, void* reserved)
{
    JNIEnv *env;
    javaVM = vm;
    if(vm->GetEnv((void **) &env, JNI_VERSION_1_6) != JNI_OK)
    {
        return -1;
    }
    return JNI_VERSION_1_6;
}

JNIEXPORT void JNICALL JNI_OnUnload (JavaVM *vm, void* reserved)
{


}


CallBackJava *callBackJava=NULL;
HFFmpeg *hfFmpeg=NULL;
/**
 * 解码为AVPacket
 */
extern "C"
JNIEXPORT void JNICALL
Java_com_example_voicelib_Player_startDecode(JNIEnv *env, jobject instance, jstring path_) {
    const char *path = env->GetStringUTFChars(path_, 0);

    if(hfFmpeg==NULL)
    {
        if(callBackJava==NULL)
        {
            callBackJava=new CallBackJava(javaVM,env,env->NewGlobalRef(instance));
        }

        hfFmpeg=new HFFmpeg(path,callBackJava);
        hfFmpeg->prepare();
    }



//    env->ReleaseStringUTFChars(path_, path);
}

Player.java

package com.example.voicelib;

import android.util.Log;

/**
 * 作者 huozhenpeng
 * 日期 2018/10/18
 * 邮箱 huohacker@sina.com
 */

public class Player {

    // Used to load the 'native-lib' library on application startup.
    static {
        System.loadLibrary("native-lib");
        System.loadLibrary("avutil-55");
        System.loadLibrary("swresample-2");
        System.loadLibrary("avcodec-57");
        System.loadLibrary("avformat-57");
        System.loadLibrary("swscale-4");
        System.loadLibrary("postproc-54");
        System.loadLibrary("avfilter-6");
        System.loadLibrary("avdevice-57");
    }

    public void onError(int code,String msg)
    {
        Log.e("VoicePlayer","code:"+code+"msg:"+msg);

    }

    public void onPrepared(String msg)
    {
        Log.e("VoicePlayer","[Java]msg:"+msg);
    }

    /**
     * 开始解码
     */
    public native void startDecode(String path);



}

调用:

 public void startDecode(View view) {
        Player player=new Player();

        player.startDecode(Environment.getExternalStorageDirectory()+"/lame.mp3");
    }

输出结果:
image.png
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,504评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,434评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,089评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,378评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,472评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,506评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,519评论 3 413
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,292评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,738评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,022评论 2 329
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,194评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,873评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,536评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,162评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,413评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,075评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,080评论 2 352

推荐阅读更多精彩内容