.net 调用海康SDK的常用操作封装

  • 📢欢迎点赞 :👍 收藏 ⭐留言 📝 如有错误敬请指正,赐人玫瑰,手留余香!
  • 📢本文作者:由webmote 原创
  • 📢作者格言:新的征程,我们面对的不仅仅是技术还有人心,人心不可测,海水不可量,唯有技术,才是深沉黑夜中的一座闪烁的灯塔

序言

上篇海康SDK使用以及常见的坑受到了许多网友的喜爱,这也说明了在工控领域内,使用.net开发还是非常便捷省事的。 结合硬件的开发经验来谈语言,也是非常切合实用主义的,这里继续上篇未完成的事情,对海康威视的SDK进行进一步封装,已解决其在x86和x64系统执行时的疑难杂症,并且对海康的SDK进行进一步封装,第一版代码发在github上,供大家测试和使用。

声明下,海康威视没有给赞助费,希望厂家能够看到,给点打赏,哈哈~~~

在这里插入图片描述

1. 支持x86和x64的运行时自动加载

做过开发的同学都知道,我们目前的操作系统分为x86和x64两个版本,海康SDK是基于C++编写,在编译时已经定义了系统的操作版本,因此需要我们在开发时根据自己的运行环境载入不同的SDK版本。

这是不是有些复杂,一不小心就掉入了不能加载正确的dll镜像的陷阱中?有没有更方便的办法呢?

当然是有的,基于.net Framework版本的应用可以采用一种巧妙的方式绕开[DllImport] 属性的载入限制, 当然如果是.net core 3以后的版本,还有更精妙的办法来适应linux平台的方式,这里按下不表,暂时聚焦在.net Framework 4运行环境下的载入方式。

大家都知道,DllImportAttribute是编译时确定路径的,因此无法在运行时进行dll路径的设置,所以我们利用windows系统的SetDllDirectory函数,来解决加载时自动搜索的路径,这里巧妙的设置加载路径,便绕开了这个限制。

 public HkCamera(){_sdkInit = CHCNetSDK.NET_DVR_Init();            }static HkCamera(){//设置搜索路径,兼容x86 和 x64var path = Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location);path = Path.Combine(path,"Lib", IntPtr.Size == 8 ? "x64" : "x86");bool ok = SetDllDirectory(path);if (!ok) throw new System.ComponentModel.Win32Exception();}
[DllImport("kernel32.dll", CharSet = CharSet.Auto, SetLastError = true)]private static extern bool SetDllDirectory(string path);

在海康原始的SDK文件中,我们需要检查引入类库部分,让它们都采用相对路径,如下:

 [DllImportAttribute(@".\HCNetSDK.dll")]public static extern int NET_DVR_SendWithRecvRemoteConfig(...

这里运行时的环境,简单的使用IntPtr.Size == 8 ? "x64" : "x86"进行检测,简单使用便捷!

2. 海康设备的在线检测

经过仔细阅读海康的SDK帮助手册,海康在新的SDK中引入了在线检测函数,这里提供新版海康SDK的在线检测方案,以替换上篇文章中阐述的方法。

当然如果是使用老版本的SDK,可以保留之前的方式,效果是一样的。

CHCNetSDK.NET_DVR_RemoteControl(_userId, CHCNetSDK.NET_DVR_CHECK_USER_STATUS, IntPtr.Zero, 0);

使用NET_DVR_CHECK_USER_STATUS来检测设备是否在线,经测试,断联后检测的时间会稍微长点。

3. 封装登录和登出

海康SDK在使用时,需要对SDK进行初始化,在使用完毕后,需要释放资源,这里可以继承IDisposable接口,实现释放接口,以便销毁该类时,自动卸载这个非托管资源,避免资源泄漏。

protected virtual void Dispose(bool disposing)
{if (!_disposedValue){if (disposing){// TODO: 释放托管状态(托管对象)}                if (_sdkInit){if (_userId >= 0){Logout();}try{CHCNetSDK.NET_DVR_Cleanup();}catch { }}_disposedValue = true;}
}

登录和登出是使用海康视频,录像机等设备必须做的操作,这里也进行了统一封装,方便上层调用。

public bool Login(CameraLoginInfo cameraLoginInfo){try{if (_userId >= 0){Logout();}_loginInfo = cameraLoginInfo;var loginInfo = new CHCNetSDK.NET_DVR_USER_LOGIN_INFO();loginInfo.sDeviceAddress = cameraLoginInfo.IP;loginInfo.wPort = cameraLoginInfo.Port;loginInfo.sUserName = cameraLoginInfo.UserName;loginInfo.sPassword = cameraLoginInfo.Password;                //是否异步登录:0- 否,1- 是 loginInfo.bUseAsynLogin = false;var _deviceInfo = new CHCNetSDK.NET_DVR_DEVICEINFO_V40();_userId = CHCNetSDK.NET_DVR_Login_V40(ref loginInfo, ref _deviceInfo);_serailNumber =  _deviceInfo.struDeviceV30.sSerialNumber;return _userId >= 0;}catch {return false;}}public bool IsOnline(){try{return CHCNetSDK.NET_DVR_RemoteControl(_userId, CHCNetSDK.NET_DVR_CHECK_USER_STATUS, IntPtr.Zero, 0);}catch { return false; }}public void Logout(){try{if (_userId >= 0){CHCNetSDK.NET_DVR_Logout(_userId);_userId = -1;}}catch { }}private bool CheckLogin(){if (IsOnline()) return true;return Login(_loginInfo);}

这里定义了登录的接口类,限定了登录需要的资源,包含IP和端口,用户名和密码,以及默认的视频通道号。

端口默认为8000,因此初始化时可以不用配置。

/// <summary>/// 海康DVR登录信息/// </summary>public class CameraLoginInfo{/// <summary>/// 摄像头的IP地址/// </summary>public string IP { get; set; }/// <summary>/// 登录端口号/// </summary>public ushort Port { get; set; } = 8000;/// <summary>/// 登录用户名/// </summary>public string UserName { get; set; }/// <summary>/// 登录密码/// </summary>public string Password { get; set; }/// <summary>/// 默认通道/// </summary>public int ChannelNo { get; set; } = 1;}

4.云台控制

海康监控相机最常用到的应该是云台控制和抓取图片了,这里也进行了封装,当然这里还缺少对硬盘录像机,红外测温等的封装,后续将一一完善。

云台的控制基本是一样的,都是配对操作,一个开始,一个停止。

 /// <summary>/// 控制云台,设备接收到控制命令后直接返回成功。不关心云台是否进行相应的动作/// </summary>/// <param name="cmd"></param>/// <param name="speed">取值范围[1,7] </param>/// <returns></returns>public bool StartPTZControl(PtzCommand cmd, Int32 speed = 4){try{if (!CheckLogin()) return false;var result = CHCNetSDK.NET_DVR_PTZControlWithSpeed_Other(_userId,_loginInfo.ChannelNo, (uint)cmd, 0, (uint)speed);return result;}catch {return false;}}/// <summary>/// 停止云台/// </summary>/// <param name="cmd"></param>/// <returns></returns>public bool StopPTZControl(PtzCommand cmd){try{if (!CheckLogin()) return false;var result = CHCNetSDK.NET_DVR_PTZControlWithSpeed_Other(_userId, _loginInfo.ChannelNo, (uint)cmd, 1, 4);return result;}catch{return false;}}

注意,所有操作没有抛出异常,因此外部使用的时候,不需要进行try/catch,如果需要提示错误信息,那么需要调用 GetLastError去获取即可。

这里还是有优化空间的,标准的做法是抛出异常方式,定义一堆异常类,不知道大家能接受那种方式呢?

5.抓取图片

最后一步,对抓取图片进行封装,这里需要你提供合适的文件路径和名称。

/// <summary>/// 抓图/// </summary>/// <param name="fileName"></param>/// <returns></returns>public bool CapturePicture(string fileName){if (!CheckLogin()) return false;try{FileInfo fi = new FileInfo(fileName);if (fi.Exists){File.Delete(fi.FullName);}if (!Directory.Exists(fi.DirectoryName)){Directory.CreateDirectory(fi.DirectoryName);}CHCNetSDK.NET_DVR_JPEGPARA lpJpegPara = new CHCNetSDK.NET_DVR_JPEGPARA{wPicQuality = 0,wPicSize = 0xff};return CHCNetSDK.NET_DVR_CaptureJPEGPicture(_userId, _loginInfo.ChannelNo, ref lpJpegPara, fileName);}catch {return false;}}

总结

海康SDK的封装还是比较繁琐的,主要在如下几个方面:

  1. 兼容x86和x64
  2. 错误的处理方式包装
  3. 原始的SDK,需要人性化包装改造
  4. 需要硬件进行全面测试
  5. 大量的接口,只能先根据使用场景进行裁剪,后续的改善跟进。

你学废了吗?

👓都看到这了,还在乎点个赞吗?

👓都点赞了,还在乎一个收藏吗?

👓都收藏了,还在乎一个评论吗?

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/28576.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

2024/6/16周报

文章目录 摘要Abstract文献阅读题目问题本文贡献方法aGNN输入和输出模块嵌入模块编码器和解码器模块&#xff1a;支持多头注意的GCN多头自注意力机制GCN模型解释&#xff1a;SHAP 案例研究地下水流动与污染物运移模型研究场景设计 数据集实验结果 代码复现结论 摘要 本周阅读了…

whisper 模型源码解读

whisper官方源码 whisper 模型官方代码&#xff1a;https://github.com/openai/whisper/blob/main/whisper/model.py &#xff1b;注释如下 import base64 import gzip from dataclasses import dataclass from typing import Dict, Iterable, Optionalimport numpy as np impo…

java设计模式和面向对象编程思想

Java设计模式和面向对象编程思想是软件开发中的核心概念&#xff0c;对于构建可维护、可扩展的软件系统至关重要。下面是对这两个主题的知识点总结&#xff1a; 面向对象编程&#xff08;OOP&#xff09;思想 封装&#xff1a;将数据&#xff08;属性&#xff09;和操作这些数据…

享元和代理模式

文章目录 享元模式1.引出享元模式1.展示网站项目需求2.传统方案解决3.问题分析 2.享元模式1.基本介绍2.原理类图3.外部状态和内部状态4.类图5.代码实现1.AbsWebSite.java 抽象的网站2.ConcreteWebSite.java 具体的网站&#xff0c;type属性是内部状态3.WebSiteFactory.java 网站…

CSS从入门到精通——动画:CSS3动画执行次数和逆向播放

目录 任务描述 相关知识 动画执行次数 动画反向播放 编程要求 任务描述 本关任务&#xff1a;用 CSS3 实现loading效果。效果图如下&#xff1a; 相关知识 为了完成本关任务&#xff0c;你需要掌握&#xff1a;1.动画执行次数&#xff0c;2.动画反向播放。 需要实现的效…

Go 的 netpoll 如何避免洪泛攻击

Go 的 netpoll&#xff08;网络轮询器&#xff09;组件在其网络库中扮演了一个关键角色&#xff0c;它用来高效地处理大量的网络事件&#xff0c;特别是在高并发环境下。 然而&#xff0c;防止洪泛攻击&#xff08;如 SYN Flood、UDP Flood&#xff09;并不仅仅是 netpoll 本身…

持续总结中!2024年面试必问 20 道并发编程面试题(九)

上一篇地址&#xff1a;持续总结中&#xff01;2024年面试必问 20 道并发编程面试题&#xff08;八&#xff09;-CSDN博客 十七、请解释什么是Callable和FutureTask。 Callable和FutureTask是Java并发API中的重要组成部分&#xff0c;它们用于处理可能产生结果的异步任务。 …

Oracle中select取值后的字符串处理

取一对括号中间内容 SELECT SUBSTR(column_name, INSTR(column_name, () 1, INSTR(column_name, )) - INSTR(column_name, () - 1) AS content_between_parentheses FROM table_name; 某字符值中abc之后def之前部分 SELECTSUBSTR(your_column, INSTR(your_column, abc) LEN…

R调用Taxonkit展示系统发育信息

Introduction TaxonKit是一个用于处理生物分类学数据的命令行工具。 它的主要功能是处理NCBI的生物分类学数据&#xff0c;包括对分类单元&#xff08;如物种、属、科等&#xff09;的查找、分类单元的上下位关系查询、分类单元名称的标准化等。 为了方便R社区用户&#xff0…

Spring-web-HttpServletRequest

在spring mvc中&#xff0c;有2种方法获取当前http请求的request&#xff0c;分别如下 方式1&#xff1a; spring mvc将当前request从抽象类FrameworkServlet的processRequest方法中放进去&#xff0c;然后我们通过下面代码拿出来 HttpServletRequest request ((ServletRequ…

【计算机组成原理】指令系统考研真题详解之拓展操作码!

计算机组成原理&#xff1a;指令系统概述与深入解析 1. 指令系统概述 计算机软硬件界面的概念 在计算机组成原理中&#xff0c;指令系统扮演着至关重要的角色&#xff0c;它是计算机软硬件界面的核心。软件通过指令与硬件进行通信&#xff0c;硬件根据指令执行相应的操作。指…

装饰者模式(设计模式)

装饰模式就是对一个类进行装饰&#xff0c;增强其方法行为&#xff0c;在装饰模式中&#xff0c;作为原来的这个类使用者还不应该感受到装饰前与装饰后有什么不同&#xff0c;否则就破坏了原有类的结构了&#xff0c;所以装饰器模式要做到对被装饰类的使用者透明&#xff0c;这…

如何解决javadoc一直找不到路径的问题?

目录 一、什么是javadoc二、javadoc为什么会找不到路径三、如何解决javadoc一直找不到路径的问题 一、什么是javadoc Javadoc是一种用于生成Java源代码文档的工具&#xff0c;它可以帮助开发者生成易于阅读和理解的文档。Javadoc通过解析Java源代码中的注释&#xff0c;提取其…

英语角情景设计方案

背景介绍 在一个充满活力的清晨&#xff0c;项目团队汇集在线上会议中&#xff0c;准备开始他们定期的项目审查会议。团队成员不仅包括项目经理&#xff08;PM&#xff09;、客户代表&#xff0c;还有会计&#xff08;AM&#xff09;、设计师、前端开发者、后端开发者以及测试…

【CSS】background-size属性的作用是什么,怎么使用

background-size属性的作用&#xff1a; background-size属性在CSS中用于设置背景图像的尺寸。在CSS3之前&#xff0c;背景图片的尺寸是由图片的实际尺寸决定的。而在CSS3中&#xff0c;background-size属性允许我们规定背景图片的尺寸&#xff0c;这使得我们能够在不同的环境…

python命令行参数异常处理的方法:Exception getopt.GetoptError异常

一. 简介 前一篇文章学习了 针对 python的命令行参数进行处理时&#xff0c;提供的模块: getopt模块。文章如下&#xff1a; python3的基本语法&#xff1a;命令行参数-CSDN博客 本文继续学习 getopt模块的一个方法 getopt.GetoptError。 二. python命令行参数&#xff1a…

【Python】理解『下采样』:原理与应用

是你多么温馨的目光 教我坚毅望着前路 叮嘱我跌倒不应放弃 没法解释怎可报尽亲恩 爱意宽大是无限 请准我说声真的爱你 &#x1f3b5; Beyond《真的爱你》 在数字信号处理、图像处理和机器学习中&#xff0c;下采样&#xff08;Downsampling&#xff09;是…

42 mysql “+“ 操作符的实现

前言 问题来自于 chinaunix, mysql select 子句的小白问题 mysql 的一些基础的 算术运算符 的计算的实现 这里 整理如下 case, 执行之前 设置如下变量 set a 2; set b 3;select a b; select a b; select 1 3; select 1 3; select a b; select a b; select a b; …

【Quartus 13.0】NIOS II 部署UART 和 PWM

打算在 EP1C3T144I7 芯片上部署 nios ii 做 uart & pwm控制 这个芯片或许不够做 QT 部署 这个芯片好老啊&#xff0c;但是做控制足够了&#xff0c;我只是想装13写 leader给的接口代码是用VHDL写的&#xff0c;我不会 当然verilog我也不太会 就这样&#xff0c;随便写吧 co…

element-plus表单组件之自动补全组件el-autocomplete和级联选择器组件el-cascader

el-autocomplete 自动补全组件 自补全组件的功能和可以根据输入过滤的el-select组件有些类似。 fetch-suggestions 根据输入框的输入获取建议的内容&#xff0c;其接受值是一个函数&#xff0c;有2个参数&#xff0c;querystring:输入的内容&#xff0c;callback内置函数&…