Crow+opencv+websocket实现实时rtsp视频拉取以及显示

2024-06-04 14:44

本文主要是介绍Crow+opencv+websocket实现实时rtsp视频拉取以及显示,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

需求:需要将rtsp视频流放到openharmony界面显示

方案一:使用openharmonyAPP中集成ffmpeg(后续更新)

方案二:使用openharmonyAPP中集成opencv(实际原理和方案一一致,因为opencv中集成了ffmpeg,后续更新)

方案三:将视频在服务端拉取,转base64之后使用websocket发送到前端,在openharmonyAPP中使用一个嵌套的WEB显示html

crow的环境搭建参考Crow 一个c++的后端开发库,类似spring boot、flask等

网上没有找到能白嫖的代码,所以自己写了一个分享出来

//frame_generator.h

#include <iostream>
#include <vector>
#include <opencv2/opencv.hpp>
#include <thread.h>// 用于生成视频帧的生成器
class FrameGenerator {
public:bool isOpen = false;std::string _url;FrameGenerator(const std::string& rtsp_url) : cap(rtsp_url), _url(rtsp_url) {if (!cap.isOpened()) {reconnect();}isOpen = true;}void reconnect(){while (!cap.isOpened()){std::cout << "reconnect rtsp ." << std::endl;isOpen = false;cap.open(_url);std::this_thread::sleep_for(std::chrono::seconds(10)); }}const std::string base64_chars ="ABCDEFGHIJKLMNOPQRSTUVWXYZ""abcdefghijklmnopqrstuvwxyz""0123456789+/";std::string base64_encode(const std::string &input) {  std::string encoded;  size_t i = 0, j = 0;  uint8_t byte3[3] = {0};  uint8_t byte4[4] = {0};  // 遍历输入字符串中的每个字符  for (char byte : input) {  byte3[i++] = static_cast<uint8_t>(byte); // 假设输入是ASCII  if (i == 3) {  byte4[0] = (byte3[0] & 0xfc) >> 2;  byte4[1] = ((byte3[0] & 0x03) << 4) | ((byte3[1] & 0xf0) >> 4);  byte4[2] = ((byte3[1] & 0x0f) << 2) | ((byte3[2] & 0xc0) >> 6);  byte4[3] = byte3[2] & 0x3f;  // 添加编码后的字符到结果字符串  for (int k = 0; k < 4; k++) {  encoded += base64_chars[byte4[k]];  }  i = 0;  }  }  // 处理剩余字符(如果有)  if (i != 0) {  for (size_t k = i; k < 3; k++) {  byte3[k] = 0; // 填充剩余字节为0  }  // 执行编码,类似于前面的处理  byte4[0] = (byte3[0] & 0xfc) >> 2;  byte4[1] = ((byte3[0] & 0x03) << 4) | ((byte3[1] & 0xf0) >> 4);  byte4[2] = ((byte3[1] & 0x0f) << 2) | ((byte3[2] & 0xc0) >> 6);  // 添加编码后的字符到结果字符串  for (size_t k = 0; k < i + 1; k++) {  encoded += base64_chars[byte4[k]];  }  // 添加'='以填充到4的倍数  while (i++ < 3) {  encoded += '=';  }  }  std::cout << "base64 size:" << encoded.size() << std::endl;return encoded;  }std::string getFrame() {cv::Mat frame;if (!cap.isOpened()) {cap.open(_url);std::this_thread::sleep_for(std::chrono::seconds(10)); // throw std::runtime_error("Error opening video stream or file");}cap >> frame;if (frame.empty()) {std::cerr << "Error capturing frame" << std::endl;//throw std::runtime_error("Error capturing frame");}std::vector<uchar> buffer;cv::imencode(".jpg", frame, buffer);std::string _f = std::string(buffer.begin(), buffer.end());return base64_encode(_f);}private:cv::VideoCapture cap;
};

//crow websocket

std::mutex mtx2;
std::unordered_set<crow::websocket::connection *> users2;// opecv recv rtsp to wsCROW_WEBSOCKET_ROUTE(app, "/video").onopen([&](crow::websocket::connection &conn){std::cout << "New websocket connection from " << conn.get_remote_ip() << std::endl;std::lock_guard<std::mutex> lock(mtx2);users2.insert(&conn); // 添加新用户到集合中}).onclose([&](crow::websocket::connection &conn, const std::string &reason){std::cout << "Websocket connection closed: " << reason << std::endl;std::lock_guard<std::mutex> lock(mtx2);users2.erase(&conn); // 从集合中移除用户}).onmessage([&](crow::websocket::connection &conn, const std::string &data, bool is_binary){});std::thread videoMessageThread([&](){FrameGenerator generator("rtsp://XXX:8554/main.264");while (true){if (!generator.isOpen){generator.reconnect();continue;}std::lock_guard<std::mutex> lock(mtx2);for (auto user : users2){std::string frame = generator.getFrame();user->send_text(frame);}} });videoMessageThread.detach(); 

//前端接收

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>WebSocket Video Stream</title><style>html, body {margin: 0;padding: 0;height: 100%;overflow: hidden; /* 隐藏超出视口的内容 */}canvas {display: block;background-color: #f0f0f0; /* 可选:添加一个背景色以便在没有内容时看到canvas */width: 100vw; /* 使用视口宽度 */height: 100vh; /* 使用视口高度 */object-fit: cover; /* 如果需要的话,可以保持图像的宽高比 */}</style>
</head>
<body>
<canvas id="videoCanvas" width="1280" height="720"></canvas><script>// 获取canvas元素和它的2D渲染上下文let canvas = document.getElementById('videoCanvas');let ctx = canvas.getContext('2d');// 创建WebSocket连接let ws = new WebSocket('ws://XXX:8080/video');// WebSocket连接打开时的处理ws.onopen = function (event) {console.log('WebSocket is open now.');// 如果需要,向服务器发送开始传输的消息ws.send('START_STREAMING');};// 接收服务器发送的消息ws.onmessage = function (event) {console.log('imageData.', event.data);// 假设服务器发送的是base64编码的JPEG图片let imageData = 'data:image/jpeg;base64,' + event.data;// 创建一个新的Image对象来加载图片let img = new Image();// 图片加载完成后绘制到canvas上img.onload = function () {ctx.clearRect(0, 0, canvas.width, canvas.height);ctx.drawImage(img, 0, 0, img.width, img.height, 0, 0, canvas.width, canvas.height);};// 设置图片的src为base64编码的数据img.src = imageData;};// WebSocket连接错误时的处理ws.onerror = function (error) {console.error('WebSocket Error: ', error);};// WebSocket连接关闭时的处理ws.onclose = function (event) {if (event.wasClean) {console.log('WebSocket connection closed cleanly, code=' + event.code + ' reason=' + event.reason);} else {console.error('WebSocket connection died');}};
</script>
</body>
</html>

这篇关于Crow+opencv+websocket实现实时rtsp视频拉取以及显示的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1030347

相关文章

SpringBoot中SM2公钥加密、私钥解密的实现示例详解

《SpringBoot中SM2公钥加密、私钥解密的实现示例详解》本文介绍了如何在SpringBoot项目中实现SM2公钥加密和私钥解密的功能,通过使用Hutool库和BouncyCastle依赖,简化... 目录一、前言1、加密信息(示例)2、加密结果(示例)二、实现代码1、yml文件配置2、创建SM2工具

Mysql实现范围分区表(新增、删除、重组、查看)

《Mysql实现范围分区表(新增、删除、重组、查看)》MySQL分区表的四种类型(范围、哈希、列表、键值),主要介绍了范围分区的创建、查询、添加、删除及重组织操作,具有一定的参考价值,感兴趣的可以了解... 目录一、mysql分区表分类二、范围分区(Range Partitioning1、新建分区表:2、分

MySQL 定时新增分区的实现示例

《MySQL定时新增分区的实现示例》本文主要介绍了通过存储过程和定时任务实现MySQL分区的自动创建,解决大数据量下手动维护的繁琐问题,具有一定的参考价值,感兴趣的可以了解一下... mysql创建好分区之后,有时候会需要自动创建分区。比如,一些表数据量非常大,有些数据是热点数据,按照日期分区MululbU

MySQL中查找重复值的实现

《MySQL中查找重复值的实现》查找重复值是一项常见需求,比如在数据清理、数据分析、数据质量检查等场景下,我们常常需要找出表中某列或多列的重复值,具有一定的参考价值,感兴趣的可以了解一下... 目录技术背景实现步骤方法一:使用GROUP BY和HAVING子句方法二:仅返回重复值方法三:返回完整记录方法四:

IDEA中新建/切换Git分支的实现步骤

《IDEA中新建/切换Git分支的实现步骤》本文主要介绍了IDEA中新建/切换Git分支的实现步骤,通过菜单创建新分支并选择是否切换,创建后在Git详情或右键Checkout中切换分支,感兴趣的可以了... 前提:项目已被Git托管1、点击上方栏Git->NewBrancjsh...2、输入新的分支的

Python实现对阿里云OSS对象存储的操作详解

《Python实现对阿里云OSS对象存储的操作详解》这篇文章主要为大家详细介绍了Python实现对阿里云OSS对象存储的操作相关知识,包括连接,上传,下载,列举等功能,感兴趣的小伙伴可以了解下... 目录一、直接使用代码二、详细使用1. 环境准备2. 初始化配置3. bucket配置创建4. 文件上传到os

关于集合与数组转换实现方法

《关于集合与数组转换实现方法》:本文主要介绍关于集合与数组转换实现方法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、Arrays.asList()1.1、方法作用1.2、内部实现1.3、修改元素的影响1.4、注意事项2、list.toArray()2.1、方

使用Python实现可恢复式多线程下载器

《使用Python实现可恢复式多线程下载器》在数字时代,大文件下载已成为日常操作,本文将手把手教你用Python打造专业级下载器,实现断点续传,多线程加速,速度限制等功能,感兴趣的小伙伴可以了解下... 目录一、智能续传:从崩溃边缘抢救进度二、多线程加速:榨干网络带宽三、速度控制:做网络的好邻居四、终端交互

java实现docker镜像上传到harbor仓库的方式

《java实现docker镜像上传到harbor仓库的方式》:本文主要介绍java实现docker镜像上传到harbor仓库的方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录1. 前 言2. 编写工具类2.1 引入依赖包2.2 使用当前服务器的docker环境推送镜像2.2

C++20管道运算符的实现示例

《C++20管道运算符的实现示例》本文简要介绍C++20管道运算符的使用与实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录标准库的管道运算符使用自己实现类似的管道运算符我们不打算介绍太多,因为它实际属于c++20最为重要的