繁体   English   中英

变量''类型''从范围''引用,但它没有定义

[英]variable '' of type '' referenced from scope '', but it is not defined

那么,以下代码是自我解释的; 我想使用And运算符将两个表达式合并为一个。 最后一行导致符文时间错误:

附加信息:从范围''引用的'System.String'类型的变量'y',但未定义

码:

Expression<Func<string, bool>> e1 = y => y.Length < 100;
Expression<Func<string, bool>> e2 = y => y.Length < 200;

var e3 = Expression.And(e1.Body, e2.Body);

var e4 = Expression.Lambda<Func<string, bool>>(e3, e1.Parameters.ToArray());
e4.Compile(); // <--- causes run-time error

问题是表达式e1e2中表示变量y参数表达式对象是不同的。 两个变量命名相同且具有相同类型的事实无关紧要: e1.Parameters.First()e2.Parameters.First()不是同一个对象。

这会导致您看到的问题:只有e1的参数y可用于Lambda<> ,而e2的参数y超出范围。

要解决此问题,请使用Expression API创建e1e2 这样您就可以在它们之间共享参数表达式,从而消除了范围问题。

如另一个答案所示,您有两个表达式,其中两个都有一个名为y的参数。 这些不会自动相互关联。

要正确编译表达式,需要同时指定源表达式的参数:

Expression<Func<string, bool>> e1 = (y => y.Length > 0);
Expression<Func<string, bool>> e2 = (y => y.Length < 5);

var e3 = Expression.And(e1.Body, e2.Body);

// (string, string) by adding both expressions' parameters.
var e4 = Expression.Lambda<Func<string, string, bool>>(e3, new[] 
{ 
    e1.Parameters[0], 
    e2.Parameters[0] 
});

Func<string, string, bool> compiledExpression = e4.Compile();

bool result = compiledExpression("Foo", "Foo");

当然,您需要一个只用一个参数组合两个表达式的表达式。 您可以像这样重建表达式:

ParameterExpression param = Expression.Parameter(typeof(string), "y");
var lengthPropertyExpression = Expression.Property(param, "Length");

var e1 = Expression.GreaterThan(lengthPropertyExpression, Expression.Constant(0));
var e2 = Expression.LessThan(lengthPropertyExpression, Expression.Constant(5));

var e3 = Expression.AndAlso(e1, e2);

var e4 = Expression.Lambda<Func<string, bool>>(e3, new[] { param });

Func<string, bool> compiledExpression = e4.Compile();

bool result = compiledExpression("Foo");

至于您的注释,您不想重建表达式,而是对现有表达式的主体和参数执行此操作:这可以使用ExpressionRewriter 结合c#中的两个lambda表达式AndAlso替换表达式Body中的参数名称

Expression<Func<string, bool>> e1 = (y => y.Length > 0);
Expression<Func<string, bool>> e2 = (z => z.Length < 10);

var e3 = ParameterReplacer.AndAlso<string>(e1, e2);

Func<string, bool> compiledExpression = e3.Compile();

bool result = compiledExpression("Foo");

谢谢大家的合作。

由于@dasblinkenlight指出两个表达式中的两个参数不一样。 原因? 好吧,这是编译技巧。 在编译时,它为每个表达式创建一个类,并将每个参数命名为xxx1,xxx2,......与原始名称完全不同。

和.Net 4.0+的答案:

如何结合两个lambda

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM