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

时间:2016-03-18 10:15:28

标签: c# expression expression-trees rule-engine runtime-compilation

我正在尝试构建通用函数以在DataRows上应用规则。

但是我在运行程序时遇到了以下错误

这是我的代码。

public Func<T, bool> CompileRuleDataRow<T>(Rule r)
{
    var paramUser = Expression.Parameter(typeof(T));
    Expression expr = BuildExprDataRow<T>(r, paramUser);
    // build a lambda function User->bool and compile it
    //Expression.Lambda<Func<T, bool>>(
    return Expression.Lambda<Func<T, bool>>(expr, paramUser).Compile();
}

构建表达式的功能。

public Expression BuildExprDataRow<T>(Rule r, ParameterExpression param)
{
    ParameterExpression objExpr = Expression.Parameter(typeof(string));
    string defaultMember = "Item";
    ConstantExpression indexExpr = Expression.Constant(r.MemberName);
    Expression leftIndexAccessExpr = Expression.Property(objExpr, defaultMember, indexExpr);
    ExpressionType tBinary;
    // is the operator a known .NET operator?
    ExpressionType.TryParse(r.Operator, out tBinary);
    var right = Expression.Constant(r.TargetValue);
    // use a binary operation, e.g. 'Equal' -> 'u.Age == 15'
    return Expression.MakeBinary(tBinary, leftIndexAccessExpr, right);
}

我在main方法中以这种方式调用它。

var rule = new Rule("Name", "Equal", "3");
Func<DataRow, bool> compiledRuleDataRow = CompileRuleDataRow<DataRow>(rule);
DataTable dt = new DataTable();
dt.Columns.Add("Id");
dt.Columns.Add("Name");
dt.Columns.Add("Address");
for (int i = 0; i < 100000; i++)
{
  dt.Rows.Add(i.ToString(), i.ToString(), i.ToString());
}
//I want to do something like this.
    DataRow[] drFiltered = dt.Select().Where(r => compiledRuleDataRow(r)).ToArray();

以下是我的Rule班级

public class Rule
{
  public string MemberName { get; set; }
  public string Operator { get; set; }
  public string TargetValue { get; set; }

  public Rule(string MemberName, string Operator, string TargetValue)
  {
    this.MemberName = MemberName;
    this.Operator = Operator;
    this.TargetValue = TargetValue;
  }
}

当我运行此代码时,我收到以下错误。

  从范围''引用的'System.Data.DataRow'类型的

变量'',   但它没有定义

     

描述:执行期间发生了未处理的异常   当前的网络请求。请查看堆栈跟踪了解更多信息   有关错误的信息以及它在代码中的起源。

     

异常详细信息:System.InvalidOperationException:变量''   键入'范围''引用的'System.Data.DataRow',但事实并非如此   定义

有人可以帮帮我吗?

1 个答案:

答案 0 :(得分:1)

与@svick一样,我从发布的代码中得到了不同的错误。

无论如何,BuildExprDataRow方法存在两个问题。首先,它没有使用传递的param参数。其次,属性值没有正确转换,因此二元运算符不起作用。

以下是固定方法:

public Expression BuildExprDataRow<T>(Rule r, ParameterExpression param)
{
    var right = Expression.Constant(r.TargetValue);
    var left = Expression.Convert(
        Expression.Property(param, "Item", Expression.Constant(r.MemberName)),
        right.Type);
    var comparison = (ExpressionType)Enum.Parse(typeof(ExpressionType), r.Operator);
    return Expression.MakeBinary(comparison, left, right);
}

顺便说一句,泛型参数T在该方法中未使用,因此您可以将其删除(使该方法不通用)。