这是对prior thread的一种后续行动。我正在构建一个小包装器来对我的用户提供的动态类型化方法进行upcalls。该方案运作良好......但仅适用于静态方法。虽然CreateDelegate也适用于实例方法,但是当与那些方法一起使用时,如果方法isStatic标志为false,它会抛出“绑定错误”(实际上,因为我有throw-on-error标志为false,它返回null) 。这是一个代码示例,您可以在其中看到这种情况。
using System;
using System.Linq;
using System.Linq.Expressions;
using System.Reflection;
namespace ConsoleApplication4
{
delegate void myFoo(int i, string s);
delegate void myNull();
internal class Callable
{
internal int nParams;
internal Type[] ptypes;
internal Delegate cb;
internal static Type[] actions = { typeof(Action), typeof(Action<>), typeof(Action<,>), typeof(Action<,,>), typeof(Action<,,,>), typeof(Action<,,,,>),
typeof(Action<,,,,,>), typeof(Action<,,,,,,>), typeof(Action<,,,,,,,>), typeof(Action<,,,,,,,,>), typeof(Action<,,,,,,,,,>),
typeof(Action<,,,,,,,,,,>), typeof(Action<,,,,,,,,,,,>), typeof(Action<,,,,,,,,,,,,>), typeof(Action<,,,,,,,,,,,,,>), typeof(Action<,,,,,,,,,,,,,,>) };
internal Callable(Delegate hisCb)
{
MethodInfo mi = hisCb.Method;
ParameterInfo[] pi = mi.GetParameters();
ptypes = pi.Select(p => p.ParameterType).ToArray();
nParams = ptypes.Length;
if (nParams > 0 && nParams < 17)
{
cb = Delegate.CreateDelegate(actions[nParams].MakeGenericType(ptypes), mi, false);
if (cb == null)
Console.WriteLine("Warning: Unsuccessful attempt to CreateDelegate for " + hisCb + ", with methodinfo " + mi);
else
Console.WriteLine("Successful attempt to CreateDelegate for " + hisCb + ", with methodinfo " + mi);
}
else
cb = hisCb;
}
internal void doUpcall(object[] args)
{
if (args.Length != nParams)
throw new ArgumentException("Argument count must match number of parameters");
switch (nParams)
{
case 1:
((dynamic)cb).Invoke((dynamic)args[0]);
break;
case 2:
((dynamic)cb).Invoke((dynamic)args[0], (dynamic)args[1]);
break;
// ... cases 3-15 similar, omitted to save space
default:
cb.DynamicInvoke((dynamic)args);
break;
}
}
}
internal class FooBar
{
internal FooBar()
{
}
internal static void printFields(int i, string s)
{
Console.WriteLine("In FooBar.printField-s with i="+i+", s="+s);
}
internal void printFieldi(int i, string s)
{
Console.WriteLine("In FooBar.printField-i with i=" + i + ", s=" + s);
}
}
internal class Program
{
private static void Main(string[] args)
{
FooBar myFooBar = new FooBar();
Callable cbfb0 = new Callable((myFoo)FooBar.printFields);
cbfb0.doUpcall(new object[] { 77, "myfb" });
Callable cbfb1 = new Callable((myFoo)myFooBar.printFieldi);
cbfb1.doUpcall(new object[] { 77, "myfb" });
string pc = "Main";
Callable cb0 = new Callable((myNull)delegate() { Console.WriteLine("Hello from myNull"); });
cb0.doUpcall(new object[0]);
Callable cb1 = new Callable((myFoo)delegate(int i, string s) { Console.WriteLine("i=" + i + ", s.Length = " + s.Length); });
Console.WriteLine("About to attempt to call Foo: Good args");
cb1.doUpcall(new object[] { 2, "bar" });
Console.WriteLine("After calling Foo");
Callable cb2 = new Callable((myFoo)delegate(int i, string s) { Console.WriteLine("My parent class is " + pc + ", i=" + i + ", s.Length = " + s.Length); });
Console.WriteLine("About to attempt to call Foo: Good args");
cb2.doUpcall(new object[] { 12, "Bar" });
Console.WriteLine("After calling Foo");
System.Threading.Thread.Sleep(15000);
}
private static void Foo(int i, string s)
{
Console.WriteLine("i=" + i + ", s.Length = " + s.Length);
}
}
}
任何人都可以帮助我理解为什么CreateDelegate会以这种方式运行吗? C#和.NET参考信息表明它应该适用于静态方法和实例方法。如果您打破“不成功”的情况,您可以确认确定成功或失败的事情是mi.isStatic标志的值。
PS:注意使用(动态)在运行时将参数强制转换为所需类型!我认为这很酷。以前是不可能的 - 你想做一个强制转换(T)但不知道T类型是什么,因此可以创建一个T类型的对象,但不能动态调用使用该对象的方法,就像我的15个案例陈述一样。通过转换为(动态)我避免了这个问题 - 解决了一个问题,似乎有几十个旧线程未解决! (这改进了先前线程中建议的代码......它具有使用已知类型进行相同的转换问题)。
答案 0 :(得分:5)
CreateDelegate
创建一个可调用的委托 - 为了做到这一点,它必须有实例来调用该方法。对于静态方法,不需要实例 - 您正在调用允许创建静态方法委托的重载。
要在实例方法上创建委托,您必须使用允许传入实例的重叠:
cb = Delegate.CreateDelegate(actions[nParams].MakeGenericType(ptypes),
hisCb.Target,
mi, false);
答案 1 :(得分:1)
我看到Delegate.CreateDelegate
的唯一调用是调用Type, MethodInfo, bool
重载的调用。 documentation表示它
创建指定类型的委托以表示指定的静态方法...
如果你有一个实例方法,你必须调用一个不同的重载(一个需要object
的重载)来从中创建一个委托。