1、簡介
微軟設計COM規范的時候,有兩種選擇來保證用戶的設計的COM組件可以全球唯一:
①采用和Internet地址一樣的管理方式,成立一個管理機構,用戶如果想開發一個COM組件的時候需要向該機構提出申請,並交一定的費用。
②發明一種算法,每次都能產生一個全球唯一的COM組件標識符。
現在采用的是第二種方法,這種算法用GUID(全局唯一標識符)標識COM組件。GUID是一個128位長的數字,一般用16進制表示。
GUID例子: 54BF6567–1007–11D1–B0AA–444553540000 [一共32位16進制數,也是就是128位二進制]
算法的核心思想是結合機器的網卡、當地時間、一個隨即數來生成GUID。從理論上講,如果一台機器每秒產生10000000個GUID,則可以保證(概率意義上)3240年不重復。在微軟的 COM 中 GUID 和 UUID、CLSID、IID 是一回事,只不過各自代表的意義不同:
UUID : 代表COM組件
CLSID : 代表COM組件中的類
IID : 代表COM組件類中的導出接口
因為在程序中,實際對象數據對應的處理程序路徑string往往不盡相同,微軟不使用直接的路徑表示方法,而使用一個叫CLSID的方式間接描述這些對象數據的處理程序路徑。最終這個CLSID和COM組件路徑的對應關系是保存在系統注冊表中。
2、CLSID 的結構定義
1 typedef struct _GUID { 2 3 DWORD Data1; // 隨機數 4 WORD Data2; // 和時間相關 5 WORD Data3; // 和時間相關 6 BYTE Data4[8]; // 和網卡MAC相關 7 8 } GUID; 9 10 typedef GUID CLSID; // 組件ID 11 12 typedef GUID IID; // 接口ID 13 14 #define REFCLSID const CLSID &
// 常見的聲明和賦值方法
1 CLSID CLSID_Excel = {0x00024500,0x0000,0x0000,{0xC0,0x00,0x00,0x00,0x00,0x00,0x00,0x46}}; 2 3 struct __declspec(uuid("00024500-0000-0000-C000-000000000046")) CLSID_Excel; 4 5 class DECLSPEC_UUID("00024500-0000-0000-C000-000000000046") CLSID_Excel;
注冊表中的表示方法:
1 {00024500-0000-0000-C000-000000000046}
每一個COM組件都需要指定一個CLSID,並且不能重名。它之所以使用16個字節,就是要從概率上保證重復是“不可能”的。但是,微軟為了使用方便,也支持另一個字符串名稱方式,叫 ProgID。由於 CLSID 和 ProgID 其實是一個概念的兩個不同的表示形式,所以我們在程序中可以隨便使用任何一種。
CLSIDFromProgID()、CLSIDFromProgIDEx()由 ProgID 得到 CLSID。
通過ProgID得到CLSID,實際上就是查注冊表。
3、獲取GUID
__uuidof 關鍵字用來獲得表達式的GUID。這個表達式可以是一種類型名稱、一個指針、引用或者一個類型的數組、一個有這個類型實例化的模板或者這種類型的變量。
只要編譯器能使用該參數獲得相關的GUID,那么該參數就是有效的。這個參數的特殊情況就是它取0或NULL的時候。在這種情況下,__uuidof會返回一個由0組成的GUID。
3.1 示例
1 class __declspec(uuid("B372C9F6-1959-4650-960D-73F20CD479BA")) Class; 2 3 struct __declspec(uuid("B372C9F6-1959-4650-960D-73F20CD479BB")) Interface; 4 5 void Test() 6 7 { 8 CLSID clsid = __uuidof(Class); 9 IID iid = __uuidof(Interface); 10 ... 11 12 }
智能指針的實例化操作:
HRESULT hr = pHttpReq.CreateInstance(__uuidof(WinHttpRequest));//實例化智能指針
4、使用開發環境編寫組件程序
IDE會自動產生CLSID
可以用函數 CoCreateGuid() 產生 CLSID;
使用"vc目錄\Common\Tools\GuidGen.exe"工具產生GUID
5、CLSID 和 ProgID 之間的轉換方法和相關的函數
CLSIDFromProgID()、CLSIDFromProgIDEx() 由 ProgID 得到 CLSID。實際是查注冊表
ProgIDFromCLSID() 由 CLSID 得到 ProgID,調用者使用完成后要釋放 ProgID 的內存
CoCreateGuid() 隨機生成一個 GUID
IsEqualGUID()、IsEqualCLSID()、IsEqualIID() 比較2個ID是否相等
StringFromCLSID()、StringFromGUID2()、StringFromIID() 由 CLSID,IID 得到注冊表中CLSID樣式的字符串,注意釋放內存
REFCLSID
1 #define REFCLSID const IID & 2 typedef GUID IID; 3 typedef struct _GUID { 4 unsigned long Data1; 5 unsigned short Data2; 6 unsigned short Data3; 7 unsigned char Data4[ 8 ]; 8 } GUID;