C# 互操作性入門系列(一):C#中互操作性介紹


C#互操作系列文章:

  1. C# 互操作性入門系列(一):C#中互操作性介紹
  2. C# 互操作性入門系列(二):使用平台調用調用Win32 函數
  3. C# 互操作性入門系列(三):平台調用中的數據封送處理
  4. C# 互操作性入門系列(四):在C#中調用COM組件

 

本專題概要:

  • 引言
  • 平台調用
  • C++ Interop(互操作)
  • COM Interop(互操作)

 

一、引言

   這個系列是在C#基礎知識中遺留下來的一個系列的,因為在C# 4.0中的一個新特性就是對COM互操作改進,然而COM互操作性卻是.NET平台下其中一種互操作技術,為了幫助大家更好的了解.NET平台下的互操作技術,所以才有了這個系列。然而有些朋友們可能會有這樣的疑問——“為什么我們需要掌握互操作技術的呢?” 對於這個問題的解釋就是——掌握了.NET平台下的互操作性技術可以幫助我們在.NET中調用非托管的dll和COM組件。.NET是建立在操作系統的之上的一個開發框架,其中.NET 類庫中的類也是對Windows API的抽象封裝,然而.NET類庫不可能對所有Windows API進行封裝,當.NET中沒有實現某個功能的類,然而該功能在Windows API被實現了,此時我們完全沒必要去自己在.NET中自定義個類,這時候就可以調用Windows  API 中的函數來實現,此時就涉及到托管代碼與非托管代碼的交互,此時就需要使用到互操作性的技術來實現托管代碼和非托管代碼更好的交互。.NET 平台下提供了3種互操作性的技術:

  1. Platform Invoke(P/Invoke),即平台調用,主要用於調用C庫函數和Windows API
  2. C++ Introp, 主要用於Managed C++(托管C++)中調用C++類庫
  3. COM Interop, 主要用於在.NET中調用COM組件和在COM中使用.NET程序集。

下面就對這3種技術分別介紹下。                                                            

二、平台調用

  使用平台調用的技術可以在托管代碼中調用動態鏈接庫(Dll)中實現的非托管函數,如Win32 Dll和C/C++ 創建的dll。看到這里,有些朋友們應該會有疑問——在怎樣的場合我們可以使用平台調用技術來調用動態鏈接庫中的非托管函數呢?

  這個問題就如前面引言中說講到的一樣,當在開發過程中,.NET類庫中沒有提供相關API然而Win32 API 中提供了相關的函數實現時,此時就可以考慮使用平台調用的技術在.NET開發的應用程序中調用Win32 API中的函數;

  然而還有一個使用場景就是——由於托管代碼的效率不如非托管代碼,為了提高效率,此時也可以考慮托管代碼中調用C庫函數。

2.1 在托管代碼中通過平台調用來調用非托管代碼的步驟

(1).  獲得非托管函數的信息,即dll的名稱,需要調用的非托管函數名等信息

(2). 在托管代碼中對非托管函數進行聲明,並且附加平台調用所需要屬性

(3). 在托管代碼中直接調用第二步中聲明的托管函數

2.2 平台調用的調用過程

(1)  查找包含該函數的DLL,當需要調用某個函數時,當然第一步就需要知道包含該函數的DLL的位置,所以平台調用的第一步也就是查找DLL,其實在托管代碼中調用非托管代碼的調用過程可以想象成叫某個人做事情,首先我們要找到那個人在哪里(即查找函數的DLL過程),找到那個人之后需要把要做的事情告訴他(相當於加載DLL到內存中和傳入參數),最后讓他去完成需要完成的事情(相當於讓非托管函數去執行任務)。

(2) 將找到的DLL加載到內存中。

(3) 查找函數在內存中的地址並把其參數推入堆棧,來封送所需的數據。CLR只會在第一次調用函數時,才會去查找和加載DLL,並查找函數在內存中的地址。當函數被調用過一次之后,CLR會將函數的地址緩存起來,CLR這種機制可以提高平台調用的效率。在應用程序域被卸載之前,找到的DLL都一直存在於內存中。

(4) 執行非托管函數。

平台調用的過程可以通過下圖更好地理解:

三、C++ Interop

  第二部分主要向大家介紹了第一種互操作性技術,然后我們也可以使用C++ Interop技術來實現與非托管代碼進行交互。然而C++ Interop 方式有一個與平台調用不一樣的地方,就是C++ Interop 允許托管代碼和非托管代碼存在於一個程序集中,甚至同一個文件中。C++ Interop 是在源代碼上直接鏈接和編譯非托管代碼來實現與非托管代碼進行互操作的,而平台調用是加載編譯后生成的非托管DLL並查找函數的入口地址來實現與非托管函數進行互操作的。C++ Interop使用托管C++來包裝非托管C++代碼,然后編譯生成程序集,然后再托管代碼中引用該程序集,從而來實現與非托管代碼的互操作。 關於具體的使用和與平台調用的比較,這里就不多介紹,我將會在后面的專題中具體介紹。

四、COM Interop

  COM(Component Object Model,組件對象模型)是微軟之前推薦的一個開發技術,由於微軟過去十多年里面開發了大量的COM組件,然而不可能在使用.NET技術重寫這些COM組件實現的功能,所以為了解決在.NET中的托管代碼能夠調用COM組件的問題,.NET 平台下提供了COM Interop,即COM互操作技術,COM Interop不僅支持在托管代碼中使用COM組件,而且還支持想CMO組件功能托管對象。下面就這兩種支持分別做一個介紹。

4.1 在.NET中使用COM組件

 在.NET中使用COM對象,主要有3種方法:

      1. 使用TlbImp工具為COM組件創建一個互操作程序集來綁定早期的COM對象,這樣就可以在程序中添加互操作程序集來調用COM對象
      2. 通過反射來后期綁定COM對象
      3. 通過P/Invoke創建COM對象或使用C++ Interop為COM對象編寫包裝類

但是我們經常使用的都是方法一,下面介紹下使用方法一在.NET 中使用COM對象的步驟:

  1. 找到要使用的COM 組件並注冊它。使用 regsvr32.exe 注冊或注銷 COM DLL。                               
  2. 在項目中添加對 COM 組件或類型庫的引用。                             

添加引用時,Visual Studio 會用到Tlbimp.exe(類型庫導入程序),Tlbimp.exe程序將生成一個 .NET Framework 互操作程序集。該程序集又稱為運行時可調用包裝 (RCW),其中包含了包裝COM組件中的類和接口。Visual Studio 將生成組件的引用添加至項目。                              

      3. 創建RCW中類的實例,這樣就可以使用托管對象一樣來使用COM對象。

下面通過一個圖更好地說明在.NET中使用COM組件的過程:

                                  

4.2 在COM中使用.NET程序集

   .NET 公共語言運行時通過COM可調用包裝(COM Callable Wrapper,即CCW)來完成與COM類型庫的交互。CCW可以使COM客戶端認為是在與普通的COM類型交互,同時使.NET組件認為它正在與托管應用程序交互。在這里CCW是非托管COM客戶端與托管對象之間的一個代理。 CCW既可以維護托管對象的生命周期,也負責數據類型在COM和.NET之間的相互轉換。實現在COM使用.NET 類型的基本步驟如:

1. 在C#項目中添加互操作特性

     可以修改C#項目屬性使程序集對COM可見。右鍵解決方案選擇屬性,在“應用程序標簽”中選擇“程序集信息”按鈕,在彈出的對話框中選擇 “使程序集COM可見” 選項,如下圖所示:

2. 生成COM類型庫並對它進行注冊以供COM客戶端使用

 在“生成”標簽中,選中 “為COM互操作注冊”選項,如下圖:

  勾選“為COM互操作注冊”選項后,Visual Studio會調用類型庫導出工具(Tlbexp.exe)為.NET程序集生成COM類型庫再使用程序集注冊工具(Regasm.exe)來完成對.NET程序集和生成的COM類型庫進行注冊,這樣COM客戶端可以使用CCW服務來對.NET對象進行調用了

五、總結

  介紹到這里,本專題的內容就結束,本專題主要對.NET 提供的互操作的技術做了一個總的概括,在后面的專題中將會對具體的技術進行詳細的介紹和給出一些簡單的使用例子。

 

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM