C#調用C++的非托管類的dll其實很簡單基本就是固定的調用格式.
dll的編寫,首先是打開VS新建一個C++的控制台程序,下一步后選擇dll以及空文檔即可。然后就是添加一個類添加一個方法。方法排頭固定格式 extern"C"__declspec(dllexport) 后面加方法即可。 例如如下代碼:
C++dll代碼:
extern "C" __declspec(dllexport) char* ShowImages(BYTE img[],int w,int h){;}
C#調用dll基本也是固定格式,如下樣式,這邊有添加unsafe這是因為有用到指針的緣故[沒用指針不需要unsafe],C#用指針必須添加
unsafe,並且在項目屬性里面的生成允許不安全代碼打鈎才行.
還有一點是C#調用DLL需要添加命名空間
using System.Runtime.InteropServices;
代碼樣式如下:
C#調用代碼:
[DllImport("dll名字.dll", CharSet = CharSet.Auto, CallingConvention = CallingConvention.Cdecl)] unsafe public static extern byte* ShowImages(byte[] src, int w, int h);
按照如上編寫dll以及調用失敗原因多數可能就是C++的C#的數據類型沒有對應的原因。
最后還有一點要提的是,具有指針返回值的方法,該指針變量在C++中必須是全局的,方法體里的私有的會出錯。原因是出了方法體就被釋放了,雖然是return出來的,也沒有效果。
這個問題我之前就遇到過, C++里面return出來一個指針地址,然后C#里面調用,出來的數據亂七八糟,當時一直很郁悶,后來終於發現原因就是指針return出來還是會被釋放掉,所以獲得的數據亂七八糟。解決方法當然是把指針變量設成全局變量。
補充:[2014/01/16]
關於MFC編寫DLL和C++寫DLL是一樣的;[以下是MFC規則DLL]
唯一不同點是在方法體最前添加一句話:AFX_MANAGE_STATE(AfxGetStaticModuleState());
如:
extern "C" __declspec(dllexport) void ShowImages() { AFX_MANAGE_STATE(AfxGetStaticModuleState());//MFC比普通的dll增加的內容(如何不涉及控件可以不加這句) //...... }
關於擴展DLL編寫:
新建擴展DLL工程,添加類后,在類的開頭修改 class AFX_EXT_CLASS CFormatBuilder : public CObject{;}
主要就是添加一個AFX_EXT_CLASS這樣這個類就會作為DLL輸出了。
C# 調用方法不變...
但MFC/Win32調用MFC/Win32的DLL有一些變化,方法如下:
1、顯示調用:[MQXS.dll復制到運行目錄下]
typedef void (*pFunction)(void);// 定義函數變量 ::HINSTANCE hinstance = ::LoadLibraryW(_T("MQXS.dll")); if (hinstance == NULL) MessageBox(_T("DLL加載失敗")); pFunction function = (pFunction)GetProcAddress(hinstance,"ShowImages"); function();
2、隱式調用:[規則DLL][MQXS.dll和MQXS.lib復制到運行目錄下]
#include<stdio.h>
#pragma comment(lib,"MQXS.lib") extern "C" _declspec(dllimport) void ShowImages(); ShowImages();//調用
3、隱式調用:[擴展DLL][MQXS.dll和MQXS.lib以及調用的類的頭文件FormatBuilder.h復制到運行目錄下]
#include "FormatBuilder.h" #pragma comment(lib,"MQXSDLL.lib")
CFormatBuilder builder;//之后就可以使用該類的所有方法,全局變量了
補充:結構體傳輸定義
C++:
typedef struct Student { char name[20]; int age; double scores[32]; }Student; extern "C" __declspec(dllexport) int GetStudent(Student stu);
C#:
[StructLayout(LayoutKind.Sequential)] struct Student { [MarshalAs(UnmanagedType.ByValTStr,SizeConst=20)] public string name; public int age; [MarshalAs(UnmanagedType.ByValArray,SizeConst=32)] public double[] scores; }
數組的傳遞:
C++:
extern "C" __declspec(dllexport) void SetData(int *data)
C#:
public static extern void SetData([MarshalAs(UnmanagedType.LPArray, SizeParamIndex = 1)] ref int[] data);