今天把读取本地摄像头将视频流推流到nginx服务器的直播代码学习完了,这里对代码的流程做一下记录,以便以后进行复习。

这边用到了opencv和ffmpeg的开源库(PS:在前面有进行分享),配置环境在之前也有进行分享。

第一步:先用到了opencv的VideoCapture类的open函数打开摄像头,这边的参数可以自己设置,我这边是打开本地的摄像头,所以参数为0。

第二步:初始化格式转化转换上下文,由于从opencv的函数读取出来的图片格式是rgb格式的,后面需要进行编码成h264格式的视频,h264格式的视频需要的是yuv格式的图片格式进行编码,这里初始化使用的函数是sws_getCachedContext()函数。

第三步:配置输出格式的数据结构,即配置yuv图片格式的数据结构,主要包括,宽、高、显示时间等配置项,具体的配置在后面的代码会进行分享。

第四步:初始化编码上下文,主要分为4个小步骤

1.找到编码器,主要利用avcodec_find_encoder()函数。

2.创建编码器的上下文avcodec_alloc_context3()函数。

3.配置编码器的参数,在代码中分享。这边有个坑,由于需要进行配置时间基数time_base,原来我是直接读取摄像头获取到的帧率,后面发现本地获取到的帧率会是0,所以会一直出现没有设置time_base的错误,后面我就改成直接设置成25帧(因为一般视频播放认为不卡顿的帧率为25帧)。

4.打开编码器上下文,利用avcodec_open2()函数。

第五步:封装器和视频流的配置,主要分为3步

1.创建输出封装上下文,主要利用avformat_alloc_output_context2()函数。

2.添加视频流,avformat_new_stream()函数

3.从编码器复制参数,avcodec_parameters_from_context()函数。

第六步:打开rtmp的网络输出io,主要分为2步

1.打开网络输出io,avio_open()函数

2.写入封装头,avformat_write_header()函数

进行初始化配置完后就可以开始进行直播推流了,接下会进入一个死循环,一直进行读取摄像头的视频帧。

在死循环中的步骤如下:

1.开始先利用opencv的grab()函数读取视频帧并进行解码,然后利用retrieve()获取到该rgb帧。

2.接着需要对获取的rgb帧进行转换成yuv帧,利用sws_scale()函数,由于opencv获取到的

图像的存储方式是类似于bgrbgrbgr形式的,即只有一个维度,所以定义的指针数组只需要使用下标为0的数组进行存储即可。

3.转化成yuv格式的视频帧数据后,就可以开始进行编码了,利用avcodec_send_frame()函数进行编码,然后再利用avcodec_receive_packet()函数获取到编码后的数据,这边需要对编码的pts进行配置,不然会出现错误,我这边是定义一个变量,从0开始计数,每解码完一帧后就进行++,增加计数。

4.获取到h264帧数据后就可以开始进行推流了。主要利用av_interleaved_write_frame()函数进行推流,这边要注意需要配置推流的pts和dts的时间基数(time_base),不然在后续同步的时候会出现错误,主要利用av_rescale_q()函数。

大致流程就是这样,接下来分享一下源代码。

#include <opencv2/highgui.hpp>
#include <iostream>
extern "C"
{
#include <libswscale/swscale.h>
#include <libavcodec/avcodec.h>
#include <libavformat/avformat.h>
}
using namespace std;
using namespace cv;
#pragma comment(lib,"opencv_world320.lib")
#pragma comment(lib,"swscale.lib")
#pragma comment(lib,"avcodec.lib")
#pragma comment(lib,"avutil.lib")
#pragma comment(lib,"avformat.lib")
int main(int argc,char *argv[])
{//nginx-rtmp 直播服务器rtmp推流URLchar *outUrl = "rtmp://192.168.198.128/live";//注册所有的编解码器avcodec_register_all();//注册所有的封装器av_register_all();//注册所有的网络协议avformat_network_init();VideoCapture cam;Mat frame;AVFrame *yuv = NULL;namedWindow("video");//像素格式转换上下文SwsContext *vsc = NULL;//编码器上下文AVCodecContext *vc = NULL;//rtmp flv 封装器AVFormatContext *ic = NULL;try {//使用opencv打开本地相机cam.open(0);///1.打开摄像头if (!cam.isOpened()){throw exception("cam open failed!");}cout << "cam open success" << endl;int inWidth = cam.get(CAP_PROP_FRAME_WIDTH);int inHeight = cam.get(CAP_PROP_FRAME_HEIGHT);int fps = cam.get(CAP_PROP_FPS);cout << "fps=" << fps << endl;///2.初始化格式转换上下文vsc = sws_getCachedContext(vsc,//源宽、高、像素格式inWidth,inHeight, AV_PIX_FMT_BGR24,//目标宽、高、像素格式inWidth, inHeight,AV_PIX_FMT_YUV420P,SWS_BICUBIC,  //尺寸变化使用的算法0,0,0);if (!vsc){throw exception("sws_getCachedContext failed!");}///3.输出的数据结构yuv = av_frame_alloc();yuv->format = AV_PIX_FMT_YUV420P;yuv->width = inWidth;yuv->height = inHeight;yuv->pts = 0;//分配yuv空间int ret = av_frame_get_buffer(yuv,32);if (ret != 0){char buf[1024] = { 0 };av_strerror(ret, buf, sizeof(buf) - 1);throw exception(buf);}///4.初始化编码上下文//a 找到编码器AVCodec *codec = avcodec_find_encoder(AV_CODEC_ID_H264);if (!codec){throw exception("Can't find h264 encoder!");}//b 创建编码器上下文vc = avcodec_alloc_context3(codec);if (!vc){throw exception("avcodec_alloc_context3 failed!");}//c 配置编码器参数vc->flags |= AV_CODEC_FLAG_GLOBAL_HEADER; //全局参数vc->codec_id = codec->id;vc->thread_count = 8;vc->bit_rate = 50 * 1024 * 8; //压缩后每秒视频的bit位大小  200kBvc->width = inWidth;vc->height = inHeight;vc->time_base = {1,25};vc->framerate = {25,1};//画面组的大小,多少帧一个关键帧vc->gop_size = 50;//设置不需要b帧vc->max_b_frames = 0;//设置像素格式vc->pix_fmt = AV_PIX_FMT_YUV420P;//d 打开编码器上下文ret = avcodec_open2(vc, 0, 0);if (ret != 0){char buf[1024] = { 0 };av_strerror(ret, buf, sizeof(buf) - 1);throw exception(buf);}cout << "avcodec_open2 success!" << endl;///5 封装器和视频流配置//a.创建输出封装器上下文ret = avformat_alloc_output_context2(&ic,0,"flv",outUrl);if (ret != 0){char buf[1024] = { 0 };av_strerror(ret, buf, sizeof(buf) - 1);throw exception(buf);}//b.添加视频流AVStream *vs = avformat_new_stream(ic, NULL);if (!vs){throw exception("avformat_new_stream failed!");}vs->codecpar->codec_tag = 0;//从编码器复制参数avcodec_parameters_from_context(vs->codecpar,vc);av_dump_format(ic,0,outUrl,1);///打开rtmp的网络输出ioret = avio_open(&ic->pb,outUrl,AVIO_FLAG_WRITE);if (ret != 0){char buf[1024] = { 0 };av_strerror(ret, buf, sizeof(buf) - 1);throw exception(buf);}//写入封装头ret = avformat_write_header(ic, NULL);if (ret != 0){char buf[1024] = { 0 };av_strerror(ret, buf, sizeof(buf) - 1);throw exception(buf);}AVPacket pack;memset(&pack, 0, sizeof(pack));int vpts = 0;//读取帧for (;;){///只做解码,读取视频帧,解码视频帧if (!cam.grab()){continue;}///yuv转化为rgbif (!cam.retrieve(frame)){continue;}imshow("video", frame);waitKey(1);///rgb to yuv//3.初始化输入的数据结构uint8_t *indata[AV_NUM_DATA_POINTERS] = {0};//indata[0] bgrbgrbgr//plane  indata[0] bbbbb indata[1]ggggg indata[2] rrrrrindata[0] = frame.data;int insize[AV_NUM_DATA_POINTERS] = { 0 };//一行(宽)数据的字节数insize[0] = frame.cols * frame.elemSize();int h = sws_scale(vsc,indata,insize,0,frame.rows,//源数据yuv->data,yuv->linesize);if (h <= 0){continue;}//cout << h <<"" <<flush;///h264编码yuv->pts = vpts;vpts++;ret = avcodec_send_frame(vc,yuv);if (ret != 0){continue;}ret = avcodec_receive_packet(vc,&pack);if (ret != 0||pack.size>0){cout << "*" << pack.size << flush;}else{continue;}//推流pack.pts = av_rescale_q(pack.pts, vc->time_base, vs->time_base);pack.dts = av_rescale_q(pack.dts, vc->time_base, vs->time_base);ret = av_interleaved_write_frame(ic,&pack);if (ret == 0){cout << "#" << flush;}}}catch (exception &ex){if (cam.isOpened())cam.release();if (vsc){sws_freeContext(vsc);vsc = NULL;}if (vc){avio_closep(&ic->pb);avcodec_free_context(&vc);}cerr << ex.what() << endl;}getchar();return 0;
}

然后可以在bin文件夹下创建一个批处理文件,利用ffplay进行拉流播放视频。

批处理文件的内容如下:

最后是运行的效果图。

音视频开发系列(7):完成本地摄像头直播推流相关推荐

  1. 【音视频开发系列】一学就会,快速掌握音视频开发的第一个开源项目FFmpeg

    快速掌握音视频开发的第一个开源项目:FFmpeg 1.为什么要学FFmpeg 2.FFmpeg面向对象思想分析 3.FFmpeg各种组件剖析 视频讲解如下,点击观看: [音视频开发系列]一学就会,快速 ...

  2. 【音视频开发系列】盘点音视频直播RTSP/RTMP推流一定会遇到的各种坑,教你快速解决

    聊聊RTSP/RTMP推流那些坑 1.推流架构分析 2.推流缓存队列的设计 3.FFmpeg函数阻塞问题分析 [音视频开发系列]盘点音视频直播一定会遇到的各种坑,教你快速解决 更多精彩内容包括:C/C ...

  3. 【音视频开发系列】srs-webrtc-janus开源流媒体服务器分析

    全球最牛开源流媒体服务器源码分析 1.如何学习流媒体服务器 2.全球最牛流媒体服务器架构分析 3.我们能从全球最牛流媒体服务器得到什么 [音视频开发系列]srs-webrtc-janus流媒体服务器分 ...

  4. Android音视频学习系列(九) — Android端实现rtmp推流

    系列文章 Android音视频学习系列(一) - JNI从入门到精通 Android音视频学习系列(二) - 交叉编译动态库.静态库的入门 Android音视频学习系列(三) - Shell脚本入门 ...

  5. 音视频开发系列-H264编码原理

    H264简介 来自百度百科的介绍: H.264是国际标准化组织(ISO)和国际电信联盟(ITU)共同提出的继MPEG4之后的新一代数字视频压缩格式. H.264是ITU-T以H.26x系列为名称命名的 ...

  6. 音视频开发系列--H264编解码总结

    一.概述 H264,通常也被称之为H264/AVC(或者H.264/MPEG-4 AVC或MPEG-4/H.264 AVC) 对摄像头采集的每一帧视频需要进行编码,由于视频中存在空间和时间的冗余,需要 ...

  7. 音视频开发系列(19)玩转 WebRTC 安全通信:一文读懂 DTLS 协议

    在 WebRTC 中,为了保证媒体传输的安全性,引入了 DTLS 来对通信过程进行加密.DTLS 的作用.原理与 SSL/TLS 类似,都是为了使得原本不安全的通信过程变得安全.它们的区别点是 DTL ...

  8. 音视频开发系列(16)技术解码 | SRT和RIST协议综述

    概要 近些年来,互联网行业出现了几波和音视频相关的热潮:VR.短视频.直播等.除了VR因技术成熟度问题,还在蓄势待发,短视频和直播持续热度不减,以各种方式进入新的行业应用领域.视频直播方向,RTMP仍 ...

  9. 音视频开发系列(17)文章分享-提速 30%腾讯TQUIC 网络传输协议

    作者:腾讯 sTGW-TQUIC 腾讯sTGW如何助力核心业务用户登录耗时降低30%,下载场景500ms内请求成功率从HTTPS的60%提升到90%,移动端APP在弱网.跨网场景下同样取得媲美正常网络 ...

  10. 音视频开发系列(34) OpenGL ES 绘制平面图形

    我们前两篇介绍了OpenGL ES 基本概念和GLSL及Shader的渲染流程,这篇我们开始实战,通过GLSurfaceView加载着色器,来绘制三角形.正方形和直线这些平面图形.在实践过程中遇到的问 ...

最新文章

  1. ORA-01109:数据库未打开(解决)
  2. 关于JS 事件冒泡和onclick,click,on()事件触发顺序
  3. 浅议数据中心规划设计阶段暖通系统节能措施
  4. 前、中、后缀表达式概述及转换+栈的计算器原理及代码分析(含完整源码)
  5. 用回车键实现MFC对话框中TAB键控件输入焦点在控件中跳转的效果(转)
  6. 怎样呵护友谊_呵护真正的友情,助力漫长的人生
  7. 2021中国短视频和直播电商行业人才发展报告
  8. 【Java】JShell工具上手即用
  9. “王炸”来了?苹果下一代Mac Pro将搭载两颗M1 Ultra芯片
  10. 如何才能在jsp文件中使用el表达式
  11. 在NodeJS中使用Redis缓存数据
  12. 河里的水哪里去了——兼谈气候变暖与西方文明
  13. python提取qq群成员代码_Python提取QQ群成员QQ号及昵称
  14. sql语句大全+实例讲解
  15. Python 去除白色背景
  16. linux centos7 录屏,centos7在线安装视频录制软件
  17. 苹果cms怎么添加2019和2020年份筛选
  18. Matlab 2018b 安装问题 - License checkout failed
  19. HD5450显卡驱动 for mac 官方免费版
  20. 加盐密码哈希:如何正确使用 (密码加密的经典文章)

热门文章

  1. 移动互联网感言(董烨/Joven.Dong)
  2. 古筝d调变降e调怎么办_古筝转调方法_古筝怎么转调
  3. divgrad怎么求_请问高等数学中div(grad u)中的div是什么意思?
  4. RTK如何进行面积测量,跟攻略学就对了
  5. 《小岛经济学》八、金本位的破灭、房地产的泡沫
  6. 6-1 求实数和的函数
  7. python爬京东延迟加载_python大规模爬取京东
  8. 《红警2》防空步兵都说了些什么?
  9. java 对象逃逸 解决_Java中的逃逸问题心得
  10. Springboot中使用freemarker动态生成word文档