我想定义一个带有out
参数的 Lambda 表达式。有可能做到吗?
下面是我尝试过的 C# .Net 4.0 控制台应用程序的代码片段。
正如您在 Procedure25 中看到的那样,我可以使用 lambda 表达式来定义具有输出参数的委托,但是,当我想使用 linq 表达式来执行相同操作时,过程 24 中的代码失败并显示:
System.ArgumentException 未处理 Message=“System.Boolean”类型的 ParameterExpression 不能用于“System.Boolean&”类型的委托参数 Source=System.Core
我知道我可以使用带有 bool 成员的输入类对象并以这种方式将值传递回调用者,但我很好奇是否可以以某种方式定义出参数。
谢谢
static void Main(string[] args)
{
Procedure25();
Procedure24();
Console.WriteLine("Done!");
Console.ReadKey();
}
private delegate int Evaluate(string value, out bool usesVars);
private static void Procedure24()
{
// This fails to compile:
//Expression<Evaluate> x = (string val, out bool usesSimVals) =>
//{
// usesSimVals = true;
// Console.WriteLine(val);
// return 1;
//};
ParameterExpression valueParameter = Expression.Parameter(typeof (string));
MethodCallExpression methodCall = Expression.Call(typeof(Console).GetMethod("WriteLine", new Type[] { typeof(object) }), valueParameter);
bool usesVars;
ParameterExpression usesVarsParameter = Expression.Parameter(typeof (bool), "out usesVars");
Expression.Lambda<Evaluate>(methodCall, valueParameter, usesVarsParameter).Compile()("test", out usesVars);
Console.WriteLine(usesVars);
}
private static void Procedure25()
{
Evaluate x = (string value, out bool vars) => { vars = true;
Console.WriteLine(value);
return 1;
};
bool usesVars;
x("test", out usesVars);
}
编辑:
阿尼,太棒了,谢谢。所以关键是在参数类型上调用 MakeByRefType。
为了记录,这里是一个基于 Ani 建议的代码片段:
private static void Procedure24()
{
ParameterExpression valueParameter = Expression.Parameter(typeof (string));
MethodCallExpression methodCall = Expression.Call(typeof(Console).GetMethod("WriteLine", new Type[] { typeof(object) }), valueParameter);
bool usesVars;
ParameterExpression usesVarsParameter = Expression.Parameter(typeof (bool).MakeByRefType(), "out usesVars");
Expression block = Expression.Block(methodCall, Expression.Assign(usesVarsParameter, Expression.Constant(true)), Expression.Constant(1));
int result = Expression.Lambda<Evaluate>(block, valueParameter, usesVarsParameter).Compile()("test", out usesVars);
Console.WriteLine("Result={0}, usesVars={1}", result, usesVars);
}