用ffmpeg解码H264视频流

一、前言

最近在做直播(监控类)的项目,刚开始一窍不通,各种难啊,没办法,总得做啊,于是就查资料,一步一步,最后总算是做出来了,下面就先讲一下利用ffmpeg解码H264视频流这一块。首先在iOS平台配置ffmpeg就不再详解,具体请看:
https://cnbin.github.io/blog/2015/05/19/iospei-zhi-ffmpegkuang-jia/ 这篇博客,我在项目中使用的ffmpeg版本号是3.1,我在看上篇博客配置ffmpeg时,按步骤一步一步来,还是出现了.a文件是红色的情况,后来我就把.a文件重新导入项目中,总算是好了,真是不易啊。

二、解码H264视频流

1.首先创建一个文件专门用来解码,在DDH264Decoder.h文件中对外暴露以下三个方法:

#import <Foundation/Foundation.h>
#import <UIKit/UIKit.h>
#import "libavcodec/avcodec.h"

@interface DDH264Decoder : NSObject

/* 初始化解码器 */
- (BOOL)initH264DecoderWithWidth:(int)width height:(int)height;

/* 解码视频数据并且返回图片 */
- (void)H264decoderWithVideoData:(NSData *)VideoData completion:(void (^)(AVPicture picture))completion;

/* 释放解码器 */
- (void)releaseH264Decoder;

2.在.m文件中实现所暴露的方法

#import "DDH264Decoder.h"
#import "libswscale/swscale.h"
#include <libavformat/avformat.h>
#import <AVFoundation/AVFoundation.h>

@interface DDH264Decoder ()

@property (assign, nonatomic) AVFrame *frame;
@property (assign, nonatomic) AVCodec *codec;
@property (assign, nonatomic) AVCodecContext *codecCtx;
@property (assign, nonatomic) AVPacket packet;
@property (assign, nonatomic) AVFormatContext *formatCtx;

@end

@implementation DDH264Decoder

/**
 *  初始化视频解码器
 *
 *  @param width  宽度
 *  @param height 高度
 *
 *  @return YES:解码成功
 */
- (BOOL)initH264DecoderWithWidth:(int)width height:(int)height {
    
    av_register_all();
    
    avformat_network_init();
    self.codec = avcodec_find_decoder(AV_CODEC_ID_H264);
    av_init_packet(&_packet);
    if (self.codec != nil) {
       self.codecCtx = avcodec_alloc_context3(self.codec);
        
        // 每包一个视频帧
        self.codecCtx->frame_number = 1;
        self.codecCtx->codec_type = AVMEDIA_TYPE_VIDEO;
        
        // 视频的宽度和高度
        self.codecCtx->width = width;
        self.codecCtx->height = height;
        
        // 打开codec
        if (avcodec_open2(self.codecCtx, self.codec, NULL) >= 0) {
            self.frame = av_frame_alloc();
            if (self.frame == NULL) {
                return NO;
            }
        }
    }
    
    return (BOOL)self.frame;
}

/**
 *  视频解码
 *
 *  @param data 被解码视频数据
 *
 *  @return 图片
 */
- (void)H264decoderWithVideoData:(NSData *)VideoData completion:(void (^)(AVPicture))completion {
    @autoreleasepool {
        _packet.data = (uint8_t *)VideoData.bytes;
        _packet.size = (int)VideoData.length;
        
        int getPicture;
            avcodec_send_packet(_codecCtx, &_packet);
            getPicture = avcodec_receive_frame(self.codecCtx, self.frame);
             av_packet_unref(&_packet);
            if (getPicture == 0) {
                AVPicture picture;
                avpicture_alloc(&picture, AV_PIX_FMT_RGB24, self.codecCtx->width, self.codecCtx->height);
                
                struct SwsContext *img_convert_ctx = sws_getContext(self.codecCtx->width,
                                                                    self.codecCtx->height,
                                                                    AV_PIX_FMT_YUV420P,
                                                                    self.codecCtx->width,
                                                                    self.codecCtx->height,
                                                                    AV_PIX_FMT_RGB24,
                                                                    SWS_FAST_BILINEAR,
                                                                    NULL,
                                                                    NULL,
                                                                    NULL);
                // 图像处理
                sws_scale(img_convert_ctx, (const uint8_t* const*)self.frame->data, self.frame->linesize, 0, self.codecCtx->height, picture.data, picture.linesize);

                sws_freeContext(img_convert_ctx);
                img_convert_ctx = NULL;
                
                if (completion) {
                    completion(picture);
                }
                
                avpicture_free(&picture);
        }
    }
}

/**
 *  释放视频解码器
 */
- (void)releaseH264Decoder {
    if(self.codecCtx) {
        avcodec_close(self.codecCtx);
        avcodec_free_context(&_codecCtx);
       self.codecCtx = NULL;
    }
    
    if(self.frame) {
        av_frame_free(&_frame);
        self.frame = NULL;
    }
    av_packet_unref(&_packet);
}

注意:使用完后,一定要释放,要不然会内存泄漏。

3.在控制器中对解码后的视频数据进行处理
我是在另外一个文件中,对数据进行了另一层加工及处理,在这里只写在控制器中对解码后的数据进行显示。

- (void)dealAVPicture:(AVPicture)picture width:(int)width height:(int)height {
    CGBitmapInfo bitmapInfo = kCGBitmapByteOrderDefault;
    CFDataRef refData = CFDataCreate(kCFAllocatorDefault, picture.data[0], picture.linesize[0] * height);
    CGDataProviderRef refProvider = CGDataProviderCreateWithCFData(refData);
    CGColorSpaceRef colorSpace = CGColorSpaceCreateDeviceRGB();
   
    CGImageRef refImage = CGImageCreate(width,
                                        height,
                                        8,
                                        24,
                                        picture.linesize[0],
                                        colorSpace,
                                        bitmapInfo,
                                        refProvider,
                                        NULL,
                                        NO,
                                        kCGRenderingIntentDefault);
    UIImage *targetImage = [UIImage imageWithCGImage:refImage];
    __weak typeof(self) weakSelf = self;
    dispatch_async(dispatch_get_main_queue(), ^{
        [XSLToast hideLoadingAnimation:NO inView:weakSelf.panelImageView];
        weakSelf.panelImageView.image = targetImage;
    });
    CGColorSpaceRelease(colorSpace);
    CGImageRelease(refImage);
    CGDataProviderRelease(refProvider);
    CFRelease(refData);
}

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

推荐阅读更多精彩内容