首页 > 基础资料 博客日记
java调用海康威视SDK实现车牌识别
2024-10-15 20:00:06基础资料围观73次
本篇文章分享java调用海康威视SDK实现车牌识别,对你有帮助的话记得收藏一下,看Java资料网收获更多编程知识
我采用的是 报警布防方式 SDK版本为 CH-HCNetSDKV6.1.9.48_build20230410_win32
如何引用dll 我用的是jna 就不描述了 SDK在官网自行下载 以下代码亲测可用 自行参考~
1.1接口调用流程
虚线框的内容是可选的,设备事先安装配置好,能力集和配置接口可不调用,不会影响其他接口功能的使用。
- 初始化NET_DVR_Init接口在程序开始是调用,只需要调用一次。
- 用户注册即登录设备,调用NET_DVR_Login_V40接口,每一台设备只需要登录一次。
- 登录设备之后,可以通过NET_DVR_GetDeviceAbility获取智能交通能力集可以判断设备是否支持相关功能,能力集类型:DEVICE_ABILITY_INFO,能力集:ITDeviceAbility,节点:<ITCAbility>。
- 车牌识别包括IO触发、虚拟线圈触发等自动触发抓拍模式和网络触发等手动抓拍模式。
如果是自动触发模式,一般建议通过WEB网页登录设备进行配置调试,也可以通过SDK接口进行配置,相关接口:NET_DVR_GetDVRConfig(命令:NET_ITC_GET_TRIGGERCFG)、NET_DVR_SetDVRConfig(命令:NET_ITC_SET_TRIGGERCFG)、NET_DVR_GetDeviceConfig(命令:NET_DVR_GET_TRIGGEREX_CFG)、NET_DVR_SetDeviceConfig(命令:NET_DVR_SET_TRIGGEREX_CFG)等。
如果是手动抓拍,有两种方式:1)通过NET_DVR_ManualSnap可以在接口直接返回结果信息;2)通过NET_DVR_ContinuousShoot发送网络触发连拍命令,抓拍结果跟自动触发模式一样,通过报警布防方式在报警回调函数里面返回。 - 配置好设备相关参数,车辆通过时进行抓拍和识别,结果信息通过报警布防方式获取,具体实现方法:
1) 先调用NET_DVR_SetDVRMessageCallBack_V50设置报警回调函数(V31、V30接口也支持,新接口兼容老接口),在SDK初始化之后即可以调用,多台设备对接时也只需要调用一次设置一个回调函数,回调函数里面接收数据之后可以通过报警设备信息(NET_DVR_ALARMER)中lUserID等参数判断区分设备。
2) 每台设备分别登录,分别调用NET_DVR_SetupAlarmChan_V41进行布防,布防即建立设备跟客户端之间报警上传的连接通道,这样设备发生报警之后通过该连接上传报警信息,SDK在报警回调函数中接收和处理报警信息数据即可。对接智能交通摄像机,布防时可以选择布防等级,一级布防(byLevel为0)最大连接数为1个,二级布防(byLevel为1)最大连接数为3个,如果已经达到上限了,再布防就会失败,SDK将返回28的错误号。
3) 程序退出前或者不需要接收报警信息时调用NET_DVR_CloseAlarmChan_V30进行撤防,释放资源,此时连接断开,设备将不再上传报警信息。 - 车牌识别的报警信息类型为COMM_ITS_PLATE_RESULT(新报警信息)和COMM_UPLOAD_PLATE_RESULT(老报警信息),分别对应接口NET_DVR_SetupAlarmChan_V41中布防参数byAlarmInfoType=1和byAlarmInfoType=0。建议使用新的报警信息类型。
1)设备是否支持新报警信息可从注册返回的能力获知,详见NET_DVR_DEVICEINFO_V30结构中bySupport1&0x80(表示是否支持车牌新报警信息),如果注册返回能力不支持,设备仅支持老报警信息上传。
2)COMM_UPLOAD_PLATE_RESULT:兼容旧型号(976/986/966等)抓拍机,一次回调只上传一张图片信息,对应报警信息结构体:NET_DVR_PLATE_RESULT。
3)COMM_ITS_PLATE_RESULT:应用于违章图片组一次性上传以及合成图片上传等新功能,对应报警信息结构体:NET_ITS_PLATE_RESULT。 - 退出程序时调用NET_DVR_Logout注销设备,每一台设备调用一次。最后调用NET_DVR_Cleanup释放SDK所有资源。
1.2示例代码
初始化+登录
private boolean init(String ip, int port, String userName, String password) {
/**加载日志*/
if (!NET_DVR_Init()) {
logger.error("摄像头初始化失败!错误码为" + hCNetSDK.NET_DVR_GetLastError());
return false;
}
hCNetSDK.NET_DVR_SetLogToFile(3, "./sdklog", false);
userId = NET_DVR_Login_V30(ip, port, userName, password);
if (userId > -1) {
isinitialized = true;
}
return isinitialized;
}
设置回调+布防
public boolean licensePlateRecognition(CarNumRecognizerConfig carNumRecognizerConfig) {
logger.debug("*********************licensePlateRecognition**************************");
String ip = carNumRecognizerConfig.getCarNumRecognizerIP();
int port = carNumRecognizerConfig.getPort();
String userName = carNumRecognizerConfig.getUserName();
String password = carNumRecognizerConfig.getPassword();
if (StringUtil.isNotBlank(carNumRecognizerConfig.getEncoding())) {
encode = carNumRecognizerConfig.getEncoding();
}
if (StringUtil.isNotBlank(carNumRecognizerConfig.getImageDir())) {
imageDirPath = carNumRecognizerConfig.getImageDir();
}
if (!init(ip, port, userName, password)) {
logger.error("摄像头初始化失败!");
return false;
}
carNumRecognizerDevices = deviceManager.getCarNumRecognizerDevices();
//设置连接时间与重连时间
hCNetSDK.NET_DVR_SetConnectTime(2000, 1);
hCNetSDK.NET_DVR_SetReconnect(100000, true);
//设备信息, 输出参数
//设置报警回调函数
if (fMSFCallBack_V31 == null) {
fMSFCallBack_V31 = new FMSGCallBack_V31();
String UserData = "UserData";
HCNetSDK.BYTE_ARRAY UserDataByte = new HCNetSDK.BYTE_ARRAY(25);
UserDataByte.read();
UserDataByte.byValue = UserData.getBytes();
UserDataByte.write();
Pointer pUserDataByte = UserDataByte.getPointer();
if (!hCNetSDK.NET_DVR_SetDVRMessageCallBack_V31(fMSFCallBack_V31, pUserDataByte)) {
logger.error("设置回调函数失败!+" + hCNetSDK.NET_DVR_GetLastError());
return false;
} else {
logger.info("设置回调函数成功!");
}
}
HCNetSDK.NET_DVR_LOCAL_GENERAL_CFG struNET_DVR_LOCAL_GENERAL_CFG = new HCNetSDK.NET_DVR_LOCAL_GENERAL_CFG();
struNET_DVR_LOCAL_GENERAL_CFG.byAlarmJsonPictureSeparate = 1; //设置JSON透传报警数据和图片分离
struNET_DVR_LOCAL_GENERAL_CFG.write();
Pointer pStrNET_DVR_LOCAL_GENERAL_CFG = struNET_DVR_LOCAL_GENERAL_CFG.getPointer();
hCNetSDK.NET_DVR_SetSDKLocalCfg(17, pStrNET_DVR_LOCAL_GENERAL_CFG);
//尚未布防,需要布防
if (lAlarmHandle < 0) {
//报警布防参数设置
HCNetSDK.NET_DVR_SETUPALARM_PARAM m_strAlarmInfo = new HCNetSDK.NET_DVR_SETUPALARM_PARAM();
m_strAlarmInfo.dwSize = m_strAlarmInfo.size();
m_strAlarmInfo.byLevel = 0; //布防等级
m_strAlarmInfo.byAlarmInfoType = 1; // 智能交通报警信息上传类型:0- 老报警信息(NET_DVR_PLATE_RESULT),1- 新报警信息(NET_ITS_PLATE_RESULT)
m_strAlarmInfo.byDeployType = 1; //布防类型:0-客户端布防,1-实时布防
m_strAlarmInfo.write();
lAlarmHandle = hCNetSDK.NET_DVR_SetupAlarmChan_V41(Integer.valueOf(userId + ""), m_strAlarmInfo);
logger.info("lAlarmHandle: " + lAlarmHandle);
if (lAlarmHandle == -1) {
logger.info("布防失败,错误码为" + hCNetSDK.NET_DVR_GetLastError());
return false;
} else {
logger.info("布防成功");
}
} else {
logger.info("设备已经布防,请先撤防!");
}
return true;
}
回调
public class FMSGCallBack_V31 implements HCNetSDK.FMSGCallBack_V31 {
//报警信息回调函数
public boolean invoke(int lCommand, HCNetSDK.NET_DVR_ALARMER pAlarmer, Pointer pAlarmInfo, int dwBufLen, Pointer pUser) {
logger.info("报警事件类型: lCommand:" + Integer.toHexString(lCommand));
//lCommand是传的报警类型
switch (lCommand) {
case 3058:
logger.info("报警事件类型: 0x3058 车辆黑白名单数据需要同步报警上传");
case HCNetSDK.COMM_UPLOAD_PLATE_RESULT:
logger.info("报警事件类型: COMM_UPLOAD_PLATE_RESULT");
case HCNetSDK.COMM_ITS_PLATE_RESULT://交通抓拍结果(新报警信息)
logger.info("报警事件类型: COMM_ITS_PLATE_RESULT");
HCNetSDK.NET_ITS_PLATE_RESULT strItsPlateResult = new HCNetSDK.NET_ITS_PLATE_RESULT();
strItsPlateResult.write();
Pointer pItsPlateInfo = strItsPlateResult.getPointer();
pItsPlateInfo.write(0, pAlarmInfo.getByteArray(0, strItsPlateResult.size()), 0, strItsPlateResult.size());
strItsPlateResult.read();
try {
String sLicense = new String(strItsPlateResult.struPlateInfo.sLicense, encode);
String carNum = StringUtil.extractTheLicensePlateNumber(sLicense);//这个方法是提取车牌号 因为我接收到的车牌号会带颜色在前面 比如 蓝京A---- 这样子
logger.info("识别车号:{},提取车号:{}",sLicense,carNum);
CarNumRecognizerResult carNumRecognizerResult = new CarNumRecognizerResult();
carNumRecognizerResult.setCarNum(carNum);
boolean isOK = carNumRecognizerDevices.notifyCarNumResult(carNumRecognizerResult);
logger.info("isOK:::" + isOK);
} catch (UnsupportedEncodingException e1) {
e1.printStackTrace();
logger.error(e1.getMessage(),e1);
} catch (IOException e) {
e.printStackTrace();
logger.error(e.getMessage(),e);
}
/**
* 报警图片保存,车牌,车辆图片
*/
if (StringUtil.isNotBlank(imageDirPath)){
File imageDir = new File(imageDirPath);
if(!imageDir.exists() || !imageDir.isDirectory()){
imageDir.mkdirs();
}
for (int i = 0; i < strItsPlateResult.dwPicNum; i++) {
if (strItsPlateResult.struPicInfo[i].dwDataLen > 0) {
String baseName = DateUtil.getTimeStr((int)(System.currentTimeMillis()/1000), "yyyyMMddHHmmss");
File imageFile = getImageFilePath(i+"", baseName, imageDir);
FileOutputStream fout;
try {
fout = new FileOutputStream(imageFile);
//将字节写入文件
long offset = 0;
ByteBuffer buffers = strItsPlateResult.struPicInfo[i].pBuffer.getByteBuffer(offset, strItsPlateResult.struPicInfo[i].dwDataLen);
byte[] bytes = new byte[strItsPlateResult.struPicInfo[i].dwDataLen];
buffers.rewind();
buffers.get(bytes);
fout.write(bytes);
fout.close();
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
break;
default:
logger.info("报警类型:" + Integer.toHexString(lCommand));
break;
}
return true;
}
}
重点是这里 我起初选择0 就不起作用 可能是客户端没配置好?不懂
m_strAlarmInfo.byDeployType = 1; //布防类型:0-客户端布防,1-实时布防
总结两点:
1 确认摄像机客户端中可以识别到车号!!!
2 确认使用哪种布防!!!
再提一嘴 我个人认为海康的车牌识别做的不好,因为我开发其他车牌识别产品 例如文通、VM、大华 这些设备的车牌识别都是采用自动识别 主动推送的方式到http接口 更好处理 还不占资源。
海康也有监听方法 但我没研究明白 不会用..
文章来源:https://blog.csdn.net/m0_59680416/article/details/139957902
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:jacktools123@163.com进行投诉反馈,一经查实,立即删除!
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:jacktools123@163.com进行投诉反馈,一经查实,立即删除!
标签: