FFmpeg笔记(七)-- 视频转为多张图片

1.初始化格式上下文。
    AVFormatContext *formatCtx = avformat_alloc_context();
    if (avformat_open_input(&formatCtx, [videoPath UTF8String], NULL, NULL) < 0) {
        NSLog(@"不能打开输入文件");
        return nil;
    }
    
    if (avformat_find_stream_info(formatCtx, NULL) < 0) {
        NSLog(@"没有流信息");
        return nil;
    }
2.获取视频流。
    int stream_index = av_find_best_stream(formatCtx, AVMEDIA_TYPE_VIDEO, -1, -1, NULL, 0);
    if (stream_index < 0) {
        NSLog(@"没有发现视频流");
        return nil;
    }
    AVStream *videoStream = formatCtx->streams[stream_index];
3.获取解码器、解码上下文。
    AVCodecContext *codeCtx = avcodec_alloc_context3(NULL);
    if (!codeCtx) {
        NSLog(@"没有发现解码上下文");
        return nil;
    }

    if ((avcodec_parameters_to_context(codeCtx, videoStream->codecpar)) < 0) {
        NSLog(@"复制解码参数失败");
        return nil;
    }
    avcodec_open2(codeCtx, codec, NULL);
4.解码出视频帧。
    AVPacket *packet = av_packet_alloc();
    AVFrame *frame = av_frame_alloc();
    int frameCount = 0;
    NSMutableArray *imgArray = [NSMutableArray array];
    while (av_read_frame(formatCtx, packet) >= 0) {
        if (packet->stream_index == stream_index) {
            int send_result = avcodec_send_packet(codeCtx, packet);
            int receive_result = 0;
            if (send_result == AVERROR(EAGAIN)) {
                while (receive_result != AVERROR(EAGAIN)) {
                    receive_result = avcodec_receive_frame(codeCtx, frame);
                    if (receive_result == 0) {
                        if (onlyIFrame && frame->pict_type != AV_PICTURE_TYPE_I) {
                            continue;
                        }

                        NSString *imagePath = [DYMediaTools creatFile:[NSString stringWithFormat:@"img%@",@(frameCount++)] ofType:@"jpg"];
                        int r = frameToJPG(frame, [imagePath UTF8String]);
                        if (r == 0){
                            [imgArray addObject:[UIImage imageWithContentsOfFile:imagePath]];
                        }
                    }
                }
            } else {
                receive_result = avcodec_receive_frame(codeCtx, frame);
                if (receive_result == AVERROR(EAGAIN)) {
                    continue;
                } else if (receive_result == 0) {
                    if (onlyIFrame && frame->pict_type != AV_PICTURE_TYPE_I) {
                        continue;
                    }
        
                    NSString *imagePath = [DYMediaTools creatFile:[NSString stringWithFormat:@"img%@",@(frameCount++)] ofType:@"jpg"];
                    int r = frameToJPG(frame, [imagePath UTF8String]);
                    if (r == 0){
                        [imgArray addObject:[UIImage imageWithContentsOfFile:imagePath]];
                    }
                }
            }
        }
        av_packet_unref(packet);
    }
5.将视频帧编码为jpg。
int frameToJPG(AVFrame *videoFrame,const char *imageName) {
    
    AVFormatContext *formatCtx = avformat_alloc_context();
    avformat_alloc_output_context2(&formatCtx, NULL, NULL, imageName);

    if (avio_open(&formatCtx->pb, imageName, AVIO_FLAG_READ_WRITE) < 0) {
        NSLog(@"打开输出文件失败");
        return -1;
    }

    AVStream *pAVStream = avformat_new_stream(formatCtx, 0);
    if (pAVStream == NULL) {
        return -1;
    }

    AVCodecParameters *parameters = pAVStream->codecpar;
    parameters->codec_id = formatCtx->oformat->video_codec;
    parameters->codec_type = AVMEDIA_TYPE_VIDEO;
    parameters->format = AV_PIX_FMT_YUVJ420P;
    parameters->width = videoFrame->width;
    parameters->height = videoFrame->height;

    AVCodec *codec = avcodec_find_encoder(pAVStream->codecpar->codec_id);

    if (!codec) {
        NSLog(@"没有发现编码器");
        return -1;
    }

    AVCodecContext *codectx = avcodec_alloc_context3(codec);
    if (!codectx) {
        NSLog(@"没有发现编码上下文");
        return -1;
    }

    if ((avcodec_parameters_to_context(codectx, pAVStream->codecpar)) < 0) {
        NSLog(@"复制编码参数失败");
        return -1;
    }

    codectx->time_base = (AVRational) {1, 25};

    if (avcodec_open2(codectx, codec, NULL) < 0) {
        NSLog(@"打开编码器失败");
        return -1;
    }

    int ret = avformat_write_header(formatCtx, NULL);
    if (ret < 0) {
        NSLog(@"头文件写入失败");
        return -1;
    }

    AVPacket *packet = av_packet_alloc();
    //分配足够的空间保证能一次接收完
    av_new_packet(packet, videoFrame->width * videoFrame->height * 3);
    ret = avcodec_send_frame(codectx, videoFrame);
    if (ret < 0) {
        NSLog(@"发送视频帧失败");
        return -1;
    }

    ret = avcodec_receive_packet(codectx, packet);
    if (ret < 0) {
        NSLog(@"接收视频帧失败");
        return -1;
    }

    ret = av_write_frame(formatCtx, packet);

    if (ret < 0) {
        NSLog(@"写入帧失败");
        return -1;
    }

    av_packet_free(&packet);
    av_write_trailer(formatCtx);
    
    avcodec_close(codectx);
    avio_close(formatCtx->pb);
    avformat_free_context(formatCtx);
    return 0;
}

你可能感兴趣的:(FFmpeg笔记(七)-- 视频转为多张图片)