蓝牙技术-设备查询及服务发现

蓝牙技术作为短距离无线通讯技术的新生儿,有着其独特的技术优势,比如低成本、抗信号衰落性较好、使用2.4GHz的ISM频段,无需申请使用...由于其独特的技术优势,在无线通讯市场上有其发挥的大片天地。而蓝牙技术的使用更是如今火热的话题,本篇文章就先从蓝牙技术使用的第一步入手,先讲解如何利用蓝牙技术实现设备查询及相应设备的服务发现。
   --风小云

正如标题所示,设备查询及服务发现是使用蓝牙技术的第一步工作,只有这步工作能够处理好,才可以接下来使用Bluetooth(蓝牙) for Windows DK API中其他的类,从而实现我们需要完成的功能,传输文件、语音发送、fax...

运行实现环境:Windows 2000 sp4 Bluetooth for Windows DK API Reference Guide(WIDCOMM) VC++6.0英文版

首先介绍开发蓝牙工程所在环境的一些必要设置:
确认安装了提供Bluetooth开发的相应SDK,本例是Bluetooth for Windows DK API Reference Guide,WIDCOMM公司提供的,版本1.4.2.10
(接下来的设置因sdk的不同而不同,本例设置可以参照上述dk的帮助说明有关Implementation部分)
接着介绍几个关键的类:
CBtIf 提供接口级函数,请求及服务发现使用该类
CSdpService管理sdp服务记录
CSdpDiscoveryRec获取sdp发现记录并提供查询方式

---------------------------------
由于蓝牙技术的实现类似socket,必须有服务器端及客户端部分。有所不同的是蓝牙的服务器一般来说可以支持多个客户端,而不像p2p中只能有一个。
服务器端和客户端都可以作为连接的发起者,因此作为设备查询和服务发现来说,服务器端和客户端的实现完全一致,所不同的是服务器端必须有相应的服务提供,而客户端则可以没有任何服务支持。因本文的重点放在如何实现设备查询及服务发现,因此暂不考虑如何在服务器端添加相应服务(具体实现是通过上述CSdpService、CSdpDiscoveryRec)下面就来讲述sdp的具体实现。

具体来说实现主要都是通过类CBtIf完成的,下面有相应过程的简要分析:

首先,必须从CBtIf继承相应类的对象,这一步必须完成,否则无法使用sdk中的函数。
具体函数调用关系如下:
                  Client                                                         Server
        CBtIf::StartInquiry()---------------------------------->
   CBtIf::OnDeviceResponded()<-------------------------------
                     .
                     .
   CBtIf::OnDeviceResponded()<-------------------------------
   CBtIf::OnInquiryComplete()<---------------------------------      设备查询阶段

        CBtIf::StartDiscovery(address)---------------------->
  CBtIf::OnDiscoveryComplete()<------------------------------      服务发现阶段

  CBtIf::ReadDiscoveryRecords()

上述的调用关系比较明确,有几点需要说明,首先提出任务可以从客户端发起,同样服务器端也可以发起,上图是客户端发起。
一方调用StartInquiry()函数后,一旦有设备回应,则OnDeviceResponded()函数就被唤起。值得注意的是这个回应可以是同一设备多次回应,也可以是不同设备的不同回应。(主要因为同一设备名字和其他属性都可以作为不同独立的回应)需要你在该函数中加入相应处理。
等OnInquiryComplete()函数被调用,表示设备查找完成,意味着可以进行服务发现。
然后可以调用StartDiscovery(),这个函数调用时必须附带一具体设备的地址,这个地址在上面的设备查询过程中可以得到。
另外最后一点就是当服务发现完成后,得到数据的一方是通过调用ReadDiscoveryRecords()函数来读取具体的服务内容。
(上述函数都有相应的参数,这里省略,具体可以查询相应sdk)
下面给一段具体代码,便于大家理解上述过程。
---------------------------------
void CBTDlg::OnSearch()
{
 // TODO: Add your control notification handler code here
 if (!StartInquiry())  //调用
  SetDlgItemText(IDC_STATUS_TEXT,"ERROR--Unable to start device inquiry!");
 }

void CBTDlg::OnDeviceResponded (BD_ADDR bda, DEV_CLASS devClass, BD_NAME bdName, BOOL bConnected) //对回应进行相应处理
{
   
    // Add the device address and name (if any)
    // to the Server List.  It is OK to pass
    // duplicates.  This method will ignore them.
 CString item_text;
 int     item_count = m_ServerList.GetItemCount();  //listview control

 for(int x=0;x<item_count;x++)
 {
  CBdInfo* p_CmpInfo = (CBdInfo*)m_ServerList.GetItemData(x);
  if (p_CmpInfo->isBdAddrEqual(bda))
  {
   if (p_CmpInfo->m_Name.GetLength() == 0)
   {
    p_CmpInfo->m_Name = bdName;
    m_ServerList.SetItemText(x,0,p_CmpInfo->DeviceAsString());
   }
   return;
  }
 }
 CBdInfo* p_Info = new CBdInfo(bda,bdName);

 m_ServerList.InsertItem(item_count,p_Info->DeviceAsString());
 m_ServerList.SetItemData(item_count,(LPARAM)p_Info);

}

void CBTDlg::OnDiscovery()
{
 // TODO: Add your control notification handler code here
 StopInquiry();
//通过新线程完成服务发现
 m_pDiscoveryThread = AfxBeginThread(CBTDlg::DiscoverServices,this);
}

UINT CBTDlg::DiscoverServices(LPVOID pParam)
{
 CBTDlg* p_Dlg = static_cast<CBTDlg*>(pParam);
 CString strServiceName = "";
 POSITION pos = p_Dlg->m_ServerList.GetFirstSelectedItemPosition();
 if (pos!=NULL)
 {
  int nItem = p_Dlg->m_ServerList.GetNextSelectedItem(pos);
  CBdInfo* p_Info = (CBdInfo*)p_Dlg->m_ServerList.GetItemData(nItem);

  p_Dlg->SetDlgItemText(IDC_STATUS_TEXT,"Please Waiting for Services discovering...:"+p_Info->DeviceAsString());

  if (!p_Dlg->StartDiscovery(p_Info->m_BdAddr,NULL)) //开始服务发现
  {
   p_Dlg->SetDlgItemText(IDC_STATUS_TEXT,"Discovery Failed!"+p_Info->DeviceAsString());
  }
  else
  {
   WaitForSingleObject(p_Dlg->m_hEventStopDiscoveryThread,30000);//等待完成到达
   
   p_Dlg->SetDlgItemText(IDC_STATUS_TEXT,"Services on : "+p_Info->DeviceAsString());
   p_Dlg->m_ServerList.DeleteAllItems();
   p_Dlg->m_ServerList.DeleteColumn(0);
   p_Dlg->m_ServerList.InsertColumn(0,"Service",LVCFMT_CENTER,180);
   
   if (!p_Dlg->m_bDialogClosed)
   {
    CSdpDiscoveryRec sdp_Record[MAX_SERVICE];
    int nServiceNumber = 0;
    nServiceNumber = p_Dlg->ReadDiscoveryRecords(p_Info-   >m_BdAddr,MAX_SERVICE,sdp_Record);      //读取具体服务内容

    if (nServiceNumber != 0)
    {
     for(int i=0;i<nServiceNumber;i++)
     {
      strServiceName = sdp_Record[i].m_service_name;
      p_Dlg->m_ServerList.InsertItem(i,strServiceName);
     }
    }
    else
     p_Dlg->m_ServerList.InsertItem(0,"None!");
  
    p_Dlg->m_ServerList.EnableWindow(FALSE);
   }
   else
    SetEvent(p_Dlg->m_hEventKillDiscoveryThread);
  }
 } 
 p_Dlg->m_pDiscoveryThread = NULL;
 return 0;
}

通过上面的介绍,只有掌握了蓝牙技术如何实现设备查询及服务发现后,才可以正确确认对方是否提供有相应的服务内容,从而为下一步使用其他传输或通讯技术打下基础。
                                                                         --风小云原创,转贴请注明出处!

你可能感兴趣的:(windows,api,服务器,service,reference,通讯)