经过上一篇博客的介绍,相信大家已经搭建起Onvif的开发框架。这篇博客将介绍一个Onvif的简单应用----设备搜索。博主用的IDE是Visual Studio 2005。
工程中包含了之前生成的Onvif框架里的soapH.h、soapStub.h、stdsoap2.h、soapC.cpp、 stdsoap2.cpp、soapwsddProxy.h、soapwsddProxy.cpp、wsdd.nsmap。因为实现的是客户端程序,所以包含了soapwsddProxy.h和soapwsddProxy.cpp,如果要实现服务器端程序应包含 soapwsddService.h和soapwsddService.cpp。另外还需包含duration.h和duration.cpp,否则编译报错。编译前请在包含的Onvif的.cpp文件加上#include "stdafx.h"头文件。
编译时可能遇到的问题:
编译时如遇到1>stdsoap2.obj : error LNK2001: 无法解析的外部符号 _namespaces,请在stdsoap2.cpp中加上 #include "wsdd.nsmap"即可通过编译。
编译时如果遇到fatal error C1128: 节数超过对象文件格式限制: 请使用 /bigobj 进行编译,请在项目-属性-通用属性-C/C++-命令行的附加选项框里加上/bigobj即可通过编译。
void CdevicediscoveryDlg::OnBnClickedOk()
{
SetDlgItemText(IDC_STATUS, _T("正在搜索设备,大约需要5秒,请耐心等待!"));
SetTimer(1, 5000, NULL);
m_list.DeleteAllItems();
wsddProxy discovery;
//获取guid(windows下叫guid,linux下叫uuid),格式为urn:uuid:8-4-4-4-12,由系统随机产生
static char buf[64] = {0};
GUID guid;
if (S_OK == ::CoCreateGuid(&guid))
{
//如果guid生成成功,则将其转为字符串,保存在buf中
_snprintf_s(buf, sizeof(buf)
, "{%08X-%04X-%04x-%02X%02X-%02X%02X%02X%02X%02X%02X}"
, guid.Data1
, guid.Data2
, guid.Data3
, guid.Data4[0], guid.Data4[1]
, guid.Data4[2], guid.Data4[3], guid.Data4[4], guid.Data4[5]
, guid.Data4[6], guid.Data4[7]
);
}
//超过5秒没有数据就退出
discovery.soap->recv_timeout = 5;
//给头赋值
char* wsa_To = "urn:schemas-xmlsoap-org:ws:2005:04:discovery";
char* wsa_Action = "http://schemas.xmlsoap.org/ws/2005/04/discovery/Probe";
discovery.soap_header(buf, NULL, NULL, NULL, NULL, wsa_To, wsa_Action,
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
//设置所需寻找设备的类型和范围,二者至少设定一个,否则可能收到非ONVIF设备,出现异常
discovery.soap_endpoint = "soap.udp://239.255.255.250:3702";
struct wsdd__ProbeType wsdd__Probe;
//设置所需设备的类型,tdn为命名空间前缀,为wsdd.nsmap文件中
//{"tdn","http://www.onvif.org/ver10/network/wsdl"}的tdn,
//如果不是tdn,而是其它,如ns1,这里也要随之改为ns1
wsdd__Probe.Types = "tdn:NetworkVideoTransmitter";
int result = 0, count = 0;
//通过组播发送Probe探针,发送成功返回0,否则返回-1
result = discovery.send_Probe(discovery.soap_endpoint, NULL, &wsdd__Probe);
if(-1 == result){
AfxMessageBox(_T("Probe发送失败!"));
return;
}
else{
struct __wsdd__ProbeMatches tmp;
while(1){
//接收ProbeMatches,成功返回0,否则返回-1
result = discovery.recv_ProbeMatches(tmp);
if(-1 == result){
//AfxMessageBox(_T("Probe接收失败!"));
return;
}
else{
//将接收到的数据插入列表中
CString str;
str.Format(_T("%d"), count + 1);
m_list.InsertItem(count, str);
str = tmp.wsdd__ProbeMatches->ProbeMatch->XAddrs;
m_list.SetItemText(count, 1, str);
int val = tmp.wsdd__ProbeMatches->ProbeMatch->MetadataVersion;
str.Format(_T("%d"), val);
m_list.SetItemText(count, 2, str);
str = tmp.wsdd__ProbeMatches->ProbeMatch->Types;
m_list.SetItemText(count, 3, str);
str = tmp.wsdd__ProbeMatches->ProbeMatch->wsa__EndpointReference.Address;
m_list.SetItemText(count, 4, str);
str = tmp.wsdd__ProbeMatches->ProbeMatch->Scopes->__item;
m_list.SetItemText(count, 5, str);
++count;
}
}
//清除变量
discovery.destroy();
}
}
void CdevicediscoveryDlg::OnTimer(UINT_PTR nIDEvent)
{
SetDlgItemText(IDC_STATUS, _T("搜索设备完成!"));
CDialog::OnTimer(nIDEvent);
}
完整代码可从该处下载(http://download.csdn.net/detail/u011110151/9487316)。