C # Call DLL

Source: Internet
Author: User
Tags button attributes

The methods for calling DLL in each programming language are different. Here we will only introduce the methods for calling DLL with C. First, you need to know what is managed and what is not managed. It is generally considered that the unmanaged code is mainly developed based on the DLL and activeX Components developed on the win 32 platform, and the managed code is developed based on the. net platform. If you want to learn more about the relationship and differences between hosting and non-hosting, and their operating mechanism, please search for information on your own. This document will not be discussed here.

(1) General method for calling unmanaged functions in DLL

First, the external method should be declared in the C # language source program. The basic form is:

[DLLImport ("DLL file")]

Modifier extern return variable type method name (parameter list)

Where:

DLL file: contains the library file that defines external methods.

Modifier: access modifier, which can be used in addition to abstract when declaring a method.

Return variable type: In the DLL file, you need to call the return variable type of the method.

Method Name: name of the method to be called in the DLL file.

Parameter List: list of methods to be called in the DLL file.

Note: The System. Runtime. InteropServices namespace must be used in the program declaration.

DllImport can only be placed on method declaration.

The DLL file must be located in the current program directory or the system-defined query Path (that is, the Path set by Path in the system environment variable ).

The returned variable type, method name, and parameter list must be consistent with the definition in the DLL file.

 

To use other function names, you can use the EntryPoint attribute settings, such:

[DllImport ("user32.dll", EntryPoint = "MessageBoxA")]

Static extern int MsgBox (int hWnd, string msg, string caption, int type );

Other optional DllImportAttribute attributes:

CharSet indicates the character set used in the entry point, for example, CharSet = CharSet. Ansi;

SetLastError indicates whether the method retains Win32 "previous error", for example, SetLastError = true;

ExactSpelling indicates whether EntryPoint must exactly match the spelling of the indicated entry point, for example, ExactSpelling = false;

PreserveSig indicates whether the method signature should be retained or converted, for example, PreserveSig = true;

CallingConvention indicates the call convention of the entry point, for example: CallingConvention = CallingConvention. Winapi;

 

For more information about "data sending and writing" and "sending numbers and logical scalar", see [2].

C # example:

1. Start VS. NET and create a new project named "Tzb". The template is "Windows application ".

2. Double-click the "Button" item in the "Windows Forms" item in the "toolbox" to add a Button to the "Form1" form.

3. Change the button attributes: Name is "B1", Text is "use DllImport To Call DLL to bring up a prompt box", and adjust the button B1 to the appropriate size and move it to the appropriate position.

4. double-click "Form1" in the Class View, open the "Form1.cs" Code view, and enter "using System" on "namespace Tzb. runtime. interopServices; "to import the namespace.

5. in the "Form1.cs [design]" view, double-click the button B1, use the keyword static and extern to declare the method "MsgBox" on the "B1_Click" method, and append the DllImport attribute to this method, here we will use the "MessageBoxA" function in "user32.dll". The specific code is as follows:
[DllImport ("user32.dll", EntryPoint = "MessageBoxA")]

Static extern int MsgBox (int hWnd, string msg, string caption, int type );

Then add the following code in the "B1_Click" method to call the method "MsgBox ":
MsgBox (0, "this is the prompt box for calling DLL with DllImport! "," Mug ", 0x30 );

6. Press F5 to run the program and click B1. The following prompt box is displayed:

 

(2) Dynamic Loading and calling of unmanaged functions in DLL

The preceding figure shows how to use DllImport to call the unmanaged function in DLL, but this is a global function. If the unmanaged function in DLL has a static variable S, each time you call this function, the static variable S automatically adds 1. Results: When a new count is required, the expected results cannot be obtained. The following is an example:

1. Create DLL

1) Start Visual C ++ 6.0;

2) create a "Win32 Dynamic-Link Library" project named "Count ";

3) on the "Dll kind" selection page, select "A simple dll project ";

4) Open Count. cpp and add the following code:
// Export the function, which is called using the _ stdcall standard.

Extern "C" _ declspec (dllexport) int _ stdcall count (int init );

Int _ stdcall count (int init)

{// Count function, uses the init parameter to initialize the static integer variable S, and returns this value after S Auto-increment 1.

Static int S = init;

S ++;

Return S;

}

5) Press "F7" to compile and get Count. dll (in the Debug folder under the project directory ).

 

2. Use DllImport to call the count function in DLL

1) Open the project "Tzb" and add a button to the "Form1" form.

2) Change the button attributes: Name is "B2", Text is "use DllImport to call the count function in DLL", and adjust the button B1 to the appropriate size and move it to the appropriate position.

3) Open the "Form1.cs" Code view, use the static and extern keywords to declare the method "count", and enable it to implement the export function Count from count. dll. The Code is as follows:

[DllImport ("Count. dll")]

Static extern int count (int init );

4) double-click the button B2 in the "Form1.cs [design]" view and add the following code in the "B2_Click" method:
MessageBox. show ("use DllImport to call the count function in DLL. The input value of n is 0, and the result is:" + count (0 ). toString (), "Challenge Cup ");

MessageBox. show ("use DllImport to call the count function in DLL. The input value of n is 10. The result is:" + count (10 ). toString () + "n is not the expected 11 !!! "," Challenge Cup ");

MessageBox. Show ("The result shows that nuse DllImport to call the unmanaged n function in DLL is a global and static function !!! "," Challenge Cup ");

5) copy Count. dll to the binDebug folder of the project "Tzb", Press "F5" to run the program, and click "B2" to bring up the following three prompt boxes:

The 1st prompt boxes show the results of calling "count (0)", and the 2nd prompt boxes show the results of calling "count (10, the results show that "using DllImport to call the unmanaged functions in DLL is a global and static function ". Therefore, sometimes it cannot achieve our goal, so we need to use the method described below: C # dynamic call of functions in DLL.

 

3. C # dynamically call functions in DLL

Because the use of DllImport in C # is not like dynamic load/unload assembly, so you can only use API functions. In kernel32.dll, functions related to dynamic library calling include [3]:

① LoadLibrary (or AfxLoadLibrary of MFC) to load dynamic libraries.

② GetProcAddress: Get the function to be introduced and convert the symbol name or ID number to the internal DLL address.

③ FreeLibrary (or AfxFreeLibrary of MFC) to release the dynamic link library.

Their prototypes are:

HMODULE LoadLibrary (LPCTSTR lpFileName );

FARPROC GetProcAddress (HMODULE hModule, LPCWSTR lpProcName );

BOOL FreeLibrary (HMODULE hModule );

 

Now, we can use IntPtr hModule = LoadLibrary ("Count. dll); to obtain the Dll handle, use IntPtr farProc = GetProcAddress (hModule, "_ count @ 4"); to obtain the function entry address.

But how can I call this function after knowing the function entry address? Because there is no function pointer in C # and there is no function pointer call method like C ++ to call the function, we have to use other methods. After research, we found that we can achieve our goal by combining the classes and functions in System. Reflection. Emit and System. Reflection. Assembly. For future convenience and code reuse, we can write a class.

1) write the dld class:

1. open the project "Tzb", open the Class View, right-click "Tzb", select "add" --> "class", and set the class name to "dld ", that is, the start letter of each word in dynamic loading dll.

2. Add the required namespace and declare the parameter transfer method enumeration:
Using System. Runtime. InteropServices; // This namespace is required for using DllImport

Using System. Reflection; // This namespace is required for using the Assembly class

Using System. Reflection. Emit; // This namespace is required to use ILGenerator

 

Add the following code to the public class dld to declare the parameter transfer method enumeration:
/// <Summary>

/// Parameter transfer mode enumeration, ByValue indicates value transfer, ByRef indicates Address Transfer

/// </Summary>

Public enum ModePass

{

ByValue = 0x0001,

ByRef = 0x0002

}

3. Declare LoadLibrary, GetProcAddress, FreeLibrary, and private variables hModule and farProc:
/// <Summary>

/// The prototype is HMODULE LoadLibrary (LPCTSTR lpFileName );

/// </Summary>

/// <Param name = "lpFileName"> DLL file name </param>

/// <Returns> handle of the function library module </returns>

[DllImport ("kernel32.dll")]

Static extern IntPtr LoadLibrary (string lpFileName );

/// <Summary>

/// The prototype is FARPROC GetProcAddress (HMODULE hModule, LPCWSTR lpProcName );

/// </Summary>

/// <Param name = "hModule"> handle of the function library module to be called </param>

/// <Param name = "lpProcName"> name of the called function </param>

/// <Returns> function pointer </returns>

[DllImport ("kernel32.dll")]

Static extern IntPtr GetProcAddress (IntPtr hModule, string lpProcName );

/// <Summary>

/// Prototype: BOOL FreeLibrary (HMODULE hModule );

/// </Summary>

/// <Param name = "hModule"> handle of the function library module to be released </param>

/// <Returns> whether the specified Dll has been released </returns>

[DllImport ("kernel32", EntryPoint = "FreeLibrary", SetLastError = true)]

Static extern bool FreeLibrary (IntPtr hModule );

/// <Summary>

/// Handle of the function library module returned by Loadlibrary

/// </Summary>

Private IntPtr hModule = IntPtr. Zero;

/// <Summary>

/// Function pointer returned by GetProcAddress

/// </Summary>

Private IntPtr farProc = IntPtr. Zero;

4. added the LoadDll method and reloaded this method for convenience during the call:

/// <Summary>

/// Load the Dll

/// </Summary>

/// <Param name = "lpFileName"> DLL file name </param>

Public void LoadDll (string lpFileName)

{

HModule = LoadLibrary (lpFileName );

If (hModule = IntPtr. Zero)

Throw (new Exception ("not found:" + lpFileName + "."));

}

If you already have a Dll handle, you can use the second version of the LoadDll method:
Public void LoadDll (IntPtr HMODULE)

{

If (HMODULE = IntPtr. Zero)

Throw (new Exception ("the handle hmodule of the passed function library module is empty ."));

HModule = HMODULE;

}

5. Add the LoadFun method and reload this method for the convenience of calling. The code and comments of the method are as follows:
/// <Summary>

/// Obtain the function pointer

/// </Summary>

/// <Param name = "lpProcName"> name of the called function </param>

Public void LoadFun (string lpProcName)

{// If the handle of the function library module is null, an exception is thrown.

If (hModule = IntPtr. Zero)

Throw (new Exception ("the handle of the function library module is empty. Make sure that the LoadDll operation has been performed! "));

// Obtain the function pointer

FarProc = GetProcAddress (hModule, lpProcName );

// If the function pointer is used, an exception is thrown.

If (farProc = IntPtr. Zero)

Throw (new Exception ("not found:" + lpProcName + "entry point of this function "));

}

/// <Summary>

/// Obtain the function pointer

/// </Summary>

/// <Param name = "lpFileName"> contains the DLL file name for the function to be called. </param>

/// <Param name = "lpProcName"> name of the called function </param>

Public void LoadFun (string lpFileName, string lpProcName)

{// Obtain the handle of the function library module

HModule = LoadLibrary (lpFileName );

// If the handle of the function library module is empty, an exception is thrown.

If (hModule = IntPtr. Zero)

Throw (new Exception ("not found:" + lpFileName + "."));

// Obtain the function pointer

FarProc = GetProcAddress (hModule, lpProcName );

// If the function pointer is used, an exception is thrown.

If (farProc = IntPtr. Zero)

Throw (new Exception ("not found:" + lpProcName + "entry point of this function "));

}

6. added the UnLoadDll and Invoke methods, and the Invoke method was also overloaded:
/// <Summary>

/// Uninstall the Dll

/// </Summary>

Public void UnLoadDll ()

{

FreeLibrary (hModule );

HModule = IntPtr. Zero;

FarProc = IntPtr. Zero;

}

Contact Us

The content source of this page is from Internet, which doesn't represent Alibaba Cloud's opinion; products and services mentioned on that page don't have any relationship with Alibaba Cloud. If the content of the page makes you feel confusing, please write us an email, we will handle the problem within 5 days after receiving your email.

If you find any instances of plagiarism from the community, please send an email to: info-contact@alibabacloud.com and provide relevant evidence. A staff member will contact you within 5 working days.

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.