0

0

c++如何操作海康威视sdk_c++ 摄像头连接与实时视频流获取【指南】

穿越時空

穿越時空

发布时间:2026-01-03 19:03:08

|

498人浏览过

|

来源于php中文网

原创

c++kquote>海康威视HCNetSDK无官方C++封装,需直接调用C接口;必须先调用且仅调用一次NET_DVR_Init(),登录失败需用NET_DVR_GetLastError()查具体错误,拉流回调须为extern "C"并区分帧头与数据,资源释放须严格按停止预览→登出→清理顺序。

c++如何操作海康威视sdk_c++ 摄像头连接与实时视频流获取【指南】

海康威视的 HCNetSDK 官方不提供纯 C++ 封装,所谓“sdk_c++”只是社区或厂商基于 C 接口做的薄层 C++ 封装(如 CHCNetSDK 类),实际调用仍是 C 风格函数。直接用 C 接口更可控,也避免封装层引入的资源管理漏洞和线程安全陷阱。

初始化 SDK 并登录设备前必须调用 NET_DVR_Init()

这是所有操作的前提,漏掉会导致后续所有接口返回失败(如 NET_DVR_Login_V40 返回 -1)。它内部初始化网络、日志、内存池等全局资源,且**只能调用一次**——重复调用可能引发崩溃或句柄泄漏。

  • 建议在程序启动时尽早调用,例如 main() 开头或单例类构造中
  • 若需卸载(极少见),用 NET_DVR_Cleanup(),但必须确保所有设备已登出、所有预览/回放已停止
  • 调试时可启用 SDK 日志:调用 NET_DVR_SetLogToFile(3, "log/", true),日志级别 3 覆盖大部分错误

NET_DVR_Login_V40() 登录失败常见原因

返回值为 -1 时不能只看返回码,要立刻调用 NET_DVR_GetLastError() 获取真实错误码。高频问题包括:

  • ERROR_INVALID_PARAMETER (5)NET_DVR_USER_LOGIN_INFO 结构体未 memset 初始化,或 sDeviceAddress 含多余空格/换行
  • ERROR_SDK_VERSION_NOT_SUPPORT (7):SDK 版本低于设备固件要求,需升级 SDK 或降级设备固件
  • ERROR_PASSWORD_ERROR (6):密码错误,或设备开启了“密码复杂度校验”但未满足(如长度、大小写、特殊字符)
  • ERROR_DEVICE_ONLINE (28):设备已在线,但 SDK 默认不允许重复登录;需设置 struLoginInfo.bUseAsynLogin = false 并检查 struDeviceInfo.byChanNum 是否为 0(设备未就绪)

NET_DVR_RealPlay_V40() 拉流必须配对处理回调

实时流数据通过回调函数传递,不是同步返回。关键点在于:

Text-To-Pokemon口袋妖怪
Text-To-Pokemon口袋妖怪

输入文本生成自己的Pokemon,还有各种选项来定制自己的口袋妖怪

下载

立即学习C++免费学习笔记(深入)”;

  • 回调函数必须是 C 风格(extern "C"),不能是类成员函数(除非用 static 包装并传 void* pUserData
  • 回调中收到的 lpBuffer 是 H.264/H.265 原始帧(含 SPS/PPS),**不是 RGB/BGR 图像**;需用 FFmpeg 或硬解码器进一步解码渲染
  • 务必检查 dwDataType:值为 NET_DVR_SYSHEAD 表示帧头(含 SPS/PPS),需缓存供解码器初始化;值为 NET_DVR_STREAMDATA 才是视频帧数据
  • 回调里禁止做耗时操作(如图像显示、文件写入),应仅 memcpy 到队列,由另一线程处理,否则 SDK 内部缓冲区会满导致断流
extern "C" void CALLBACK fRealDataCallBack_V30(LONG lRealHandle, DWORD dwDataType, BYTE *pBuffer, DWORD dwBufSize, void *pUser)
{
    if (dwDataType == NET_DVR_SYSHEAD) {
        // 缓存 SPS/PPS
        g_sps_pps.assign(pBuffer, pBuffer + dwBufSize);
    } else if (dwDataType == NET_DVR_STREAMDATA) {
        // 推入解码队列
        g_video_queue.push(std::vector(pBuffer, pBuffer + dwBufSize));
    }
}

退出时按顺序释放资源,顺序错会 crash

SDK 资源释放有强依赖顺序,颠倒极易导致访问非法内存:

  • 先停止预览:NET_DVR_StopRealPlay(lRealHandle)
  • 再登出设备:NET_DVR_Logout(lUserID)(注意是 lUserID,不是 lRealHandle
  • 最后才清理 SDK:NET_DVR_Cleanup()
  • 如果用了语音对讲,需额外调用 NET_DVR_CloseSound()NET_DVR_StopVoiceCom()

多路拉流时,每个 lRealHandle 必须对应一次 NET_DVR_StopRealPlay(),漏掉一路就会让该路线程卡死在 SDK 内部。

相关专题

更多
golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

194

2025.06.09

golang结构体方法
golang结构体方法

本专题整合了golang结构体相关内容,请阅读专题下面的文章了解更多。

186

2025.07.04

javascriptvoid(o)怎么解决
javascriptvoid(o)怎么解决

javascriptvoid(o)的解决办法:1、检查语法错误;2、确保正确的执行环境;3、检查其他代码的冲突;4、使用事件委托;5、使用其他绑定方式;6、检查外部资源等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

174

2023.11.23

java中void的含义
java中void的含义

本专题整合了Java中void的相关内容,阅读专题下面的文章了解更多详细内容。

94

2025.11.27

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1002

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

56

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

272

2025.12.29

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

475

2023.08.10

python设置中文版教程合集
python设置中文版教程合集

本专题整合了python改成中文版相关教程,阅读专题下面的文章了解更多详细内容。

1

2026.01.05

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Go 教程
Go 教程

共32课时 | 3.3万人学习

Go语言实战之 GraphQL
Go语言实战之 GraphQL

共10课时 | 0.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号