中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

C#中怎么調用DLL函數

發布時間:2021-06-21 19:03:12 來源:億速云 閱讀:994 作者:Leah 欄目:編程語言

今天就跟大家聊聊有關C#中怎么調用DLL函數,可能很多人都不太了解,為了讓大家更加了解,小編給大家總結了以下內容,希望大家根據這篇文章可以有所收獲。

因為C#中使用DllImport是不能像動態load/unload assembly那樣,所以只能借助API函數了。在kernel32.dll中,與動態庫調用有關的函數包括[3]:

①LoadLibrary(或MFC 的AfxLoadLibrary),裝載動態庫。

②GetProcAddress,獲取要引入的函數,將符號名或標識號轉換為DLL內部地址。

③FreeLibrary(或MFC的AfxFreeLibrary),釋放動態鏈接庫。

它們的原型分別是:

HMODULE LoadLibrary(LPCTSTR lpFileName);

FARPROC GetProcAddress(HMODULE hModule, LPCWSTR lpProcName);

BOOL FreeLibrary(HMODULE hModule);

現在,我們可以用IntPtr hModule=LoadLibrary(“Count.dll”);來獲得Dll的句柄,用IntPtr farProc=GetProcAddress(hModule,”_count@4”);來獲得函數的入口地址。

但是,知道函數的入口地址后,怎樣調用這個函數呢?因為在C#中是沒有函數指針的,沒有像C++那樣的函數指針調用方式來調用函數,所以我們得借助其它方法。經過研究,發現我們可以通過結合使用System.Reflection.Emit及System.Reflection.Assembly里的類和函數達到我們的目的。為了以后使用方便及實現代碼的復用,我們可以編寫一個類。

1) dld類的編寫:

1.打開項目“Test”,打開類視圖,右擊“Tzb”,選擇“添加”-->“類”,類名設置為“dld”,即dynamic loading dll 的每個單詞的開頭字母。

2.添加所需的命名空間及聲明參數傳遞方式枚舉:

using System.Runtime.InteropServices; // 用DllImport 需用此命名空間   using System.Reflection; // 使用Assembly 類需用此命名空間   using System.Reflection.Emit; // 使用ILGenerator 需用此命名空間

3. 在namespace test中,“public class dld”的上面,添加如下代碼聲明參數傳遞方式枚舉:

/// < summary>        /// 參數傳遞方式枚舉,ByValue 表示值傳遞,ByRef 表示址傳遞       /// < /summary>        public enum ModePass       {           ByValue = 0x0001,           ByRef = 0x0002   }

4、在public class DLD中,添加如下代碼:

public class DLD       {           [DllImport("kernel32.dll")]           public static extern IntPtr LoadLibrary(string lpFileName);               [DllImport("kernel32.dll")]           public static extern IntPtr GetProcAddress(IntPtr hModule, string lpProceName);               [DllImport("kernel32", EntryPoint = "FreeLibrary", SetLastError = true)]           public static extern bool FreeLibrary(IntPtr hModule);               /// < summary>                /// Loadlibrary 返回的函數庫模塊的句柄                /// < /summary>                private IntPtr hModule = IntPtr.Zero;               /// < summary>                /// GetProcAddress 返回的函數指針                /// < /summary>                private IntPtr farProc = IntPtr.Zero;               /// < summary>                /// 裝載 Dll                /// < /summary>                /// < param name="lpFileName">DLL 文件名 < /param>                public void LoadDll(string lpFileName)           {                   hModule = LoadLibrary(lpFileName);                   if (hModule == IntPtr.Zero)                       throw (new Exception(" 沒有找到 :" + lpFileName + "."));           }                   /// < summary>                /// 獲得函數指針                /// < /summary>                /// < param name="lpProcName"> 調用函數的名稱 < /param>                public void LoadFun(string lpProcName)           { // 若函數庫模塊的句柄為空,則拋出異常                    if (hModule == IntPtr.Zero)                       throw (new Exception(" 函數庫模塊的句柄為空 , 請確保已進行 LoadDll 操作 !"));                   // 取得函數指針                    farProc = GetProcAddress(hModule, lpProcName);                   // 若函數指針,則拋出異常                    if (farProc == IntPtr.Zero)                       throw (new Exception(" 沒有找到 :" + lpProcName + " 這個函數的入口點 "));               }               /// < summary>                /// 卸載 Dll                /// < /summary>                public void UnLoadDll()           {                   FreeLibrary(hModule);                   hModule = IntPtr.Zero;                   farProc = IntPtr.Zero;               }                   /// < summary>                /// 調用所設定的函數                /// < /summary>                /// < param name="ObjArray_Parameter"> 實參 < /param>                /// < param name="TypeArray_ParameterType"> 實參類型 < /param>                /// < param name="ModePassArray_Parameter"> 實參傳送方式 < /param>                /// < param name="Type_Return"> 返回類型 < /param>                /// < returns> 返回所調用函數的 object< /returns>                public object Invoke(object[] ObjArray_Parameter, Type[] TypeArray_ParameterType, ModePass[] ModePassArray_Parameter, Type Type_Return)           {                   // 下面 3 個 if 是進行安全檢查 , 若不能通過 , 則拋出異常                    if (hModule == IntPtr.Zero)                       throw (new Exception(" 函數庫模塊的句柄為空 , 請確保已進行 LoadDll 操作 !"));                   if (farProc == IntPtr.Zero)                       throw (new Exception(" 函數指針為空 , 請確保已進行 LoadFun 操作 !"));                   if (ObjArray_Parameter.Length != ModePassArray_Parameter.Length)                       throw (new Exception(" 參數個數及其傳遞方式的個數不匹配 ."));                   // 下面是創建 MyAssemblyName 對象并設置其 Name 屬性                    AssemblyName MyAssemblyName = new AssemblyName();                   MyAssemblyName.Name = "InvokeFun";                   // 生成單模塊配件                    AssemblyBuilder MyAssemblyBuilder = AppDomain.CurrentDomain.DefineDynamicAssembly(MyAssemblyName, AssemblyBuilderAccess.Run);                   ModuleBuilder MyModuleBuilder = MyAssemblyBuilder.DefineDynamicModule("InvokeDll");                   // 定義要調用的方法 , 方法名為“ MyFun ”,返回類型是“ Type_Return ”參數類型是“ TypeArray_ParameterType ”                    MethodBuilder MyMethodBuilder = MyModuleBuilder.DefineGlobalMethod("MyFun", MethodAttributes.Public | MethodAttributes.Static, Type_Return, TypeArray_ParameterType);                   // 獲取一個 ILGenerator ,用于發送所需的 IL                    ILGenerator IL = MyMethodBuilder.GetILGenerator();                   int i;                   for (i = 0; i <  ObjArray_Parameter.Length; i++)               {// 用循環將參數依次壓入堆棧                        switch (ModePassArray_Parameter[i])                   {                           case ModePass.ByValue:                               IL.Emit(OpCodes.Ldarg, i);                               break;                           case ModePass.ByRef:                               IL.Emit(OpCodes.Ldarga, i);                               break;                           default:                               throw (new Exception(" 第 " + (i + 1).ToString() + " 個參數沒有給定正確的傳遞方式 ."));                       }                   }                   if (IntPtr.Size == 4)               {// 判斷處理器類型                        IL.Emit(OpCodes.Ldc_I4, farProc.ToInt32());                   }                   else if (IntPtr.Size == 8)               {                       IL.Emit(OpCodes.Ldc_I8, farProc.ToInt64());                   }                   else              {                       throw new PlatformNotSupportedException();                   }                   IL.EmitCalli(OpCodes.Calli, CallingConvention.StdCall, Type_Return, TypeArray_ParameterType);                   IL.Emit(OpCodes.Ret); // 返回值                    MyModuleBuilder.CreateGlobalFunctions();                   // 取得方法信息                    MethodInfo MyMethodInfo = MyModuleBuilder.GetMethod("MyFun");                   return MyMethodInfo.Invoke(null, ObjArray_Parameter);// 調用方法,并返回其值                }       }

2) dld類的使用:

1.打開項目“Test”,向“Form1”窗體中添加一個按鈕,和一個TestBox,Name改為txRet。視圖中雙擊按鈕,在“button1_Click”方法體上面添加代碼,創建一個dld類實例,并進行測試。具體如下:

private void button1_Click(object sender, EventArgs e)           {                   int ret = 0;               dld myDLD = new dld();                   myDLD.LoadDll("xxx.dll");                   myDLD.LoadFun("InitSDK");                   object[] Parameters = new object[] { }; // 實參為0                    Type[] ParameterTypes = new Type[] { }; // 實參類型為int                    ModePass[] themode = new ModePass[] { }; // 傳送方式為值傳                   Type Type_Return = typeof(int); // 返回類型為int                   ret = (int)myDLD.Invoke(Parameters, ParameterTypes, themode, Type_Return);               txRet.Text = ret.ToString();               if (ret != 1)               {                   MessageBox.Show("InitSDK failed !");               }               if (ret == 1)               {                   MessageBox.Show("InitSDK Sucessed !");               }           }

其中,xxx為要測試的dll名稱,InitSDK為dll中的要測試的函數。

看完上述內容,你們對C#中怎么調用DLL函數有進一步的了解嗎?如果還想了解更多知識或者相關內容,請關注億速云行業資訊頻道,感謝大家的支持。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

麻江县| 浑源县| 红桥区| 襄樊市| 梅州市| 花垣县| 怀柔区| 都江堰市| 禹城市| 樟树市| 壶关县| 普格县| 黄浦区| 黑水县| 泗洪县| 松桃| 德清县| 澜沧| 鹤岗市| 鞍山市| 宣汉县| 虞城县| 常州市| 肥东县| 丽江市| 民勤县| 永顺县| 洛宁县| 枣强县| 斗六市| 临沧市| 颍上县| 吴桥县| 攀枝花市| 宁陵县| 湘阴县| 德清县| 遂川县| 阿瓦提县| 伊宁市| 宁陕县|