C++的回調函數中有一個參數是,是返回一個字符串,原則如下:
typedef void (*TDataEvent)(char *AData ,int ALen);
其中char *AData是從DLL中返回一個字符串,串的內存已經在DLL中分配了
下面中我在C#中定義的委托
public delegate void TDataEvent(Byte[] AData, int ALen);
下面是回調函數的設置代碼:
Event=new clReceivelDllPoxy.TDataEvent(getDate);
ReceDllPoxy.AddServer(1024,Event,2);
其中Event是上面委托的實例,我定義成一個成員這樣就不會被自己釋放
下面是C#中回調函數的實現
public void getDate(byte[] AData,int ALen)
{
//發現每次回調是 AData只有一個字節
}
下面轉載一個別人的代碼,謝謝
using System;
using System.Collections.Generic;
using System.Text;
using System.Runtime.InteropServices;
using System.Reflection;
using System.Reflection.Emit;
namespace AppDllTest
{
/**//// <summary>
/// 非托管動態調用基類,基類不能直接使用,需要實現 FunTable()的虛函數
/// </summary>
public abstract class clDllBasePoxy
{
//-裝入DLL---------
public bool Open(string dllFileName)
{
Hand = LoadLibrary(dllFileName);
if (Hand == 0)
{
return false;
}
FunSet(GetFunTable());
return true;
}
//-關閉DLL---
public bool Close()
{
return FreeLibrary(Hand)!=0;
}
public abstract string[] GetFunTable(); //函數對應表由外部代理類通過 GetFunTable來設置
//調用Windows32下的Kernele32庫中的裝入函數來完成對非托管DLL的引用-------#region //調用Windows32下的Kernele32庫中的裝入函數來完成對非托管DLL的引用-------
//--------------------------------------------------------------
[DllImport("Kernel32")]
private static extern int GetProcAddress(int handle, String funcname);
[DllImport("Kernel32")]
private static extern int LoadLibrary(String funcname);
[DllImport("Kernel32")]
private static extern int FreeLibrary(int handle);
private int Hand = 0; //DLL的句柄
private static Delegate GetAddress(int dllModule, string functionname, Type t) //把指針轉變成C#的代理
{
int addr = GetProcAddress(dllModule, functionname);
if (addr == 0)
{
return null;
}
else
{
return Marshal.GetDelegateForFunctionPointer(new IntPtr(addr), t);
}
}
//--關聯代理和DLL中的函數-----------
private bool FunSet(string[] aFun)
{
Type tp = this.GetType();
string[] Value;
for (int i = 0; i < aFun.Length; i++)
{
Value = aFun[i].Split(','); //"Box, TBox, _Box" 第一項是代理的實例名,第二項是代理的定義名,第三項是DLL中的函數名
if (Value.Length == 3)
{
FieldInfo fi = tp.GetField(Value[0].Trim()); //找實例
Type type = tp.GetNestedType(Value[1].Trim());//找尾托
if (fi != null && type != null)
{
fi.SetValue(this, GetAddress(Hand, Value[2].Trim(), type)); //創建關聯
}
}
}
return true;
}
#endregion
}
public class clDllPoxy : clDllBasePoxy
{
public override string[] GetFunTable()
{
string[] FunTable = new string[]{
"GetFixParamCount, TGetFixParamCount, _GetFixParamCount",
"GetFixParam, TGetFixParam, _GetFixParam"
};
return FunTable;
}
//--輸出函數----------------------------------------------
public TGetFixParamCount GetFixParamCount;
public TGetFixParam GetFixParam;
//--代理描述----------------------------------------------
public delegate int TGetFixParamCount(); //獲取固定參數個數
public delegate bool TGetFixParam(int AIndex, byte []AOutBuf); //固定參數
}
/**//// <summary>
/// C#動態調用托管DLL的基類--------------
/// </summary>
public class clNetDllPoxy
{
//--裝入動態庫----------
public bool Open(string dllFileName, string className)
{
FAsembly = Assembly.LoadFrom(dllFileName);
if (FAsembly == null)
{
return false;
}
Type type = FAsembly.GetTypes()[0]; //第一個對應的名字空間當成是調用的名字空間
FDllName = dllFileName;
FClassName = className;
if (type != null)
{
FNameSpace = type.Namespace;
}
return true;
}
//--設置Dll中的函數范圍---------
public void SetArea(string nameSpace, string className)
{
FNameSpace = nameSpace;
FClassName = className;
}
//--調用指定的方法,注:方法的所有參數都轉化成對象類型
public object Invoke(string funName, object[] ObjArray_Parameter)
{
try
{
Type[] types = FAsembly.GetTypes();
foreach (Type tp in types)
{
if (tp.Namespace == FNameSpace && tp.Name == FClassName)
{
MethodInfo metInfo = tp.GetMethod(funName);
if (metInfo != null)
{
object obj = Activator.CreateInstance(tp); //創建類對象
if (obj != null)
{
return metInfo.Invoke(obj, ObjArray_Parameter);
}
}
}
}
}
catch
{ }
return null;
}
private Assembly FAsembly; //Dll的程序集
private string FDllName;
private string FNameSpace;
private string FClassName;
}
}