美文网首页
FFmpeg笔记(七)-- 视频转为多张图片

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

作者: rookiesss | 来源:发表于2020-05-19 17:06 被阅读0次
    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笔记(七)-- 视频转为多张图片

          本文链接:https://www.haomeiwen.com/subject/kfkpohtx.html