FFmpeg音訊解碼

weixin_33978044發表於2018-03-23

//第一步:組冊元件

    av_register_all();

    //第二步:開啟封裝格式->開啟檔案

    //引數一:封裝格式上下文

    //作用:儲存整個視訊資訊(解碼器、編碼器等等...)

    //資訊:位元速率、幀率等...

    AVFormatContext* avformat_context = avformat_alloc_context();

    //引數二:視訊路徑

    const char *url = [jinFilePath UTF8String];

    //引數三:指定輸入的格式

    //引數四:設定預設引數

    int avformat_open_input_result = avformat_open_input(&avformat_context, url,NULL, NULL);

    if (avformat_open_input_result !=0){

        NSLog("開啟檔案失敗");

        return;

    }

    NSLog("開啟檔案成功");

    //第三步:拿到視訊基本資訊

    //引數一:封裝格式上下文

    //引數二:指定預設配置

    int avformat_find_stream_info_result = avformat_find_stream_info(avformat_context,NULL);

    if (avformat_find_stream_info_result <0){

        NSLog("查詢失敗");

        return;

    }

    //第四步:查詢音訊解碼器

    //第一點:查詢音訊流索引位置

    int av_stream_index = -1;

    for (int i =0; i < avformat_context->nb_streams; ++i) {

        //判斷是否是音訊流

        if (avformat_context->streams[i]->codec->codec_type == AVMEDIA_TYPE_AUDIO){

            //AVMEDIA_TYPE_AUDIO->表示音訊型別

            av_stream_index = i;

            break;

        }

    }

    //第二點:獲取音訊解碼器上下文

    AVCodecContext * avcodec_context = avformat_context->streams[av_stream_index]->codec;

    //第三點:獲得音訊解碼器

    AVCodec *avcodec = avcodec_find_decoder(avcodec_context->codec_id);

    if (avcodec ==NULL){

        NSLog("查詢音訊解碼器失敗");

        return;

    }

    //第五步:開啟音訊解碼器

    int avcodec_open2_result = avcodec_open2(avcodec_context, avcodec,NULL);

    if (avcodec_open2_result !=0){

        NSLog("開啟音訊解碼器失敗");

        return;

    }

    //第六步:讀取音訊壓縮資料->迴圈讀取

    //建立音訊壓縮資料幀

    //音訊壓縮資料->acc格式、mp3格式

    AVPacket* avpacket = (AVPacket*)av_malloc(sizeof(AVPacket));

    //建立音訊取樣資料幀

    AVFrame* avframe = av_frame_alloc();

    //音訊取樣上下文->開闢了一快記憶體空間->pcm格式等...

    //設定引數

    //引數一:音訊取樣資料上下文

    //上下文:儲存音訊資訊(記錄)->目錄

    SwrContext* swr_context = swr_alloc();

    //引數二:out_ch_layout->輸出聲道佈局型別(立體聲、環繞聲、機器人等等...)

    //立體聲

    int64_t out_ch_layout = AV_CH_LAYOUT_STEREO;

//    int out_ch_layout = av_get_default_channel_layout(avcodec_context->channels);

    //引數三:out_sample_fmt->輸出取樣精度->編碼

    //直接指定

    int out_sample_fmt = AV_SAMPLE_FMT_S16;

    //例如:取樣精度8位 = 1位元組,取樣精度16位 = 2位元組

    //引數四:out_sample_rate->輸出取樣率(44100HZ)

    int out_sample_rate = avcodec_context->sample_rate;

    //引數五:in_ch_layout->輸入聲道佈局型別

    int64_t in_ch_layout = av_get_default_channel_layout(avcodec_context->channels);

    //引數六:in_sample_fmt->輸入取樣精度

    AVSampleFormat in_sample_fmt = avcodec_context->sample_fmt;

    //引數七:in_sample_rate->輸入取樣率

    int in_sample_rate = avcodec_context->sample_rate;

    //引數八:log_offset->log日誌->從那裡開始統計

    int log_offset =0;

    //引數九:log_ctx->log上下文

    swr_alloc_set_opts(swr_context,

                       out_ch_layout,

                       out_sample_fmt,

                       out_sample_rate,

                       in_ch_layout,

                       in_sample_fmt,

                       in_sample_rate,

                       log_offset,NULL);

    //初始化音訊取樣資料上下文

    swr_init(swr_context);

    //輸出音訊取樣資料

    //緩衝區大小 = 取樣率(44100HZ) * 取樣精度(16位 = 2位元組)

    int MAX_AUDIO_SIZE =44100 * 2;

    uint8_t *out_buffer = (uint8_t *)av_malloc(MAX_AUDIO_SIZE);

    //輸出聲道數量

    int out_nb_channels = av_get_channel_layout_nb_channels(out_ch_layout);

    int audio_decode_result =0;

    //開啟檔案

    const char *coutFilePath = [joutFilePath UTF8String];

    FILE* out_file_pcm = fopen(coutFilePath,"wb");

    if (out_file_pcm ==NULL){

        NSLog("開啟音訊輸出檔案失敗");

        return;

    }

    int current_index =0;

    //計算:4分鐘一首歌曲 = 240ms = 4MB

    //現在音訊時間:24ms->pcm格式->8.48MB

    //如果是一首4分鐘歌曲->pcm格式->85MB

    while (av_read_frame(avformat_context, avpacket) >=0){

        //讀取一幀音訊壓縮資料成功

        //判定是否是音訊流

        if (avpacket->stream_index == av_stream_index){

            //第七步:音訊解碼

            //1、傳送一幀音訊壓縮資料包->音訊壓縮資料幀

            avcodec_send_packet(avcodec_context, avpacket);

            //2、解碼一幀音訊壓縮資料包->得到->一幀音訊取樣資料->音訊取樣資料幀

            audio_decode_result = avcodec_receive_frame(avcodec_context, avframe);

            if (audio_decode_result ==0){

                //表示音訊壓縮資料解碼成功

                //3、型別轉換(音訊取樣資料格式有很多種型別)

                //我希望我們的音訊取樣資料格式->pcm格式->保證格式統一->輸出PCM格式檔案

                //swr_convert:表示音訊取樣資料型別格式轉換器

                //引數一:音訊取樣資料上下文

                //引數二:輸出音訊取樣資料

                //引數三:輸出音訊取樣資料->大小

                //引數四:輸入音訊取樣資料

                //引數五:輸入音訊取樣資料->大小

                swr_convert(swr_context,

                            &out_buffer,

                            MAX_AUDIO_SIZE,

                            (const uint8_t **)avframe->data,

                            avframe->nb_samples);

                //4、獲取緩衝區實際儲存大小

                //引數一:行大小

                //引數二:輸出聲道數量

                //引數三:輸入大小

                int nb_samples = avframe->nb_samples;

                //引數四:輸出音訊取樣資料格式

                //引數五:位元組對齊方式

                int out_buffer_size = av_samples_get_buffer_size(NULL,

                                           out_nb_channels,

                                           nb_samples,

                                           out_sample_fmt,

                                           1);

                //5、寫入檔案(你知道要寫多少嗎?)

                fwrite(out_buffer,1, out_buffer_size, out_file_pcm);

                current_index++;

                NSLog("當前音訊解碼第%d幀", current_index);

            }

        }

    }

    //第八步:釋放記憶體資源,關閉音訊解碼器

    fclose(out_file_pcm);

    av_packet_free(&avpacket);

    swr_free(&swr_context);

    av_free(out_buffer);

    av_frame_free(&avframe);

    avcodec_close(avcodec_context);

    avformat_close_input(&avformat_context);

相關文章