如何在C#中创建一个表达式树来表示'String.Contains("term")'?

75

我是一个新手,正在学习表达式树,希望我的翻译能够让你理解。我正在尝试创建一个表示以下内容的表达式树:

t => t.SomeProperty.Contains("stringValue");

到目前为止,我已经得到了:
    private static Expression.Lambda<Func<string, bool>> GetContainsExpression<T>(string propertyName, string propertyValue)
    {
        var parameterExp = Expression.Parameter(typeof(T), "type");
        var propertyExp = Expression.Property(parameter, propertyName);
        var containsMethodExp = Expression.*SomeMemberReferenceFunction*("Contains", propertyExp) //this is where I got lost, obviously :)
        ...
        return Expression.Lambda<Func<string, bool>>(containsMethodExp, parameterExp); //then something like this
    }

我不知道如何引用String.Contains()方法。

请帮忙解答。

4个回答

162

Something like:

class Foo
{
    public string Bar { get; set; }
}
static void Main()
{
    var lambda = GetExpression<Foo>("Bar", "abc");
    Foo foo = new Foo { Bar = "aabca" };
    bool test = lambda.Compile()(foo);
}
static Expression<Func<T, bool>> GetExpression<T>(string propertyName, string propertyValue)
{
    var parameterExp = Expression.Parameter(typeof(T), "type");
    var propertyExp = Expression.Property(parameterExp, propertyName);
    MethodInfo method = typeof(string).GetMethod("Contains", new[] { typeof(string) });
    var someValue = Expression.Constant(propertyValue, typeof(string));
    var containsMethodExp = Expression.Call(propertyExp, method, someValue);

    return Expression.Lambda<Func<T, bool>>(containsMethodExp, parameterExp);
}

你可能会发现这篇文章很有帮助。


如果我想调用不以“子字符串”开头的内容怎么办? - amd
7
不在Expression.Call周围,而是将GetMethod更改为StartsWith? - Marc Gravell
1
@Steve,任何基于表达式的ORM如果不能正确转义或参数化ConstantExpression,那么它就是彻底破损的。这本身不会导致SQL注入漏洞。如果你使用了破损的工具,那么你已经处于任何风险之中。你有实际的例子吗?这听起来像是“FUD”。 - Marc Gravell
1
@Steve 但你有没有一个实际不进行转义的例子。如果ORM正确转义了输入,不使用参数也是有效的...就个人而言,为了简单起见我会使用参数,但这是一种设计选择。 - Marc Gravell
@MarcGravell 您是正确的。 Linq2db在将文本包含到SQL中之前正确转义了它。 我一直困惑于认为需要使用参数。非常感谢您的评论以进行更正!我已关闭我创建的Linq2db问题。 - Appetere
显示剩余2条评论

9

执行以下搜索操作:

ef.Entities.Where(entity => arr.Contains(entity.Name)).ToArray();
跟踪字符串将是:
SELECT .... From Entities ... Where Name In ("abc", "def", "qaz")

我使用下面我创建的方法:

ef.Entities.Where(ContainsPredicate<Entity, string>(arr, "Name")).ToArray();

public Expression<Func<TEntity, bool>> ContainsPredicate<TEntity, T>(T[] arr, string fieldname) where TEntity : class {
  ParameterExpression entity = Expression.Parameter(typeof(TEntity), "entity");
  MemberExpression member = Expression.Property(entity, fieldname);

  var containsMethods = typeof(Enumerable).GetMethods(BindingFlags.Static | BindingFlags.Public)
  .Where(m => m.Name == "Contains");
  MethodInfo method = null;
  foreach (var m in containsMethods) {
    if (m.GetParameters().Count() == 2) {
      method = m;
      break;
    }
  }
  method = method.MakeGenericMethod(member.Type);
  var exprContains = Expression.Call(method, new Expression[] { Expression.Constant(arr), member });
  return Expression.Lambda<Func<TEntity, bool>>(exprContains, entity);
}

7
这个怎么样:
Expression<Func<string, string, bool>> expFunc = (name, value) => name.Contains(value);

在客户端代码中:
    bool result = expFunc.Compile()("FooBar", "Foo");   //result true
    result = expFunc.Compile()("FooBar", "Boo");        //result false

2

以下是创建字符串包含表达树的步骤。

var method = typeof(Enumerable)
    .GetRuntimeMethods()
    .Single(m => m.Name == nameof(Enumerable.Contains) && m.GetParameters().Length == 2);
var containsMethod = method.MakeGenericMethod(typeof(string));
var doesContain = Expression
.Call(containsMethod, Expression.Constant(criteria.ToArray()),
 Expression.Property(p, "MyParam"));

实际使用情况请参见https://raw.githubusercontent.com/xavierjohn/Its.Cqrs/e44797ef6f47424a1b145d69889bf940b5581eb8/Domain.Sql/CatchupEventFilter.cs


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接