代码之家  ›  专栏  ›  技术社区  ›  Rei Miyasaka

如何在linq(to ef或sql)中组合多个重构的select表达式?

  •  3
  • Rei Miyasaka  · 技术社区  · 14 年前

    假设我有这个视图模型:

    public class SeriesLinkViewModel
    {
        public static Expression<Func<Series, SeriesLinkViewModel>> FromSeries =
            s => new SeriesLinkViewModel
            {
                Name = s.Name,
                Slug = s.Slug,
            };
    
        public string Name { get; set; }
        public string Slug { get; set; }
    }
    

    为了方便起见,我把投影函数放在里面,现在我可以说:

    var links = dc.Series.Select(SeriesLinkViewModel.FromSeries);
    

    令人惊叹的。但是如果我想添加到这个查询中,我该怎么做呢?说我还想拉一个 Description 表中的列。通常,我可以做一个 select new { } 描述 在这里,但是我不能完全这样做,因为我只能在`.select()中放置一个投影函数。

    我希望我可以这样做:

    q = from s in dc.Series
        select new
        {
            Series = SeriesLinkViewModel.FromSeries.Compile()(s),
            Description = s.Description
        };
    

    但我有一个例外:

    System.InvalidcastException:无法 类型的强制转换对象 'System.Linq.Expressions.FieldExpression' 键入 “System.Linq.Expressions.LambdaExpression”。

    或者我至少可以在一次往返中完成所有这些查询吗?我知道TransactionScope用于进行更改,但我认为它不会导致一次性完成查询。

    4 回复  |  直到 14 年前
        1
  •  6
  •   Rei Miyasaka    14 年前

    用它解决了 LinqKit :

    var fs = SeriesLinkViewModel.FromSeries; //needs to be local for some reason
    q = from s in dc.Series.AsExpandable() //enables LinqKit to do its magic
        select new
        {
            Series = fs.Invoke(s), //and voila!
            Description = s.Description
        };
    
        2
  •  2
  •   Community CDub    7 年前

    这是对Rei答案的补充(我将其标记为正确)。如果要删除对局部变量的依赖关系,请查看 this answer from Dan Abramov on how to fix LinqKit .

        3
  •  0
  •   Peter Willis    14 年前

    我知道这不是你想要的,但一个可能的解决方法是创建这样的方法

    private IQueryable<SeriesLinkViewModel> FromSeries(IQueryable<Series> seriesQuery)
    {
        return from s in seriesQuery
               select new SeriesLinkViewModel
               {
                    Name = s.Name,
                    Slug = s.Slug
               };
    }
    

    然后,当您想要使用投影时,通过它运行您的查询。

    return FromSeries(from s in Series
                      where s.Name == "foo"
                      select s);
    

    不理想,因为您没有创建可以与其他表达式组合的可重用表达式,但至少您只有一个映射函数,所有类似的查询都可以运行。

        4
  •  0
  •   hernant    7 年前

    我认为这是一个更好的解决办法。

    public class SeriesLinkViewModel
    {
        public static Expression<Func<Series, SeriesLinkViewModel>> FromSeries =
            s => new SeriesLinkViewModel
            {
                Name = s.Name,
                Slug = s.Slug,
            };
    
        public string Name { get; set; }
        public string Slug { get; set; }
    }
    
    public class SeriesLinkExtendedViewModel: SeriesLinkViewModel
    {
        public new static Expression<Func<Series, SeriesLinkExtendedViewModel>> FromSeries = 
            SeriesLinkViewModel.FromSeries.Merge(s => new SeriesLinkExtendedViewModel
            {
                Description = s.Description
            });
    
        public string Description { get; set; }
    }
    
    // Somewhere else...
    var q = from s in dc.Series.Select(SeriesLinkExtendedViewModel.FromSeries);
    

    “merge”扩展方法将返回一个投影,该投影是合并两个投影表达式的结果,因此包含三列:name、slug和description。

    实际实现可以在以下链接中找到: https://coding.abel.nu/2013/01/merging-expression-trees-to-reuse-in-linq-queries/