基于Windows CE.NET實(shí)現(xiàn)藍(lán)牙通訊模塊的幾種方法
利用P/Invoke方法編寫藍(lán)牙通訊模塊,DllI-port屬性非常有用。下面的代碼將用例子說(shuō)明此通用方案,例中托管程序?qū)⒄{(diào)用MessageBox(位于User32.lib中):
using
usingnamespaceSystem::Runtime::InteropSer-vices;
namespaceSysWin32
{
[DllImport(user32.dll,EntryPoint=MessageBox,CharSet=Unicode)]
intMessageBox(void*hWnd,wchar_t*lpText,wchar_t*lpCaption,unsignedintuType);
}
intmain()
SysWin32::MessageBox(0,LHelloworld!,LGreetings,0)
}
注意包含DllImport的代碼行。此代碼行根據(jù)參數(shù)值通知編譯器,使之聲明位于User32.dll中的函數(shù),并將簽名中出現(xiàn)的所有字符串(如參數(shù)或 返回值)視為Unicode字符串。如果缺少EntryPoint參數(shù),則默認(rèn)值為函數(shù)名。另外,由于CharSet參數(shù)指定Unicode,因此公共語(yǔ) 言運(yùn)行庫(kù)將首先查找稱為MessageBoxW的函數(shù)。如果運(yùn)行庫(kù)未找到此函數(shù),它將根據(jù)調(diào)用約定查找MessageBox以及相應(yīng)的修飾名。
當(dāng)調(diào)用用戶定義的DLL中所包含的函數(shù)時(shí),有必要將externC添加在DLL函數(shù)聲明之前,如下所示:externCSAMPLEDLL_APIintfnSam-pleDLL(void);
在調(diào)用非本機(jī)碼時(shí),需要注意的是要將非結(jié)構(gòu)化參數(shù)由托管封送處理為本機(jī)碼形式??梢岳肅harSet參數(shù)值的作用,將參數(shù)中字符串(string*類 型)都自動(dòng)轉(zhuǎn)換為wchar_t*。同樣,所有Int32參數(shù)類型轉(zhuǎn)換為非托管int,UInt32參數(shù)類型轉(zhuǎn)換為非托管unsignedint,而 Intl6參數(shù)類型轉(zhuǎn)換為了shortint。char*用于[in]參數(shù)的為String*(CharSet=Ansi),用于[out]參數(shù)或返回 值的為Text::StringBuilder*。wchar-t*用于[in]參數(shù)為String*(CharSet=Unicode),用于 [out]參數(shù)或返回值的為Text::StringBuilder*。需要注意的是函數(shù)指針必須具有_stdcall調(diào)用約定,這是因?yàn)檫@是 DllImport支持的唯一類型。對(duì)于數(shù)組來(lái)說(shuō)數(shù)組(如wchar_t*[]),CharSet參數(shù)僅應(yīng)用于函數(shù)參數(shù)的根類型。因此,無(wú)論 CharSet的值是什么,String*__gc[]將被封送處理為wchar_t*[]。除簡(jiǎn)單類型外,運(yùn)行庫(kù)還提供了一種機(jī)制,可以將簡(jiǎn)單結(jié)構(gòu) 由托管上下文封送處理為非托管上下文。簡(jiǎn)單結(jié)構(gòu)不包含任何內(nèi)部數(shù)據(jù)成員指針、結(jié)構(gòu)化類型的成員或其他元素。
在做一個(gè)關(guān)于藍(lán)牙通訊程序前,還需要一些關(guān)于藍(lán)牙的基礎(chǔ)知識(shí)。一個(gè)藍(lán)牙模塊程序需要包含開(kāi)啟藍(lán)牙,配對(duì),連接,建立串行通道,然后開(kāi)啟通訊過(guò)程,還需要在 應(yīng)用程序中設(shè)置串行端口。因?yàn)樗{(lán)牙技術(shù)有安全方面的設(shè)置,所以需要對(duì)藍(lán)牙設(shè)備進(jìn)行配對(duì)。藍(lán)牙的工作狀態(tài)總共有3種,分別為開(kāi)啟、關(guān)閉、可發(fā)現(xiàn)。并且所有的 通訊設(shè)備都必須有一個(gè)對(duì)應(yīng)的DeviceID,藍(lán)牙也不例外,藍(lán)牙的DeviceID是一串以“:”分隔的16進(jìn)制的數(shù)字。有了上述知識(shí),就可以在托管碼中利用P/Invoke方法開(kāi)始編寫藍(lán)牙通訊模塊了。
對(duì)應(yīng)的每一步需要調(diào)用的基本函數(shù)如下:
•獲取本地設(shè)備的ID
[DllImport(Btdrt.dll,SetLastError=true)]
publicstaticexternintBthReadLocalAddr(byte[]PBa)
•獲取遠(yuǎn)程設(shè)備的ID
[DllImport(ws2.dll,EntryPoint=WSALook-upServiceBegin,SetLastError=true)]
publicstaticexternintCeLookupServiceBegin(byte[]pQuerySet,LookupFlagsdwFlags,refintlphLookup)
•監(jiān)聽(tīng)服務(wù)
[DllImport(ws2.dll,EntryPoint=WSASetSer-vice,SetLastError=true)]
publicstaticexternintCeSetService
(byte[]pQuerySet,RNRSERVICE_REGISTER,LookupFlagsdwFlags)
•連接
[DllImport(mscoree,EntryPoint=@339)]
publicstaticexternintconnect(ints,byte[]name,intnamelen)
•藍(lán)牙的安全設(shè)置
獲取配對(duì)碼請(qǐng)求
[DllImport(Btdrt.dll,SetLastError=true)]
publicstaticexternintBthGetPINRequest(byte[]pba)
設(shè)置配對(duì)碼
[DllImport(btdrt.dll,SetLastError=true)
publicstaticexternintBthSetPIN(byte[]pba,intcPinLength,byte[]ppin)
創(chuàng)建ACL連接:
[DllImport(Btdrt.dll,SetLastError=true)
publicstaticexternintBthCreateACLConnection(byte[]pbt,refushortphandle);
然后是配對(duì)碼驗(yàn)證:
[DllImport(Btdrt.dll,SetLastError=true)]
publicstaticexternintBthAuthenticate(byte[]pbt);
然后一定要關(guān)閉連接:
[DllImport(Btdrt.dll,SetLastError=true)]
publicstaticexternintBthCloseConnection(ushorthandle);
•設(shè)置藍(lán)牙無(wú)線電狀態(tài)
[DllImport(BthUtil.dll,SetLastError=true)]publicstaticexternintBthSetMode(RadioModedwMode)
在建立好藍(lán)牙設(shè)備的連接后,就可以進(jìn)行兩個(gè)藍(lán)牙設(shè)備之間的通訊了。由于可以將藍(lán)牙通信當(dāng)作一個(gè)虛擬的串行通信來(lái)處理,所以在建立通訊的過(guò)程中可以采用類似于串口之間的通訊方式。而關(guān)于串口通訊這方面資料很多,本文就不具體詳述了。 linux操作系統(tǒng)文章專題:linux操作系統(tǒng)詳解(linux不再難懂)
c++相關(guān)文章:c++教程
評(píng)論