C#調用Windows API函數
Api函數是構筑Windws應用程序的基石,每一種Windows應用程序開發(fā)工具,它提供的底層函數都間接或直接地調用了Windows API函數,同時為了實現功能擴展,一般也都提供了調用Windows API函數的接口, 也就是說具備調用動態(tài)連接庫的能力。Visual C#和其它開發(fā)工具一樣也能夠調用動態(tài)鏈接庫的Windows API函數。.NET框架本身提供了這樣一種服務,允許受管轄的代碼調用動態(tài)鏈接庫中實現的非受管轄函數,包括操作系統(tǒng)提供的Windows API函數。它能夠定位和調用輸出函數,根據需要,組織其各個參數(整型、字符串類型、數組、和結構等等)跨越互操作邊界。
下面以C#為例簡單介紹調用API的基本過程:
動態(tài)鏈接庫函數的聲明
動態(tài)鏈接庫函數使用前必須聲明,相對于VB,C#函數聲明顯得更加羅嗦,前者通過 Api Viewer粘貼以后,可以直接使用,而后者則需要對參數作些額外的變化工作。
動態(tài)鏈接庫函數聲明部分一般由下列兩部分組成,一是函數名或索引號,二是動態(tài)鏈接庫的文件名。
譬如,你想調用User32.DLL中的MessageBox函數,我們必須指明函數的名字MessageBoxA或MessageBoxW,以及庫名字User32.dll,我們知道Win32 API對每一個涉及字符串和字符的函數一般都存在兩個版本,單字節(jié)字符的ANSI版本和雙字節(jié)字符的UNICODE版本。
下面是一個調用API函數的例子:
- [DllImport("KERNEL32.DLL", EntryPoint="MoveFileW", SetLastError=true,
- CharSetCharSet=CharSet.Unicode, ExactSpelling=true,
- CallingConventionCallingConvention=CallingConvention.StdCall)]
- public static extern bool MoveFile(String src, String dst);
其中入口點EntryPoint標識函數在動態(tài)鏈接庫的入口位置,在一個受管轄的工程中,目標函數的原始名字和序號入口點不僅標識一個跨越互操作界限的函數。而且,你還可以把這個入口點映射為一個不同的名字,也就是對函數進行重命名。重命名可以給調用函數帶來種種便利,通過重命名,一方面我們不用為函數的大小寫傷透腦筋,同時它也可以保證與已有的命名規(guī)則保持一致,允許帶有不同參數類型的函數共存,更重要的是它簡化了對ANSI和Unicode版本的調用。CharSet用于標識函數調用所采用的是Unicode或是ANSI版本,ExactSpelling=false將告訴編譯器,讓編譯器決定使用Unicode或者是Ansi版本。其它的參數請參考MSDN在線幫助.
在C#中,你可以在EntryPoint域通過名字和序號聲明一個動態(tài)鏈接庫函數,如果在方法定義中使用的函數名與DLL入口點相同,你不需要在EntryPoint域顯示聲明函數。否則,你必須使用下列屬性格式指示一個名字和序號。
[DllImport("dllname", EntryPoint="Functionname")]
[DllImport("dllname", EntryPoint="#123")]
值得注意的是,你必須在數字序號前加“?!?BR>下面是一個用MsgBox替換MessageBox名字的例子:
[C#]
- using System.Runtime.InteropServices;
- public class Win32 {
- [DllImport("user32.dll", EntryPoint="MessageBox")]
- public static extern int MsgBox(int hWnd, String text, String caption, uint type);
- }
許多受管轄的動態(tài)鏈接庫函數期望你能夠傳遞一個復雜的參數類型給函數,譬如一個用戶定義的結構類型成員或者受管轄代碼定義的一個類成員,這時你必須提供額外的信息格式化這個類型,以保持參數原有的布局和對齊。以上介紹Windows API函數
【編輯推薦】