亚洲激情专区-91九色丨porny丨老师-久久久久久久女国产乱让韩-国产精品午夜小视频观看

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Android 音視頻深入 十一 FFmpeg和AudioTrack播放聲音(附源碼下載)

發布時間:2020-06-28 04:13:00 來源:網絡 閱讀:1395 作者:qq5a70561b29238 欄目:移動開發

項目地址,求star
https://github.com/979451341/AudioVideoStudyCodeTwo/tree/master/FFmpeg%E6%92%AD%E6%94%BE%E9%9F%B3%E4%B9%90%EF%BC%88%E4%BF%9D%E7%A8%8B%E5%BA%8F%E4%B8%8D%E6%AD%BB%EF%BC%89

這個是FFmpeg解碼出音頻,給AudioTrack播放,這回才算是java與c語言之間合作

這回我們將會從c++里調用java函數,下面就是關于c++使用AudioTrack的代碼

private AudioTrack audioTrack;
//    這個方法  是C進行調用  通道數
public void createTrack(int sampleRateInHz,int nb_channals) {

    int channaleConfig;//通道數
    if (nb_channals == 1) {
        channaleConfig = AudioFormat.CHANNEL_OUT_MONO;
    } else if (nb_channals == 2) {
        channaleConfig = AudioFormat.CHANNEL_OUT_STEREO;
    }else {
        channaleConfig = AudioFormat.CHANNEL_OUT_MONO;
    }
    int buffersize=AudioTrack.getMinBufferSize(sampleRateInHz,
            channaleConfig, AudioFormat.ENCODING_PCM_16BIT);
    audioTrack = new AudioTrack(AudioManager.STREAM_MUSIC,sampleRateInHz,channaleConfig,
            AudioFormat.ENCODING_PCM_16BIT,buffersize,AudioTrack.MODE_STREAM);
    audioTrack.play();
}

//C傳入音頻數據
public void playTrack(byte[] buffer, int lenth) {
    if (audioTrack != null && audioTrack.getPlayState() == AudioTrack.PLAYSTATE_PLAYING) {
        audioTrack.write(buffer, 0, lenth);
    }
}

我們再來看看c++的代碼

首先注冊組件,然后得到音頻流

 av_register_all();
AVFormatContext *pFormatCtx = avformat_alloc_context();
//open
if (avformat_open_input(&pFormatCtx, input, NULL, NULL) != 0) {
    LOGE("%s","打開輸入視頻文件失敗");
    return;
}
//獲取視頻信息
if(avformat_find_stream_info(pFormatCtx,NULL) < 0){
    LOGE("%s","獲取視頻信息失敗");
    return;
}
int audio_stream_idx=-1;
int i=0;
for (int i = 0; i < pFormatCtx->nb_streams; ++i) {
    if (pFormatCtx->streams[i]->codec->codec_type == AVMEDIA_TYPE_AUDIO) {
        LOGE("  找到音頻id %d", pFormatCtx->streams[i]->codec->codec_type);
        audio_stream_idx=i;
        break;
    }
}

獲取×××

//獲取×××上下文
AVCodecContext *pCodecCtx=pFormatCtx->streams[audio_stream_idx]->codec;
//獲取×××
AVCodec *pCodex = avcodec_find_decoder(pCodecCtx->codec_id);
//打開×××
if (avcodec_open2(pCodecCtx, pCodex, NULL)<0) {
}

設置緩存區,保存解碼前后的數據

//申請avpakcet,裝解碼前的數據
AVPacket *packet = (AVPacket *)av_malloc(sizeof(AVPacket));
//申請avframe,裝解碼后的數據
AVFrame *frame = av_frame_alloc();

設置解碼出的聲音一系列的屬性,比如:單聲道、雙聲道、采集點大小、采集率,還可以在這里對聲音添加特效,

//得到SwrContext ,進行重采樣,具體參考http://blog.csdn.net/jammg/article/details/52688506
SwrContext *swrContext = swr_alloc();
//緩存區
uint8_t *out_buffer = (uint8_t *) av_malloc(44100 * 2);

//輸出的聲道布局(立體聲)
uint64_t out_ch_layout=AV_CH_LAYOUT_STEREO;
//輸出采樣位數 16位
enum AVSampleFormat out_formart=AV_SAMPLE_FMT_S16;
//輸出的采樣率必須與輸入相同
int out_sample_rate = pCodecCtx->sample_rate;

//swr_alloc_set_opts將PCM源文件的采樣格式轉換為自己希望的采樣格式
swr_alloc_set_opts(swrContext, out_ch_layout, out_formart, out_sample_rate,
pCodecCtx->channel_layout, pCodecCtx->sample_fmt, pCodecCtx->sample_rate, 0,
NULL);

swr_init(swrContext);

// 獲取通道數 2
int out_channer_nb = av_get_channel_layout_nb_channels(AV_CH_LAYOUT_STEREO);

通過反射能夠運行java函數

// 反射得到Class類型
jclass david_player = env->GetObjectClass(instance);
// 反射得到createAudio方法
jmethodID createAudio = env->GetMethodID(david_player, "createTrack", "(II)V");
// 反射調用createAudio
env->CallVoidMethod(instance, createAudio, 44100, out_channer_nb);
jmethodID audio_write = env->GetMethodID(david_player, "playTrack", "([BI)V");

在一邊解碼的時候一邊給數據給AudioTrack播放

while (av_read_frame(pFormatCtx, packet) >= 0) {
    if (packet->stream_index == audio_stream_idx) {

// 解碼 mp3 編碼格式frame----pcm frame
avcodec_decode_audio4(pCodecCtx, frame, &got_frame, packet);
if (got_frame) {
LOGE("解碼");
swr_convert(swrContext, &out_buffer, 44100 * 2, (const uint8_t *) frame->data, frame->nb_samples);
// 緩沖區的大小
int size = av_samples_get_buffer_size(NULL, out_channer_nb, frame->nb_samples,
AV_SAMPLE_FMT_S16, 1);
jbyteArray audio_sample_array = env->NewByteArray(size);
env->SetByteArrayRegion(audio_sample_array, 0, size, (const jbyte
) out_buffer);
env->CallVoidMethod(instance, audio_write, audio_sample_array, size);
env->DeleteLocalRef(audio_sample_array);
}
}
}

釋放資源

av_frame_free(&frame);
swr_free(&swrContext);
avcodec_close(pCodecCtx);
avformat_close_input(&pFormatCtx);
env->ReleaseStringUTFChars(input_, input);

FFmpeg只是音視頻處理的工具,他沒有播放視頻和音頻的能力,所以我們需要SurfaceView顯示視頻,AudioTrack播放聲音,而且OpenGLES也能播放聲音,這個后面說

下一次就是說如何將視頻的聲音給聽換掉,也就是將音視頻的解碼和編碼都來搞一次

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

林州市| 东丰县| 恭城| 巴里| 尚志市| 恩平市| 汝州市| 揭东县| 故城县| 蓬安县| 民丰县| 永宁县| 古浪县| 广河县| 太仓市| 特克斯县| 五华县| 澜沧| 阳春市| 聊城市| 晋中市| 房山区| 西和县| 抚宁县| 中牟县| 庆元县| 松滋市| 安西县| 平陆县| 嘉义市| 枞阳县| 新晃| 泸溪县| 安顺市| 天长市| 胶南市| 桑日县| 阳谷县| 海安县| 河南省| 大港区|