gpt4 book ai didi

c# - 一种在开放委托(delegate)和封闭委托(delegate)之间执行转换的方法

转载 作者:太空宇宙 更新时间:2023-11-03 11:54:22 27 4
gpt4 key购买 nike

我需要将一个开放委托(delegate)(未指定目标的委托(delegate))高效地转换为封闭委托(delegate)。我分析了我的代码,使用 CreateDelegate() 为实例方法生成一个封闭委托(delegate)的成本占整个运行时间的很大一部分 (>60%)(因为它发生在该类型的每个新实例)。

关于开放和封闭委托(delegate)的一些基本信息是described on the MSDN siteCreateDelegate 的文档中。

我目前的方法是找到一种方法来缓存打开的委托(delegate)(因此生成它的成本只产生一次)并使用另一种方法调用它,该方法为委托(delegate)提供隐式“this”参数。

一个复杂的因素是我不知道委托(delegate)将在编译时表示的方法的签名,除了通过代码中的通用参数。此外,我想避免反射(例如 Invoke()DynamicInvoke()),因为它们在性能方面并没有更好。

static TDel CreateOpenDelegate<TDel>( MethodInfo mi )
{
// creates and returns an open delegate for a delegate of signature TDel
// that will invoke some method as described by MethodInfo {mi}
return (TDel)(object)Delegate.CreateDelegate( typeof(TDel), mi );
}

// simplification of some other code...
// Note that Action<T> is a sample usage, the actual delegate signature and
// invocation parameters vary, and are defined by the consumers of my code
private Action<T> myAction = CreateOpenDelegate<Action<U,T>>( someMethodInfo );
myAction( this, default(T) ); // can't do this since {this} is a hidden parameter...

我已经阅读了 Jon Skeet 关于 Making Reflection Fly and Exploring Delegates 的文章,不幸的是,由于我事先不知道委托(delegate)签名,所以我看不到一种方法来适应那里描述的方法。

如有任何帮助,我们将不胜感激。

最佳答案

如果我正确理解您的要求,您可能可以使用表达式树来完成此操作 - 我有一篇探讨该主题的帖子 here *。这是一个简化版本:

public static D GetMethodAccessor<D>(MethodInfo mi) where D : class
{
Type[] args = typeof(D).GetGenericArguments();
if (args.Length == 0)
throw new ArgumentException("Type argument D must be generic.");

Type instanceType = args[0];

// If return type is not null, use one less arg
bool isAction = mi.ReturnType == typeof(void);
int callArgCount = args.Length - (isAction ? 1 : 2);
Type[] argTypes = args.Skip(1).Take(callArgCount).ToArray();

var param = Expression.Parameter(instanceType, "obj");
var arguments = argTypes.Select((t, i) => Expression.Parameter(t, "p" + i))
.ToArray();
var invoke = Expression.Call(param, mi, arguments);
var lambda = Expression.Lambda<D>(invoke,
Enumerable.Repeat(param, 1).Concat(arguments));

Debug.WriteLine(lambda.Body);
return lambda.Compile();
}

总而言之,我不确定额外的类型参数处理和表达式编译与您的方法相比如何。

就提供隐式“this”而言,可以使用扩展方法吗?

private static Action<U,T> myAction = GetMethodAccessor<Action<U,T>>(myMethod);
public static void MyAction<U,T>(this U u, T t)
{
myAction(u, t);
}

*请注意,我在博文中使用的基于字符串的字典缓存效率极低 - 您需要像示例中那样将委托(delegate)缓存到私有(private)实例。

关于c# - 一种在开放委托(delegate)和封闭委托(delegate)之间执行转换的方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/1070584/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com