【问题标题】:Is it possible to implement ExpressionTree.GreaterThan, etc for string so that LINQ can use it是否可以为字符串实现 ExpressionTree.GreaterThan 等,以便 LINQ 可以使用它
【发布时间】:2013-02-16 02:21:06
【问题描述】:

我今天早上看到一个问题 (Query my model on a range of values),似乎由 (https://stackoverflow.com/a/1447926/195550) 回答,但那里的整个情况让我对更通用的解决方案产生了兴趣。

我希望能够使用 Jon Skeet 的答案来实现 Between,它可以在非 SQL 生成的环境中使用字符串键,但似乎字符串没有实现 GreaterThan LessThanGreaterThanOrEqualLessThanOrEqual 运算符妨碍了 Linq 构建执行此操作所需的表达式树。

  • 我意识到可以使用CompareTo 方法进行查询来完成这项任务,但我真的很喜欢query.Between(v=>v.StringKey, "abc", "hjk") 表达式的优雅。

  • 我查看了 System.Linq.Expression 程序集,发现它正在寻找一个名为“op_GreaterThan”的方法,例如用于 GreaterThan 操作,但 我不知道

    1. 我是否可以为字符串实现这一点(知道我不能为字符串扩展实际的 '>' 运算符)
    2. 如何构建正确的方法签名。
  • 我创建了以下示例并进行了测试,这些示例显示了 Between 扩展方法不适用于字符串键的位置。

如果可以为字符串键实现这将是非常优雅的。有人对如何实现这一点有任何建议或见解吗?

来自 Jon Skeet 的运算符之间,添加了包含标志


public static class BetweenExtension
{
    public static IQueryable<TSource> Between<TSource, TKey>(
        this IQueryable<TSource> source,
        Expression<Func<TSource, TKey>> keySelector,
        TKey low,
        TKey high,
        bool inclusive = true) where TKey : IComparable<TKey>
    {
        var key = Expression.Invoke(keySelector, keySelector.Parameters.ToArray());

        var lowerBound = (inclusive)
                  ? Expression.GreaterThanOrEqual(key, Expression.Constant(low))
                  : Expression.GreaterThan(key, Expression.Constant(low));

        var upperBound = (inclusive)
                  ? Expression.LessThanOrEqual(key, Expression.Constant(high))
                  : Expression.LessThan(key, Expression.Constant(high));

        var and = Expression.AndAlso(lowerBound, upperBound);
        var lambda = Expression.Lambda<Func<TSource, bool>>(
                        and, keySelector.Parameters);

        return source.Where(lambda);
    }
}

使用“int”键对上述内容进行工作测试


[TestFixture]
public class BetweenIntTests
{
    public class SampleEntityInt
    {
        public int SampleSearchKey { get; set; }
    }

    private IQueryable<SampleEntityInt> BuildSampleEntityInt(params int[] values)
    {
        return values.Select(
               value => 
               new SampleEntityInt() { SampleSearchKey = value }).AsQueryable();
    }

    [Test]
    public void BetweenIntInclusive()
    {
        var sampleData = BuildSampleEntityInt(1, 3, 10, 11, 12, 15);
        var query = sampleData.Between(s => s.SampleSearchKey, 3, 10);
        Assert.AreEqual(2, query.Count());
    }

    [Test]
    public void BetweenIntNotInclusive()
    {
        var sampleData = BuildSampleEntityInt(1, 3, 10, 11, 12, 15);
        var query = sampleData.Between(s => s.SampleSearchKey, 2, 11, false);
        Assert.AreEqual(2, query.Count());
    }
}

以上使用“字符串”键的非工作测试


[TestFixture]
public class BetweenStringsTests
{

    public class SampleEntityString
    {
        public string SampleSearchKey { get; set; }
    }

    private IQueryable<SampleEntityString> BuildSampleEntityString(params int[] values)
    {
        return values.Select(
               value =>
               new SampleEntityString() {SampleSearchKey = value.ToString() }).AsQueryable();
    }

    [Test]
    public void BetweenStringInclusive()
    {
        var sampleData = BuildSampleEntityString(1, 3, 10, 11, 12, 15);
        var query = sampleData.Between(s => s.SampleSearchKey, "3", "10");
        Assert.AreEqual(2, query.Count());
    }

    [Test]
    public void BetweenStringNotInclusive()
    {
        var sampleData = BuildSampleEntityString(1, 3, 10, 11, 12, 15);
        var query = sampleData.Between(s => s.SampleSearchKey, "2", "11", false);
        Assert.AreEqual(2, query.Count());
    }
}

【问题讨论】:

  • 为什么不直接修改Between 方法以将字符串作为特殊情况处理(并调用CompareTo 而不是大于/小于运算符)?
  • 我会有一个基于 typeof(TKey)==typeof(string) 的特殊情况并使用 CompareTo。或者坦率地说,您可以手动检查运算符,然后检查 CompareTo 以获得更通用的解决方案

标签: c# linq


【解决方案1】:

您必须调用string.CompareTo 方法作为表达式树的一部分。然后,您可以测试其结果。要查看它的外观,请查看调试器中的以下值:

Expression<<Func<string, bool>> filter = str => str.CompareTo("abc") > 0;

【讨论】:

  • 我理解你的示例语句在说什么,并认为我可以做类似Expression&lt;&lt;Func&lt;string, int&gt;&gt; lowCheck = str =&gt; str.CompareTo(low); 的事情来获得一个表达式,然后我可以通过 GreaterThan 检查,但这给了我一个错误没有为 Expression> 实现 GreaterThan - 显然,但我的大脑缺少成功的链接。您能否详细说明您的建议?
  • 您必须将lowCheck 分开并深入到CompareTo 调用。现在看来您尝试使用整个 lowCheck 不仅仅是方法调用 - 它是一个包含参数(和不同参数!)的完整函数。最好的方法是直接使用Expression.Call创建方法调用。
猜你喜欢
  • 2012-08-30
  • 1970-01-01
  • 1970-01-01
  • 2017-12-08
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2017-03-24
  • 1970-01-01
相关资源
最近更新 更多