您好,登錄后才能下訂單哦!
C#調用非托管代碼的方式主要有Com調用、DllImport方式調用、加載非托管動態鏈接庫、直接執行機器碼等方式。
現在介紹一下我自己常用的DllImport方式調用MSDN中提到的GetShortPathName方法;找到GetShortPathName的方法簽名,
DWORD GetShortPathName(LPCTSTR tpszLongPath,TPTSTR lpszShortPath,DWORD cchBuffer);
非托管及托管數據類型對應關系:
LPCTSTR String
LPTSTR StringBuilder
DWORD int
DllImport的導入規則:
1、方法名與Win API完全一樣。如果在C#中調用時顯示完全不同的方法名稱,則需要引入EntryPoint屬性,使用別名顯示。
2、函數除需要DllImport類修飾符外,還需要聲明public static extern類型。
3、函數返回值和參數必須和調用的API的完全一樣。
4、必須引入System.Runtime.InteropServices命名空間。
代碼:
using System.Runtime.InteropServices;
public class Test
{
[DllImport("kernel32.dll",CharSet=CharSet.Auto,EntryPoint="GetShort")]
public static extern int GetShortPathName(
[MarshalAs(UnmanagedType.LPTStr)] String path,
[MarshalAs(UnmanagedType.LPTStr)] StringBuilder shortPath,
int shortPathLength);
}
代碼調用中kernel32.dll的路徑之所以沒寫是因為DllImport會按照以下三種順序查找Dll:
1、exe所在目錄;2、System32目錄;3、環境變量目錄。
MarshalAs為可選類型,因為每個數據類型都有默認的封送行為,該屬性指示如何在托管代碼和非托管代碼之間的封送數據,可將該屬性用于參數、字段和返回值。大多數情況下該屬性只是用UnmanagedType枚舉類型就能滿足大多數非托管的數據類型,如默認情況下字符會被當作BStr傳入到Dll中,可以使用MarshalAs將字符串指定為LPTStr、LPWStr或LPStr等。
DllImport可選屬性解釋
EntryPoint 可對方法采用不同的名稱,使用別名
CharSet 函數調用使用Unicode還是Ansi
ExactSpelling False,表示讓編譯器自己選擇使用Unicode或Ansi
CallingConvetnion 它的參數指示入口點調用的約定;不指定默認為CallingConvention.WinAPI
PreserveSig 指示方法簽名應當被保留還是被轉換,當被轉換時它被轉換為一個具有HRESULT返回值和該返回值的一個名為retval的附加輸出參數的簽名,默認為true。
SetLastError 指定是否保留上一次錯誤,默認為false
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。