当前位置: 首页 > news >正文

C++使用FFmpeg进行视频推流

一、前言

项目中对摄像头的实时画面进行一些处理后,有需求将处理后的画面进行编码并推流,然后在其他设备上可以直接拉流播放,所以需要我们直接使用 C++ API 进行推流操作了。

由于我这边服务端的设备是 RK3588 的开发板,推荐是使用 MPP 库进行编码,它会直接使用 VPU 设备进行硬编码,性能会好很多。所以方案计划为:

  1. 使用 mpp 进行单独的硬编码,将YUV数据编码为H264数据;
  2. 使用 ffmpeg 把编码好的H264数据写入到服务端;
  3. 使用 mediamtx 作为服务端,管理写入的H264数据以及和客户端的连接等。

二、Mediamtx 下载和运行

1. 下载

官方仓库:https://github.com/bluenviron/mediamtx,里面有详细的使用文档介绍。

下载链接:https://github.com/bluenviron/mediamtx/releases

上述下载链接中有已经编译好的程序,稍微往下滑一下就可以看到下载链接了,在 Assets 标签下,直接下载解压就可以使用了,注意不要下载错就行,比如我的系统是 Linux arm64 位的系统,就应该下载 mediamtx_v1.13.1_linux_arm64.tar.gz

2. 运行

解压后有一个 mediamtx 的可执行文件和一个 mediamtx.yml 配置文件。直接运行可执行文件即可。

./mediamtx

这个服务就保持在后台常驻运行就好。

三、FFmpeg 代码编写

1. 下载

关于 ffmpeg 的下载、编译、安装文章比较多,这里不再过多介绍了。

2. 头文件

#pragma once#include <string>
extern "C" {
#include <libavcodec/avcodec.h>
#include <libavformat/avformat.h>
}class FFmpegServer {public:// url是推流的目标地址FFmpegServer(const std::string& url, int width, int height, int fps);~FFmpegServer();bool initialize();// data是已经编译好的h264数据// is_key_frame标识是否是关键帧bool pushFrame(const uint8_t* data, size_t size, int frame_count, bool is_key_frame);private:std::string rtsp_url;int frame_width;int frame_height;int frame_rate;AVFormatContext* fmt_ctx = nullptr;AVStream* stream = nullptr;
};

3. 源代码

#include "FFmpegServer.hpp"#include "stdio.h"
extern "C" {
#include <libavutil/opt.h>
}FFmpegServer::FFmpegServer(const std::string& url, int width, int height, int fps): rtsp_url(url), frame_width(width), frame_height(height), frame_rate(fps) {avformat_network_init();
}FFmpegServer::~FFmpegServer() {if (fmt_ctx) {av_write_trailer(fmt_ctx);if (!(fmt_ctx->oformat->flags & AVFMT_NOFILE)) avio_closep(&fmt_ctx->pb);avformat_free_context(fmt_ctx);}avformat_network_deinit();
}bool FFmpegServer::initialize() {// 创建输出上下文avformat_alloc_output_context2(&fmt_ctx, NULL, "rtsp", rtsp_url.c_str());if (!fmt_ctx) {printf("Could not create output context\n");return false;}// 创建视频流stream = avformat_new_stream(fmt_ctx, NULL);if (!stream) {printf("Failed creating new stream\n");return false;}// 设置H264编码参数AVCodecParameters* codecpar = stream->codecpar;codecpar->codec_id = AV_CODEC_ID_H264;codecpar->codec_type = AVMEDIA_TYPE_VIDEO;codecpar->width = frame_width;codecpar->height = frame_height;codecpar->format = AV_PIX_FMT_YUV420P;// 打开输出if (!(fmt_ctx->oformat->flags & AVFMT_NOFILE)) {int ret = avio_open(&fmt_ctx->pb, fmt_ctx->filename, AVIO_FLAG_WRITE);if (ret < 0) {printf("Could not open output URL: %s, Err: %d\n", rtsp_url.c_str(), ret);return false;}}// 写文件头// 注意:如果 mediamtx 没有运行的话,这里写入就会报错了,会说连不上int ret = avformat_write_header(fmt_ctx, NULL);if (ret < 0) {printf("Error writing header: %d\n", ret);return false;}return true;
}bool FFmpegServer::pushFrame(const uint8_t* data, size_t size, int frame_count, bool is_key_frame) {AVPacket pkt;av_init_packet(&pkt);pkt.data = const_cast<uint8_t*>(data);pkt.size = size;pkt.stream_index = stream->index;int64_t pts_counter = (int64_t)frame_count * (90000 / frame_rate);pkt.pts = pts_counter;pkt.dts = pts_counter;pkt.flags = is_key_frame ? AV_PKT_FLAG_KEY : 0;// 写帧int ret = av_interleaved_write_frame(fmt_ctx, &pkt);if (ret < 0) {printf("Error writing frame: %d", ret);return false;}return true;
}

4. 调用示例

#include "FFmpegServer.hpp"int main(int argc, char* argv[]) {int width = 1920;int height = 1080;int fps = 30;// 192.168.20.200 为你的ip地址// 8554 是端口号,建议不要修改// video 字符串可以自定义为任一你喜欢的值std::string url = "rtsp://192.168.20.200:8554/video";FFmpegServer ffmpeg_server(url, width, height, fps);if (!ffmpeg_server.initialize()) {return;}int frame_count = 0;while (true) {// 模拟获取的编码数据unsigned char* h264_data = ...;size_t size = ...;bool is_key_frame = frame_count % fps == 0;ffmpeg_server.pushFrame(data, size, frame_count, is_key_frame);frame_count++;}
}

四、客户端播放

客户端的网络需要保证能连上服务端的地址,即可以 ping 通上述示例中 “192.168.20.200”。

客户端上安装任一可以播放 RTSP 流的播放器,或者安装 ffmpeg 后使用 ffplay 进行播放。

下面以 ffplay 命令播放为示例:

# 直接播放
ffplay rtsp://192.168.20.200:8554/video# 低延迟播放
ffplay -fflags nobuffer -flags low_delay -framedrop rtsp://192.168.20.200:8554/video
http://www.dtcms.com/a/320917.html

相关文章:

  • conda或mamba install 相关软件报错
  • GraphRAG 入门教程:从原理到实战
  • 【概率论】均匀分布的伪随机数
  • 代码随想录Day43:动态规划(最长递增子序列、最长连续递增序列、最长重复子数组)
  • Linux网络--1、网络基础
  • 【机器学习深度学习】微调训练数据质量
  • JSON结构的 泛型和 **非泛型两种 Java 实体类实现方案
  • 【DFS系列 | 递归】DFS算法入门:递归原理与实现详解
  • Electron——窗口
  • 工具分享05 | Python制作PDF合并拆分提取工具V1.0
  • Linux 学习 ------Linux 入门(上)
  • trae开发c#
  • Android 中 实现自定义 Dialog 提示框
  • @CacheConfig​​当前类中所有缓存方法详解
  • Go语言实战案例:简易JSON数据返回
  • Linux多线程——生产者消费者模型
  • 芯科科技成为全球首家通过PSA 4级认证的物联网芯片厂商巩固其在物联网安全领域的领导地位
  • 从零开始构建【顺序表】:C语言实现与项目实战准备
  • 《汇编语言:基于X86处理器》第12章 浮点数处理与指令编码(2)
  • 【JavaEE】(10) JavaEE 简介
  • 虚幻基础:场景actor与角色的碰撞
  • 走进Linux世界:make和makefile
  • 深入浅出线程池ThreadPoolExecutor
  • 机器人焊机智能流量调节
  • SwiftUI中的键盘快捷键、初始页面控制及网络权限管理解析
  • linux-LVM 逻辑卷管理
  • Windows域控制器部署最佳实践
  • (1-9-1) Maven 特性、安装、配置、打包
  • 安全扫描:检测到目标站点存在javascript框架库漏洞问题(vue)
  • DQL 超维分析 - 5 集算器 DQL