【问题标题】:Expression Tree GetString Result表达式树 GetString 结果
【发布时间】:2014-10-21 05:42:22
【问题描述】:

我试图复制实体框架在从表达式创建查询时的行为,并且在使用 Attribute 获取模型的属性时,我找到了使用 ExpressionVisitor 的方法

这是我目前得到的

    internal class NVisitor : ExpressionVisitor
    {
        private readonly ParameterExpression _parameter;
        private readonly Type _type;

        public NVisitor(Type type)
        {
            _type = type;
            _parameter = Expression.Parameter(type);
        }

        protected override Expression VisitParameter(ParameterExpression node)
        {
            return _parameter;
        }

        protected override Expression VisitMember(MemberExpression node)
        {

            if (node.Member.MemberType == MemberTypes.Property)
            {
                var memberName = node.Member.Name;

                PropertyInfo otherMember = _type.GetProperty(memberName);

                var ncols = node.Member.GetCustomAttributes(typeof(NColumn), true);

                if (ncols.Any())
                {
                    var ncol = (NColumn)ncols.First();

                    otherMember = _type.GetProperty(ncol.Name);
                }

                var inner = Visit(node.Expression);
                return Expression.Property(inner, otherMember);
            }

            return base.VisitMember(node);
        }
    }

我有一个属性NColumn,它表示表格列中属性的真实名称,所以我通过属性标记模型的属性

    public class BonusTypeX
    {
        [NColumn("BonusTypeName")]
        public string Name { get; set; }
    }

现在当我试图获取表达式时,

    [TestMethod]
    public void ExpressionTesting2()
    {
        string searchKey = "Xmas";

        Expression<Func<BonusTypeX, bool>> expression = x => x.Name.Contains(searchKey);

        Type t = typeof(tbl_BonusType);

        var body = new NVisitor(t).Visit(expression.Body);

        string a = string.Join(".", body.ToString().Split('.').Skip(1));


        Assert.AreEqual("BonusTypeName.Contains(\"Xmas\")", a);
    }

我知道了

BonusTypeName.Contains(value(Payroll.Test.Administration.TestRepositories+c__DisplayClass13).searchKey)

我期望得到的是

BonusTypeName.Contains("Xmas")

有没有获取表达式字符串的方法?我正在使用

string a = string.Join(".", body.ToString().Split('.').Skip(1));

我认为这可能是错误的.. :)

任何帮助将不胜感激。

【问题讨论】:

  • 你尝试过 expression.ToString() 吗?为什么要使用 Body?
  • 还是没有运气.. 顺便说一句,我为你创建了一个 pastebin 来检查你是否有时间.. pastebin.com/1qxj1irw
  • 失败的是问题..谢谢..

标签: c# expression lambda


【解决方案1】:

在运行时在编译器生成的类中捕获局部变量,这解释了Payroll.Test.Administration.TestRepositories+&lt;&gt;c__DisplayClass13).searchKey 部分。要在表达式中获取生成字段的值,您必须在访问表达式时显式替换它的值:

protected override Expression VisitMember(MemberExpression node)
{    
    if (node.Member.MemberType == MemberTypes.Property)
    {
       var memberName = node.Member.Name;
       PropertyInfo otherMember = _type.GetProperty(memberName);
       var ncols = node.Member.GetCustomAttributes(typeof(NColumn), true);
       if (ncols.Any())
       {
          var ncol = (NColumn)ncols.First();
          otherMember = _type.GetProperty(ncol.Name);
       }

       var inner = Visit(node.Expression);
       return Expression.Property(inner, otherMember);
    }
    if (node.Member.MemberType == MemberTypes.Field)
    {
       if (node.Expression is ConstantExpression)
       {
          var owner = ((ConstantExpression)node.Expression).Value;
          var value = Expression.Constant(((FieldInfo)node.Member).GetValue(owner));
          return value;
       }
    }

    return base.VisitMember(node);
}

【讨论】:

  • 当你说“aaaaaaaaaaaaaaaaaaaaaaahhhh”的那一刻......谢谢:)
  • @VincentDagpin 发生在我们最好的人身上;)
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多