我有一个类,它有一个接受委托参数的方法,它将通过反射调用,例如:
namespace NSa
{
public delegate void dlg(string p1, string p2);
public class dyn
{
void method(dlg d)
{
// call d...
}
}
}
在另一个类中,我需要使用反射调用dyn.method
:
namespace NSa
{
public delegate void dlg(string p1, string p2);
public void fun(string a, string b) { Console.Write(a); }
public class other
{
void caller_method()
{
dlg x = fun;
//...
var assembly = System.Reflection.Assembly.LoadFile("xx.dll");
System.Reflection.BindingFlags flags = (System.Reflection.BindingFlags.NonPublic | System.Reflection.BindingFlags.Public | System.Reflection.BindingFlags.Static | System.Reflection.BindingFlags.Instance | System.Reflection.BindingFlags.DeclaredOnly);
System.Reflection.Module[] dynModules = assembly.GetModules();
Type dynType = dynModules[0].GetType("NSa.dyn");
object dynObj = Activator.CreateInstance(dynType);
System.Reflection.MethodInfo dynMethod = dynType.GetMethods(flags).Where(m => m.Name == "method").First();
dynMethod.Invoke(dynObj, new object[] {x});
}
}
}
我得到例外:
Object of type 'NSa.dlg' cannot be converted to type 'NSa.dlg'.
我错过了什么?
看起来您在两个不同的程序集中声明了两次相同的类型。就CLR而言,这将是两种完全不同的类型,它们之间没有转换。
您应该使用与dyn
在同一程序集中声明的委托,以便将其传递给dyn.method
。如果您在编译时不了解该程序集,则需要使用Delegate.CreateDelegate
创建该方法的实例。
我还建议,通过对不同的程序集使用不同的命名空间声明,可以更容易地判断哪个类型来自哪个程序集。