在Windows CE.NET的開發(fā)上,目前采用微軟公司2005年年底推出的Visual Studio 2005是最理想的選擇。用 Visual Studio 2005可以進行利用C#,或VB.NET等語言開發(fā)基于.NET Compact Framework 2.0或1.0的 托管碼WinCE程序,也可以利用C++語言來開發(fā)基于MFC,ATL或Win32API的本機碼WinCE程序。
因為在微軟最新的.NET Compact Framework 2.0的類庫中還未包含針對藍牙通訊模塊的類庫,而且目前關(guān)于在Windows CE中開發(fā)藍牙通訊模塊應用程序的介紹還很少,同時開發(fā)藍牙通訊技術(shù)的應用需要十分廣泛,所以本文將就此進行一些討論。
1 基于托管碼開發(fā)藍牙通訊模塊
基于托管碼的開發(fā)就是使用一套運行時環(huán)境(run-time environment)的應用程序接口來開發(fā)。
一般情況下,托管碼應用程序的開發(fā)會比較簡單和快速,并且可跨軟件平臺和處理器來運行,所以開發(fā)出的托管碼也能重新使用并有較高的可移植性。
另外,內(nèi)存管理、資源管理、資源收集、安全性管理等瑣碎工作都由運行時環(huán)境來處理。應用程序開發(fā)工程師不必費心處理。托管應用程序在目標機器上運行,是通過目標機器端的實時編譯器來實時把托管碼編譯成目標機器碼后在目標機器上執(zhí)行。
由于在.NET平臺下,采用CLR(公共語言運行時)可以用不同的語言來調(diào)用.NET Compact Framework來開發(fā)相同功能的應用程序,所以本文托管碼部分僅采用C#語言為例來介紹藍牙通訊模塊開發(fā)。
1.1 利用P/Invoke方法編寫藍牙通訊模塊
藍牙通訊模塊是一個涉及到驅(qū)動硬件的應用程序開發(fā),而.NET Compact Framework并不是一個對Win32API進行了完整封裝的類庫。所以在基于托管碼開發(fā)藍牙通訊模塊中必須利用到托管代碼如何與非托管代碼交互技術(shù)。P/Invoke全稱為Platform Invoke,是.NET開發(fā)平臺下允許托管代碼調(diào)用DLL庫的本地代碼函數(shù)的服務,類似于JA-VA中的GNI的概念。圖1說明了P/Invoke方法的工作原理。首先用相應語言的編譯器將托管的源代碼編譯成Assembly的形式,其中包括元數(shù)據(jù)和中間語言代碼。而此時P/Invoke的聲明會以元數(shù)據(jù)的形式存在于 Assembly中,當Assembly被CLR調(diào)用的時候,CLR會根據(jù)元數(shù)據(jù)的聲明在對應的DLL函數(shù)中查找DLL的實現(xiàn)。如果找到,就將其加載到內(nèi)存中,并定位此DLL函數(shù)的人口點。將托管的參數(shù)人棧,并將函數(shù)的人口點指向?qū)膎ative dll,從而完成了托管代碼調(diào)用非托管代碼的DLL。
利用P/Invoke方法編寫藍牙通訊模塊,DllI-port屬性非常有用。下面的代碼將用例子說明此通用方案,例中托管程序?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);
}
int main()
SysWin32 :: MessageBox(0, L" Hello world ! ", L"Greetings", 0)
}
注意包含DllImport的代碼行。此代碼行根據(jù)參數(shù)值通知編譯器,使之聲明位于User32.dll中的函數(shù),并將簽名中出現(xiàn)的所有字符串(如參數(shù)或返回值)視為Unicode字符串。如果缺少EntryPoint參數(shù),則默認值為函數(shù)名。另外,由于CharSet參數(shù)指定Unicode,因此公共語言運行庫將首先查找稱為MessageBoxW的函數(shù)。如果運行庫未找到此函數(shù),它將根據(jù)調(diào)用約定查找MessageBox以及相應的修飾名。
當調(diào)用用戶定義的DLL中所包含的函數(shù)時,有必要將extern"C"添加在DLL函數(shù)聲明之前,如下所示:
extern"C"SAMPLEDLL_API int fnSam-pleDLL(void);
在調(diào)用非本機碼時,需要注意的是要將非結(jié)構(gòu)化參數(shù)由托管封送處理為本機碼形式。可以利用CharSet參數(shù)值的作用,將參數(shù)中字符串(string*類型)都自動轉(zhuǎn)換為wchar_t*。同樣,所有Int32參數(shù)類型轉(zhuǎn)換為非托管int,UInt32參數(shù)類型轉(zhuǎn)換為非托管unsignedint,而 Intl6參數(shù)類型轉(zhuǎn)換為了short int。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)用約定,這是因為這是 DllImport支持的唯一類型。對于數(shù)組來說數(shù)組(如wchar_t*[ ]),CharSet參數(shù)僅應用于函數(shù)參數(shù)的根類型。因此,無論 CharSet的值是什么,String*_ _gc[ ]將被封送處理為wchar_t*[]。除簡單類型外,運行庫還提供了一種機制,可以將簡單結(jié)構(gòu)由托管上下文封送處理為非托管上下文。簡單結(jié)構(gòu)不包含任何內(nèi)部數(shù)據(jù)成員指針、結(jié)構(gòu)化類型的成員或其他元素。
在做一個關(guān)于藍牙通訊程序前,還需要一些關(guān)于藍牙的基礎(chǔ)知識。一個藍牙模塊程序需要包含開啟藍牙,配對,連接,建立串行通道,然后開啟通訊過程,還需要在應用程序中設置串行端口。因為藍牙技術(shù)有安全方面的設置,所以需要對藍牙設備進行配對。藍牙的工作狀態(tài)總共有3種,分別為開啟、關(guān)閉、可發(fā)現(xiàn)。并且所有的通訊設備都必須有一個對應的DeviceID,藍牙也不例外,藍牙的DeviceID是一串以“:”分隔的16進制的數(shù)字。有了上述知識,就可以在托管碼中利用P/Invoke方法開始編寫藍牙通訊模塊了。
對應的每一步需要調(diào)用的基本函數(shù)如下:
?獲取本地設備的ID
[DllImport ( "Btdrt. dll", SetLastError = true) ]
public static extern int BthReadLocalAddr (byte[]PBa)
?獲取遠程設備的ID
[DllImport( "ws2. dll", EntryPoint = "WSALook-upServiceBegin", SetLastError= true)]
public static extern int CeLookupServiceBegin(byte[ ] pQuerySet, LookupFlags dwFlags, ref intlphLookup)
?監(jiān)聽服務
[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)
?藍牙的安全設置
獲取配對碼請求
[DllImport("Btdrt. dll", SetLastError= true)]
public static extern int BthGetPINRequest(byte[]pba)
設置配對碼
[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);
然后是配對碼驗證:
[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);
評論
查看更多