替换表达式树中的参数值

时间:2013-04-09 17:35:26

标签: c# linq expression

经过一段时间的搜索,我仍然没有找到我想要的答案。我找到了关于在树中添加和删除参数的答案,但没有找到有关替换特定参数的任何内容。

我的第一种方法是按照我的意愿工作,我需要用Uri转义值替换partitionKey值,然后返回未转义的结果。

public override IList<T> GetRowEntityList(string partitionKey)
{
    IList<T> rowEntities =  base.GetRowEntityList(Uri.EscapeDataString(partitionKey));
    return rowEntities.Select(UnEscapeRowEntity).ToList();
}

我遇到的问题是重写此方法以表现相同的方式。我已经知道类型T具有属性PartitionKeyRowKey,但也可以包含任何其他数量的属性。

对于样本谓词:

x => x.RowKey == "foo/bar" && x.SomeValue == "test" 

我希望它成为

x => x.RowKey == Uri.EscapeDataString("foo/bar") && x.SomeValue == "test"  

有办法做到这一点吗?

我的基类使用此谓词使用T调用对包含Where(predicate)类型实体的表执行表查找

public override IList<T> GetRowEntityList(System.Linq.Expressions.Expression<Func<T, bool>> predicate)
{
    //modify predicate value here

    return base.GetRowEntityList(predicate);
}

1 个答案:

答案 0 :(得分:2)

您需要实施ExpressionVisitor

class MyVisitor : ExpressionVisitor
{
    protected override Expression VisitBinary(BinaryExpression node)
    {
        if(CheckForMatch(node.Left))
            return Expression.Equal(node.Left, Rewrite(node.Right));

        if(CheckForMatch(node.Right))
            return Expression.Equal(Rewrite(node.Left), node.Right);

        return Expression.MakeBinary(node.NodeType, Visit(node.Left), Visit(node.Right));
    }

    private bool CheckForMatch(Expression e)
    {
        MemberExpression me = e as MemberExpression;
        if(me == null)
            return false;

        if(me.Member.Name == "RowKey" || me.Member.Name == "PartitionKey")
            return true;
        else
            return false;
    }

    private Expression Rewrite(Expression e)
    {
        MethodInfo mi = typeof(Uri).GetMethod("EscapeDataString");

        return Expression.Call(mi, e);
    }
}

我认为这是对的。这有点难以测试。请注意,这仅适用于(x => x.RowKey == "some string")的有限情况。它不适用于(x => x.RowKey.Equals("somestring")。它也不适用于(x => x.RowKey() == "some string")

然后使用实现的访问者重新编写谓词:

Expression<Func<T, bool>> predicate = (s => s.RowKey == "1 2");

ExpressionVisitor v = new MyVisitor();
Expression<Func<T, bool>> rewrittenPredicate = v.Visit(predicate);

//rewrittenPredicate then tests if s.RowKey == "1%202"