海纳思(Hi3798MV300)机顶盒遇到海思摄像头
海纳思机顶盒遇到海思摄像头,正好家里有个海思Hi3516的摄像头模组开发板,结合机顶盒来做个录像。
准备工作
- 海纳斯机顶盒
- 摄像机模组
- 两根网线、两个电源、路由器
- 一块64G固态硬盘
摄像机模组和机顶盒都接入路由器的LAN口,确保网络正常通信。
调试录像
摄像机模组
摄像机模组里的程序其实是基于海思的SDK里的demo稍作修改而成,没有做太复杂的功能,只加入了RTSP,对外提供RTSP接口服务。
这里用的rtsp服务的库代码比较好用,源码链接:https://gitee.com/fensnote/RtspServer
在电脑上用VLC测试拉流播放:
海纳斯盒子录像
关于录像,这里只是实现简单的文件存储、循环覆盖,并不是专业的录像,专业录像里会做的比较复杂。
- 直接用Ffmpeg命令行录取数据到文件里,为了方便播放保存为MP4文件。
- 写代码实现rtsp拉流存储,可以自己定义传参。
Ffmpeg录像
这个比较简单,一条命令即可,不过直接采用命令录像没法指定实现循环覆盖,要想实现可以再写个脚本取定时检测录像文件的个数。
首先需要先下载安装Ffmpeg:
sudo apt install ffmpeg
我这里已经安装过了。
接下来就用可以执行录像了:
ffmpeg -rtsp_transport tcp -i rtsp://192.168.2.168:41667/live -c copy -f segment -segment_time 60 stream_piece_%d.mp4
这条命令里是指定了录像时长60秒,即一分钟切换一个文件。
如下截图,录取一分钟后已切换文件,1分钟录像数据15M,数据量挺大了:
上传上来播放一下看看:
播放:
使用win11的系统播放器就可以播放
写个代码录像
这里选用了go语言来写这个录像代码,是因为go语言的音视频、网络相关的库实在太多,比较好用,代码量也不大,可以提需求让AI去写,AI写的基本上稍作修改测试几次就可以用了。
Go还有个好处就是静态编译,真正的跨平台,一次编译,CPU架构一样都可以运行,感觉缺点就是可执行文件比较大。
这里采用的gortsplib,源码地址:https://gitee.com/fensnote/gortsplib.git
可以基于gortsplib/examples下的client-play-format-h264-save-to-disk示例代码做修改:
我复制了命名为client-play-format-h264-save-to-disk-file,在这里修改,下面代码是调试完成的代码:
package mainimport ("flag""fmt""log""os""os/signal""syscall""time""github.com/bluenviron/gortsplib/v4""github.com/bluenviron/gortsplib/v4/pkg/base""github.com/bluenviron/gortsplib/v4/pkg/format""github.com/bluenviron/gortsplib/v4/pkg/format/rtph264""github.com/pion/rtp"
)const (filePrefix = "rec" // 文件名前缀fileSuffix = ".mp4" // 文件名后缀
)func main() {// Define command line flagsrtspURL := flag.String("r", "", "RTSP URL")maxFilesPtr := flag.Int("c", 0, "文件数")startFileNumPtr := flag.Int("s", 0, "起始文件编号")durationPtr := flag.Int("t", 60, "单个文件录像时长")modePtr := flag.String("m", "loop", "录像模式,单次模式:\"once\",循环模式:\"loop\", 注意要加双引号")// Parse command line flagsflag.Parse()// Check if the required arguments are providedif *rtspURL == "" || *maxFilesPtr == 0 { //|| *startFileNumPtr == 0 flag.PrintDefaults() // Print usage informationlog.Fatal("Missing required command line arguments")}if *startFileNumPtr < 0 || *startFileNumPtr > *maxFilesPtr {log.Fatalf("起始文件编号必须是0~%d", *maxFilesPtr)}// Check if the RTSP URL is validu, err := base.ParseURL(*rtspURL)if err != nil {log.Fatalf("无效的RTSP URL: %v", err)}c := gortsplib.Client{}// Connect to the servererr = c.Start(u.Scheme, u.Host)if err != nil {log.Fatalf("连接 RTSP server 失败: %v", err)}
// defer c.Close()// Find available mediasdesc, _, err := c.Describe(u)if err != nil {log.Fatalf("Failed to describe RTSP stream: %v", err)}// Find the H264 media and formatvar forma *format.H264medi := desc.FindFormat(&forma)if medi == nil {log.Fatal("H264 media not found")}// Setup RTP -> H264 decoderrtpDec, err := forma.CreateDecoder()if err != nil {log.Fatalf("Failed to create H264 decoder: %v", err)}var mpegtsMuxer *mpegtsMuxervar fileCounter intvar recordingStartTime time.Time// var bakPts int64;// var sub int// Create the first file immediately when the program startsfileCounter = *startFileNumPtrnewFileName := fmt.Sprintf("%s%03d%s", filePrefix, fileCounter, fileSuffix)mpegtsMuxer = newMpegtsMuxer(newFileName, forma.SPS, forma.PPS)err = mpegtsMuxer.initialize()if err != nil {log.Fatalf("Failed to initialize MPEG-TS muxer: %v", err)}log.Printf("New file created: %s", newFileName)recordingStartTime = time.Now()// Setup a single media_, err = c.Setup(desc.BaseURL, medi, 0, 0)if err != nil {log.Fatalf("Failed to setup media: %v", err)}// Create a ticker to create a new file based on the specified durationduration := time.Duration(*durationPtr) * time.Secondticker := time.NewTicker(duration)duration = duration + 100000000 // Add 200ms to the duration to ensure the ticker fires after the durationdefer ticker.Stop()// bakPts = 0// Called when a RTP packet arrivesc.OnPacketRTP(medi, forma, func(pkt *rtp.Packet) {// Decode timestamppts, ok := c.PacketPTS2(medi, pkt)if !ok {//log.Printf("Waiting for timestamp")pts = int64(pkt.Timestamp)//return}// if bakPts == 0 {// bakPts = pts// }// Extract access unit from RTP packetsau, err := rtpDec.Decode(pkt)if err != nil {if err != rtph264.ErrNonStartingPacketAndNoPrevious && err != rtph264.ErrMorePacketsNeeded {log.Printf("ERR: %v", err)}return}// sub = (int)(pts - bakPts)/100000// Encode the access unit into MPEG-TSif mpegtsMuxer != nil {err = mpegtsMuxer.writeH264(au, pts)if err != nil {log.Printf("ERR: %v", err)return}// log.Printf("Saved TS packet, pts: %d,sub:%d",pts,sub)}// Check if it's time to create a new file or exit// if sub >= *durationPtr {if time.Since(recordingStartTime) >= duration {mpegtsMuxer.close()if *modePtr == "once" {log.Println("Recording duration reached, exiting...")c.Close() // Close the RTSP client connectionos.Exit(0) // Exit the program immediately} else {fileCounter = (fileCounter + 1) % *maxFilesPtrnewFileName := fmt.Sprintf("%s%03d%s", filePrefix, fileCounter, fileSuffix)mpegtsMuxer = newMpegtsMuxer(newFileName, forma.SPS, forma.PPS)err = mpegtsMuxer.initialize()if err != nil {log.Fatalf("ERR: %v", err)c.Close() // Close the RTSP client connectionos.Exit(-1) // Exit the program immediately}log.Printf("New file created: %s", newFileName)recordingStartTime = time.Now()//bakPts = pts}}})// Start playing_, err = c.Play(nil)if err != nil {log.Fatalf("Failed to play RTSP stream: %v", err)}// Wait for interrupt signal or recording durationsigChan := make(chan os.Signal, 1)signal.Notify(sigChan, syscall.SIGINT, syscall.SIGTERM)go func() {for {select {case <-ticker.C:if *modePtr == "once" {log.Println("Recording duration reached, exiting...")c.Close()os.Exit(0)}case <-sigChan:log.Println("Interrupt signal received, exiting...")c.Close()os.Exit(0)}}}()// Block main goroutine foreverselect {}
}
代码编译:
export GOOS=linux
export GOARCH=arm
export GOARM=5
#export CGO_ENABLED=1go build -ldflags '-s -w'
录像测试:
vrec-c int文件数-m string录像模式,单次模式:"once",循环模式:"loop", 注意要加双引号 (default "loop")-r stringRTSP URL-s int起始文件编号-t int单个文件录像时长 (default 60)
2025/05/10 09:30:59 Missing required command line arguments
#录像命令参数:
vrec -c 1200 -m "loop" -s 0 -t 60 -r rtsp://192.168.2.168:41667/live
录像文件播放
录像文件查看,这是录了一晚上的,文件比较多:
通过电脑查看
在海纳思的内置web页面查看录像文件,首页还是挺好看的:
录像文件可以直接点击播放:
通过手机查看
点击播放
相关文章:

海纳思(Hi3798MV300)机顶盒遇到海思摄像头
海纳思机顶盒遇到海思摄像头,正好家里有个海思Hi3516的摄像头模组开发板,结合机顶盒来做个录像。 准备工作 海纳斯机顶盒摄像机模组两根网线、两个电源、路由器一块64G固态硬盘 摄像机模组和机顶盒都接入路由器的LAN口,确保网络正常通信。 …...
MCP项目实例 - client sever交互
1. 项目概述 项目目标 构建一个本地智能舆论分析系统。 利用自然语言处理和多工具协作,实现用户查询意图的自动理解。 进行新闻检索、情绪分析、结构化输出和邮件推送。 系统流程 用户查询:用户输入查询请求。 提取关键词:从用户查询中…...

Axure应用交互设计:表格跟随菜单移动效果(超长表单)
亲爱的小伙伴,在您浏览之前,烦请关注一下,在此深表感谢!本文如有帮助请订阅 Axure产品经理精品视频课已登录CSDN可点击学习https://edu.csdn.net/course/detail/40420 课程主题:表格跟随菜单移动 主要内容:表格交互设计、动态面板嵌套、拖动时事件、移动动作 应用场景…...

7系列 之 I/O标准和终端技术
背景 《ug471_7Series_SelectIO.pdf》介绍了Xilinx 7 系列 SelectIO 的输入/输出特性及逻辑资源的相关内容。 第 1 章《SelectIO Resources》介绍了输出驱动器和输入接收器的电气特性,并通过大量实例解析了各类标准接口的实现。 第 2 章《SelectIO Logic Resource…...

github 上的 CI/CD 的尝试
效果 步骤 新建仓库设置仓库的 page 新建一个 vite 的项目,改一下 vite.config.js 中的 base 工作流 在项目的根目录下新建一个 .github/workflows/ci.yml 文件,然后编辑一下内容 name: Build & Deploy Vue 3 Appon:push:branches: [main]permi…...
Scala和Go差异
Scala和Go(又称Golang)是两种现代编程语言,各自具有独特的特性和设计哲学。 尽管它们都可以用于构建高性能、可扩展的应用程序,但在许多方面存在显著差异。 Scala和Go的详细比较,涵盖它们的异同点: 1. 语…...

yup 使用 3 - 利用 meta 实现表单字段与表格列的统一结构配置(适配 React Table)
yup 使用 3 - 利用 meta 实现表单字段与表格列的统一结构配置(适配 React Table) Categories: Tools Last edited time: May 11, 2025 7:45 PM Status: Done Tags: form validation, schema design, yup 本文介绍如何通过 Yup 的 meta() 字段࿰…...
类初始化方法
一、类初始化方法 成员初始化列表 class Point {int x, y; public:Point(int a, int b) : x(a), y(b) {} };就地初始化(C11) 声明时初始化。 class Widget {int size 10; // 类内成员初始化vector<int> data{1,2,3}; };特殊情况:静…...

【OpenCV】imread函数的简单分析
目录 1.imread()1.1 imread()1.2 imread_()1.2.1 查找解码器(findDecoder)1.2.2 读取数据头(JpegDecoder-->readHeader)1.2.2.1 初始化错误信息(jpeg_std_error)1.2.2.2 创建jpeg解压缩对象(…...

【Linux实践系列】:进程间通信:万字详解共享内存实现通信
🔥 本文专栏:Linux Linux实践项目 🌸作者主页:努力努力再努力wz 💪 今日博客励志语录: 人生就像一场马拉松,重要的不是起点,而是坚持到终点的勇气 ★★★ 本文前置知识: …...

【笔记】BCEWithLogitsLoss
工作原理 BCEWithLogitsLoss 是 PyTorch 中的一个损失函数,用于二分类问题。 它结合了 Sigmoid 激活函数和二元交叉熵(Binary Cross Entropy, BCE)损失在一个类中。 这不仅简化了代码,而且通过数值稳定性优化提高了模型训练的效…...
Oracle SYSTEM/UNDO表空间损坏的处理思路
Oracle SYSTEM/UNDO表空间损坏是比较棘手的故障,通常会导致数据库异常宕机进而无法打开数据库。数据库的打开故障处理起来相对比较麻烦,读者可以参考本书第5章进一步了解该类故障的处理过程。如果数据库没有备份,通常需要设置官方不推荐的隐含…...
为什么 cout<<“中文你好“ 能正常输出中文
一, 简答: 受python3字符串模型影响得出的下文C字符串模型结论 是错的!C的字符串和python2的字符串模型类似,也就是普通的字符串是ASCII字符串和字节串两种语义,类似重载或多态,有时候解释为整数,有时候是字节串。Uni…...
Leetcode 3547. Maximum Sum of Edge Values in a Graph
Leetcode 3547. Maximum Sum of Edge Values in a Graph 1. 解题思路2. 代码实现 题目链接:3547. Maximum Sum of Edge Values in a Graph 1. 解题思路 这一题主要是在问题的分析上面。由题意易知,事实上给定的图必然只可能存在三种可能的结构&#x…...

关于Go语言的开发环境的搭建
1.Go开发环境的搭建 其实对于GO语言的这个开发环境的搭建的过程,类似于java的开发环境搭建,我们都是需要去安装这个开发工具包的,也就是俗称的这个SDK,他是对于我们的程序进行编译的,不然我们写的这个代码也是跑不起来…...

Flutter PIP 插件 ---- 为iOS 重构PipController, Demo界面,更好的体验
接上文 Flutter PIP 插件 ---- 新增PipActivity,Android 11以下支持自动进入PIP Mode 项目地址 PIP, pub.dev也已经同步发布 pip 0.0.3,你的加星和点赞,将是我继续改进最大的动力 在之前的界面设计中,还原动画等体验一…...
Redis 基本命令与操作全面解析:从入门到实战
前言 Redis 作为高性能内存数据库,其丰富的命令体系是发挥强大功能的基础。掌握 Redis 的基本命令,不仅能实现数据的高效读写,还能深入理解其内存模型与工作机制。本文将系统梳理 Redis 的核心命令,涵盖连接操作、键管理、数据类…...

数据库管理-第325期 ADG Failover后该做啥(20250513)
数据库管理325期 2025-05-13 数据库管理-第325期 ADG Failover后该做啥(20250513)1 故障处置2 恢复原主库3 其他操作总结 数据库管理-第325期 ADG Failover后该做啥(20250513) 作者:胖头鱼的鱼缸(尹海文&a…...

SQLi-Labs 第21-24关
Less-21 http://127.0.0.1/sqli-labs/Less-21/ 1,抓个请求包看看 分析分析cookie被base64URL编码了,解码之后就是admin 2,那么这个网站的漏洞利用方式也是和Less-20关一样的,只是攻击语句要先base64编码,再URL编码&…...
Oracle — 数据管理
介绍 Oracle数据库作为全球领先的关系型数据库管理系统,其数据管理能力以高效性、安全性和智能化为核心。系统通过多维度技术实现海量数据的存储与实时处理,支持高并发事务操作与复杂分析查询,满足企业关键业务需求。在安全领域,O…...
在 Qt Creator 中为 QDockWidget 设置隐藏和显示按钮
在 Qt Creator 中为 QDockWidget 设置隐藏和显示按钮 是的,QDockWidget 内置了隐藏和显示的功能,可以通过以下几种方式实现: 1. 使用 QDockWidget 自带的关闭按钮 QDockWidget 默认带有一个关闭按钮,可以通过以下代码启用&…...
LS-NET-012-TCP的交互过程详解
LS-NET-012-TCP的交互过程详解 附加:TCP如何保障数据传输 TCP的交互过程详解 一、TCP协议核心交互流程 TCP协议通过三次握手建立连接、数据传输、四次挥手终止连接三大阶段实现可靠传输。整个过程通过序列号、确认应答、窗口控制等机制保障传输可靠性。 1.1 三次…...
每日算法刷题Day1 5.9:leetcode数组3道题,用时1h
1.LC寻找数组的中心索引(简单) 数组和字符串 - LeetBook - 力扣(LeetCode)全球极客挚爱的技术成长平台 思想: 计算总和和左侧和,要让左侧和等于右侧和,即左侧和总和-左侧和-当前数字 代码 c代码: class Solution { public:i…...
解构认知边界:论万能方法的本体论批判与方法论重构——基于跨学科视阈的哲学-科学辩证
一、哲学维度的本体论批判 (1)理性主义的坍缩:从笛卡尔幻想到哥德尔陷阱 笛卡尔在《方法论》中构建的理性主义范式,企图通过"普遍怀疑-数学演绎"双重机制确立绝对方法体系。然而哥德尔不完备定理(Gdel, 19…...

PVE WIN10直通无线网卡蓝牙
在 Proxmox VE (PVE) 中直通 Intel AC3165 无线网卡的 **蓝牙模块**(通常属于 USB 设备,而非 PCIe 设备)需要特殊处理,因为它的蓝牙部分通常通过 USB 连接,而 Wi-Fi 部分才是 PCIe 设备。以下是详细步骤: …...

第六节第二部分:抽象类的应用-模板方法设计模式
模板方法设计模式的写法 建议使用final关键字修饰模板方法 总结 代码: People(父类抽象类) package com.Abstract3; public abstract class People {/*设计模板方法设计模式* 1.定义一个模板方法出来*/public final void write(){System.out.println("\t\t\t…...

在另一个省发布抖音作品,IP属地会随之变化吗?
你是否曾有过这样的疑惑:出差旅游时在外地发布了一条抖音视频,评论区突然冒出“IP怎么显示xx省了?”的提问?随着各大社交平台上线“IP属地”功能,用户的地理位置标识成为公开信息,而属地显示的“灵敏性”也…...

卷积神经网络-从零开始构建一个卷积神经网络
目录 一、什么是卷积神经网络CNN 1.1、核心概念 1.2、卷积层 二、什么是卷积计算 2.1、卷积计算的例子: 2.2、点积 2.3、卷积与点积的关系 2.4、Padding(填充) 2.4.1、Padding的主要作用 1、控制输出特征图尺寸 2、保留边缘信息 3. 支持深层网络训练 2.4.2、Str…...
力扣-101.对称二叉树
题目描述 给你一个二叉树的根节点 root , 检查它是否轴对称。 class Solution { public:bool check(TreeNode* p,TreeNode* q){if(!p&&!q)return true;if(!p&&q||!q&&p)return false;if(p->val!q->val)return false;return check(p…...
Tomcat和Nginx的主要区别
1、功能定位 Nginx:核心是高并发HTTP服务器和反向代理服务器,擅长处理静态资源(如HTML、图片)和负载均衡。Tomcat:是Java应用服务器,主要用于运行动态内容(如JSP、Servlet)…...