6

我在 IQueryable 类型上创建了一个扩展方法,它采用实体子集并根据一些标准过滤它们。我的问题是我不能返回一个由变量组成的联合表达式,而不是先初始化所有变量。如前所述,空值无效。

public static IQueryable<Person> FilterHairColors(this IQueryable<Person> subQuery, string[] hairColors)
    {
        IQueryable<Person> q1 = null;
        IQueryable<Person> q2 = null;
        IQueryable<Person> q3 = null;
        IQueryable<Person> q4 = null;

        foreach (var value in hairColors)
        {
            switch (value)
            {
                case "1":
                    q1 = subQuery.Where(p => p.HairColor_bright == true);
                    break;
                case "2":
                    q2 = subQuery.Where(p => p.HairColor_brown == true);
                    break;
                case "3":
                    q3 = subQuery.Where(p => p.HairColor_dark == true);
                    break;
                case "4":
                    q4 = subQuery.Where(p => p.HairColor_red == true);
                    break;
            }
        }
        return q1.AsQueryable().Union(q2.AsQueryable()).Union(q3.AsQueryable()).Union(q4.AsQueryable());
    }

呈现的代码块是多个代码块的一部分,每个代码块都会产生一个数据子集,以这种方式传送到后续过滤方法:

results = persons.FilterGender(vm.gender).FilterAge(vm.age).FilterHeight(vm.height)......  
4

2 回答 2

3

当参数之一为空时,不要调用 Union。

“空查询”对您意味着什么?如果它的意思是“没有行”,那么就不要联合它。如果它表示“所有行”,则不需要合并,因为您可以只采用底层的未过滤查询。

像这样:

var result = new [] { q1, q2, q3, q4, }.Where(query => query != null).Aggregate(Queryable.Union);

这是使用 LINQ-to-Objects 来构造 LINQ-to-SQL 查询。

一个新版本:

var result = dataContext.Persons.Where(_ => false);
if(q1 != null) result = result.Union(q1);
if(q2 != null) result = result.Union(q2);
if(q3 != null) result = result.Union(q3);
if(q4 != null) result = result.Union(q4);

SQL Server 查询优化器将删除第一个虚拟查询,使其完全没有运行时成本。

于 2012-06-16T21:52:32.480 回答
1
public static IQueryable<Person> FilterHairColors(this IQueryable<Person> subQuery, string[] hairColors)
{
    var result = new Person[] { }.AsQueryable();
    var contains = new Dictionary<string, Expression<Func<Person, bool>>>();

    contains.Add("1", p => p.HairColor_bright);
    contains.Add("2", p => p.HairColor_brown);
    contains.Add("3", p => p.HairColor_dark);
    contains.Add("4", p => p.HairColor_red);

    foreach (var color in hairColors)
    {
        result = subQuery.Where(contains[color]).Union(result);
    }

    return result;
}
于 2012-06-17T04:37:43.357 回答