创建具有运行时指定数量参数的委托的简单示例。我认为这几乎是无用的(因为你必须通过DynamicInvoke
调用委托...你没有任何安全性,而且你正在使用最慢的反射执行方法)。
public static Delegate CreateLambda(int num)
{
var parameters = new ParameterExpression[num];
for (int i = 0; i < num; i++)
{
parameters[i] = Expression.Parameter(typeof(int), "p" + i);
}
// We sum all the parameters together
Expression sum = parameters[0];
for (int i = 1; i < num; i++)
{
sum = Expression.Add(sum, parameters[i]);
}
Expression body = sum;
LambdaExpression exp = Expression.Lambda(body, parameters);
return exp.Compile();
}
的Expression.Lambda
意愿真理genete一个Expression<Func<...>>
(或Expression<Action<...>>
),其中Func<...>
是基于给出的参数计算出的,但Expression<...>
是LambdaExpression
一个子类。如果Func<>
或Action<>
的参数太多,则即使在运行时也会生成委托类型。
然后:
int num = 5;
Delegate del = CreateLambda<double>(num);
// Note that we have to convert to object the various parameters,
// because DynamicInvoke uses a object[]
object[] values = Enumerable.Range(1, num).Select(x => (object)(double)x).ToArray();
double result = (double)del.DynamicInvoke(values);
Console.WriteLine("{0}={1}", string.Join("+", values), result);
如果你想有一个Func<T[], T>
它是可能的(并且可能是一个更好的主意):
public static Func<T[], T> CreateLambda<T>(int num)
{
var parameter = Expression.Parameter(typeof(T[]), "p");
// We sum all the parameters together
Expression sum = Expression.ArrayIndex(parameter, Expression.Constant(0));
for (int i = 1; i < num; i++)
{
sum = Expression.Add(sum, Expression.ArrayIndex(parameter, Expression.Constant(i)));
}
Expression body = sum;
var exp = Expression.Lambda<Func<T[], T>>(body, parameter);
return exp.Compile();
}
您只需使用Expression.ArrayIndex()
。
再比如说:
int num = 5;
Func<double[], double> del = CreateLambda<double>(num);
double[] values = Enumerable.Range(1, num).Select(x => (double)x).ToArray();
double result = del(values);
Console.WriteLine("{0}={1}", string.Join("+", values), result);
什么只是Expression.Lambda(...)? –
表达式。Lambda是我用来从前面的操作中形成的表达式tress创建委托或者作为参数传递给其他函数的API函数,我没有在这里展示,我只是使用表达式树存储的表达式树变量ruleExpression –
我也尝试了Plain.Lamda –