您好,登錄后才能下訂單哦!
C#調(diào)用非托管代碼的方式主要有Com調(diào)用、DllImport方式調(diào)用、加載非托管動(dòng)態(tài)鏈接庫、直接執(zhí)行機(jī)器碼等方式。
現(xiàn)在介紹一下我自己常用的DllImport方式調(diào)用MSDN中提到的GetShortPathName方法;找到GetShortPathName的方法簽名,
DWORD GetShortPathName(LPCTSTR tpszLongPath,TPTSTR lpszShortPath,DWORD cchBuffer);
非托管及托管數(shù)據(jù)類型對(duì)應(yīng)關(guān)系:
LPCTSTR String
LPTSTR StringBuilder
DWORD int
DllImport的導(dǎo)入規(guī)則:
1、方法名與Win API完全一樣。如果在C#中調(diào)用時(shí)顯示完全不同的方法名稱,則需要引入EntryPoint屬性,使用別名顯示。
2、函數(shù)除需要DllImport類修飾符外,還需要聲明public static extern類型。
3、函數(shù)返回值和參數(shù)必須和調(diào)用的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);
}
代碼調(diào)用中kernel32.dll的路徑之所以沒寫是因?yàn)镈llImport會(huì)按照以下三種順序查找Dll:
1、exe所在目錄;2、System32目錄;3、環(huán)境變量目錄。
MarshalAs為可選類型,因?yàn)槊總€(gè)數(shù)據(jù)類型都有默認(rèn)的封送行為,該屬性指示如何在托管代碼和非托管代碼之間的封送數(shù)據(jù),可將該屬性用于參數(shù)、字段和返回值。大多數(shù)情況下該屬性只是用UnmanagedType枚舉類型就能滿足大多數(shù)非托管的數(shù)據(jù)類型,如默認(rèn)情況下字符會(huì)被當(dāng)作BStr傳入到Dll中,可以使用MarshalAs將字符串指定為LPTStr、LPWStr或LPStr等。
DllImport可選屬性解釋
EntryPoint 可對(duì)方法采用不同的名稱,使用別名
CharSet 函數(shù)調(diào)用使用Unicode還是Ansi
ExactSpelling False,表示讓編譯器自己選擇使用Unicode或Ansi
CallingConvetnion 它的參數(shù)指示入口點(diǎn)調(diào)用的約定;不指定默認(rèn)為CallingConvention.WinAPI
PreserveSig 指示方法簽名應(yīng)當(dāng)被保留還是被轉(zhuǎn)換,當(dāng)被轉(zhuǎn)換時(shí)它被轉(zhuǎn)換為一個(gè)具有HRESULT返回值和該返回值的一個(gè)名為retval的附加輸出參數(shù)的簽名,默認(rèn)為true。
SetLastError 指定是否保留上一次錯(cuò)誤,默認(rèn)為false
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請(qǐng)聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。