在Windows CE.NET的開(kāi)發(fā)上,目前采用微軟公司2005年年底推出的Visual Studio 2005是最理想的選擇。用 Visual Studio 2005可以進(jìn)行利用C#,或VB.NET等語(yǔ)言開(kāi)發(fā)基于.NET Compact Framework 2.0或1.0的 托管碼WinCE程序,也可以利用C++語(yǔ)言來(lái)開(kāi)發(fā)基于MFC,ATL或Win32API的本機(jī)碼WinCE程序。
因?yàn)樵谖④?a href="http://www.brongaenegriffin.com/article/zt/" target="_blank">最新的.NET Compact Framework 2.0的類(lèi)庫(kù)中還未包含針對(duì)藍(lán)牙通訊模塊的類(lèi)庫(kù),而且目前關(guān)于在Windows CE中開(kāi)發(fā)藍(lán)牙通訊模塊應(yīng)用程序的介紹還很少,同時(shí)開(kāi)發(fā)藍(lán)牙通訊技術(shù)的應(yīng)用需要十分廣泛,所以本文將就此進(jìn)行一些討論。
1 基于托管碼開(kāi)發(fā)藍(lán)牙通訊模塊
基于托管碼的開(kāi)發(fā)就是使用一套運(yùn)行時(shí)環(huán)境(run-time environment)的應(yīng)用程序接口來(lái)開(kāi)發(fā)。
一般情況下,托管碼應(yīng)用程序的開(kāi)發(fā)會(huì)比較簡(jiǎn)單和快速,并且可跨軟件平臺(tái)和處理器來(lái)運(yùn)行,所以開(kāi)發(fā)出的托管碼也能重新使用并有較高的可移植性。
另外,內(nèi)存管理、資源管理、資源收集、安全性管理等瑣碎工作都由運(yùn)行時(shí)環(huán)境來(lái)處理。應(yīng)用程序開(kāi)發(fā)工程師不必費(fèi)心處理。托管應(yīng)用程序在目標(biāo)機(jī)器上運(yùn)行,是通過(guò)目標(biāo)機(jī)器端的實(shí)時(shí)編譯器來(lái)實(shí)時(shí)把托管碼編譯成目標(biāo)機(jī)器碼后在目標(biāo)機(jī)器上執(zhí)行。
由于在.NET平臺(tái)下,采用CLR(公共語(yǔ)言運(yùn)行時(shí))可以用不同的語(yǔ)言來(lái)調(diào)用.NET Compact Framework來(lái)開(kāi)發(fā)相同功能的應(yīng)用程序,所以本文托管碼部分僅采用C#語(yǔ)言為例來(lái)介紹藍(lán)牙通訊模塊開(kāi)發(fā)。
1.1 利用P/Invoke方法編寫(xiě)藍(lán)牙通訊模塊
藍(lán)牙通訊模塊是一個(gè)涉及到驅(qū)動(dòng)硬件的應(yīng)用程序開(kāi)發(fā),而.NET Compact Framework并不是一個(gè)對(duì)Win32API進(jìn)行了完整封裝的類(lèi)庫(kù)。所以在基于托管碼開(kāi)發(fā)藍(lán)牙通訊模塊中必須利用到托管代碼如何與非托管代碼交互技術(shù)。P/Invoke全稱(chēng)為Platform Invoke,是.NET開(kāi)發(fā)平臺(tái)下允許托管代碼調(diào)用DLL庫(kù)的本地代碼函數(shù)的服務(wù),類(lèi)似于JA-VA中的GNI的概念。圖1說(shuō)明了P/Invoke方法的工作原理。首先用相應(yīng)語(yǔ)言的編譯器將托管的源代碼編譯成Assembly的形式,其中包括元數(shù)據(jù)和中間語(yǔ)言代碼。而此時(shí)P/Invoke的聲明會(huì)以元數(shù)據(jù)的形式存在于 Assembly中,當(dāng)Assembly被CLR調(diào)用的時(shí)候,CLR會(huì)根據(jù)元數(shù)據(jù)的聲明在對(duì)應(yīng)的DLL函數(shù)中查找DLL的實(shí)現(xiàn)。如果找到,就將其加載到內(nèi)存中,并定位此DLL函數(shù)的人口點(diǎn)。將托管的參數(shù)人棧,并將函數(shù)的人口點(diǎn)指向?qū)?yīng)的native dll,從而完成了托管代碼調(diào)用非托管代碼的DLL。
利用P/Invoke方法編寫(xiě)藍(lán)牙通訊模塊,DllI-port屬性非常有用。下面的代碼將用例子說(shuō)明此通用方案,例中托管程序?qū)⒄{(diào)用MessageBox(位于User32.lib中):
using
using namespace System:: Runtime::InteropSer-vices;
namespace SysWin32
{
[DllImport ( "user32. dll", EntryPoint = "MessageBox", CharSet = Unicode)]
int MessageBox(void * hWnd, wchar_t * lpText,wchar_t * lpCaption, unsigned int uType);
?。?/p>
int main()
SysWin32 :: MessageBox(0, L" Hello world ! ", L"Greetings", 0)
?。?/p>
注意包含DllImport的代碼行。此代碼行根據(jù)參數(shù)值通知編譯器,使之聲明位于User32.dll中的函數(shù),并將簽名中出現(xiàn)的所有字符串(如參數(shù)或返回值)視為Unicode字符串。如果缺少EntryPoint參數(shù),則默認(rèn)值為函數(shù)名。另外,由于CharSet參數(shù)指定Unicode,因此公共語(yǔ)言運(yùn)行庫(kù)將首先查找稱(chēng)為MessageBoxW的函數(shù)。如果運(yùn)行庫(kù)未找到此函數(shù),它將根據(jù)調(diào)用約定查找MessageBox以及相應(yīng)的修飾名。
當(dāng)調(diào)用用戶定義的DLL中所包含的函數(shù)時(shí),有必要將extern"C"添加在DLL函數(shù)聲明之前,如下所示:
extern"C"SAMPLEDLL_API int fnSam-pleDLL(void);
在調(diào)用非本機(jī)碼時(shí),需要注意的是要將非結(jié)構(gòu)化參數(shù)由托管封送處理為本機(jī)碼形式??梢岳肅harSet參數(shù)值的作用,將參數(shù)中字符串(string*類(lèi)型)都自動(dòng)轉(zhuǎn)換為wchar_t*。同樣,所有Int32參數(shù)類(lèi)型轉(zhuǎn)換為非托管int,UInt32參數(shù)類(lèi)型轉(zhuǎn)換為非托管unsignedint,而 Intl6參數(shù)類(lèi)型轉(zhuǎn)換為了short int。char*用于[in]參數(shù)的為String*(CharSet=Ansi),用于[out]參數(shù)或返回值的為T(mén)ext::StringBuilder*。wchar-t*用于[in]參數(shù)為String*(CharSet=Unicode),用于 [out]參數(shù)或返回值的為T(mén)ext::StringBuilder*。需要注意的是函數(shù)指針必須具有_stdcall調(diào)用約定,這是因?yàn)檫@是 DllImport支持的唯一類(lèi)型。對(duì)于數(shù)組來(lái)說(shuō)數(shù)組(如wchar_t*[ ]),CharSet參數(shù)僅應(yīng)用于函數(shù)參數(shù)的根類(lèi)型。因此,無(wú)論 CharSet的值是什么,String*_ _gc[ ]將被封送處理為wchar_t*[]。除簡(jiǎn)單類(lèi)型外,運(yùn)行庫(kù)還提供了一種機(jī)制,可以將簡(jiǎn)單結(jié)構(gòu)由托管上下文封送處理為非托管上下文。簡(jiǎn)單結(jié)構(gòu)不包含任何內(nèi)部數(shù)據(jù)成員指針、結(jié)構(gòu)化類(lèi)型的成員或其他元素。
在做一個(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)?a target="_blank">藍(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)始編寫(xiě)藍(lán)牙通訊模塊了。
對(duì)應(yīng)的每一步需要調(diào)用的基本函數(shù)如下:
?獲取本地設(shè)備的ID
[DllImport ( "Btdrt. dll", SetLastError = true) ]
public static extern int BthReadLocalAddr (byte[]PBa)
?獲取遠(yuǎn)程設(shè)備的ID
[DllImport( "ws2. dll", EntryPoint = "WSALook-upServiceBegin", SetLastError= true)]
public static extern int CeLookupServiceBegin(byte[ ] pQuerySet, LookupFlags dwFlags, ref intlphLookup)
?監(jiān)聽(tīng)服務(wù)
[DllImport (" ws2. dll", EntryPoint = "WSASetSer-vice", SetLastError= true)]
public static extern int CeSetService
(byte[ ] pQuerySet, RNRSERVICE_REGISTER,LookupFlags dwFlags)
?連接
[DllImport ( "mscoree", EntryPoint = "@ 339" )]
public static extern int connect (int s, byte []name, int namelen)
?藍(lán)牙的安全設(shè)置
獲取配對(duì)碼請(qǐng)求
[DllImport("Btdrt. dll", SetLastError= true)]
public static extern int BthGetPINRequest(byte[]pba)
設(shè)置配對(duì)碼
[DllImport( "btdrt. dll", SetLastError= true)
public static extern int BthSetPIN(byte[] pba, intcPinLength, byte [] ppin)
創(chuàng)建ACL連接:
[DllImport("Btdrt. dll", SetLastError= true)
public static extern int BthCreateACLConnection (byte[] pbt, ref ushort phandle);
然后是配對(duì)碼驗(yàn)證:
[DllImport("Btdrt. dll", SetLastError= true)]
public static extern int BthAuthenticate (byte []pbt);
然后一定要關(guān)閉連接:
[DllImport("Btdrt. dll", SetLastError= true)]
public static extern int BthCloseConnection(ushorthandle);
評(píng)論