这是 linq-to-sql
我有很多不同的类都在做相同的查询,但投影结果略有不同。理想情况下,我希望能够在一个地方进行查询,并将投影传递给Select方法。它适用于具体类型:
public void GetResults() {
var junk = db.SiteProducts.Select(Project());
}
public Expression<Func<DbEntities.SiteProduct, string>> Project() {
return p => p.ProductAlert;
}
但是当我尝试返回匿名类型时,它会失败
public void GetResults() {
var junk = db.SiteProducts.Select(Project());
}
public Expression<Func<DbEntities.SiteProduct, TResult>> Project<TResult>() {
return p => new { p.ProductAlert };
}
我完全理解为什么泛型类型推断在第二种情况下失败了。但是有没有从头开始制作我自己的表达式的技巧 - 我不知道可以让它起作用吗?
答案 0 :(得分:2)
IdeaBlade有一个ProjectionSelector
类,您可以使用它来抽象您的投影。当您需要构建投影查询但不知道编译时涉及的类型时,您可以创建ProjectionSelector
类的实例并在运行时传入类型信息。
可以在此处找到类和示例代码:
创建动态“选择”,“SelectMany”和“GroupBy”子句
http://drc.ideablade.com/xwiki/bin/view/Documentation/dynamic-projection
答案 1 :(得分:2)
这在编译时不起作用。使用动态的东西,你当然可以使它工作。
一个简单的解决方案不是使用匿名类型而是使用自定义DTO类。这样的DTO类只需要很少的行并且易于维护。通常这是一个很好的解决方案。
答案 2 :(得分:2)
这是一个有趣的问题。我认为DTO可以帮助你,但是有一些限制和缺陷需要注意。采用以下 LINQPad 示例:
class ProjectDTO
{
public string Name { get; set; }
public static Expression<Func<Project, ProjectDTO>> ToDTO = (e) => new ProjectDTO
{
Name = e.Name
};
public ProjectDTO() {}
public ProjectDTO(Project project)
{
Name = project.Name;
}
}
void Main()
{
Projects.Select(p => p.Name).Dump();
Projects.Select(ProjectDTO.ToDTO).Dump();
Projects.Select(p => new ProjectDTO(p)).Dump();
}
SQL Generated:
SELECT [t0].[Name]
FROM [Project] AS [t0]
GO
SELECT [t0].[Name]
FROM [Project] AS [t0]
GO
SELECT [t0].[ProjectId], [t0].[Name], [t0].[Description], [t0].[DateCreated], [t0].[DateModified], [t0].[DateComplete], [t0].[CreatedBy]
FROM [Project] AS [t0]
如您所见,您无法使用复制构造函数来分配DTO的属性,因为这会强制从数据库中撤回整个对象。
如果您想扩展基础DTO并为更专业的数据视图添加更多属性,这也会略有限制,这意味着您最终可能会使用类似代码的多个Expression。
但是,我非常喜欢选项二,但我确信这个选项很可能仅限于单一类型的投影,请考虑以下示例:
var query = from p in Projects
join t in Tasks on p.ProjectId equals t.ProjectId
select ProjectDTO.ToDTO; //Can't be used like this
我认为您不能在此类查询语法中使用Expression。一般来说,我认为不会有一个全面运作的解决方案。您可能必须检查您的设计,看看您是否可以根据查询中总是包含的一些非常便宜的属性提供更少的投影?
不使用Dynamic LINQ库或手动构建表达式树,我还想看看LINQ-SQL / LINQ-Entities是否可以创建动态选择。
答案 3 :(得分:1)
如果我正确理解您的问题,您可以使用此代码:
首先声明一种选择数据的方法,如下所示:
public List<TResult> FindAll<TResult>(Func<Regions, TResult> selector) where TResult : class
{
using (RepositoryDataContext = new DataClasses1DataContext())
{
return RepositoryDataContext.Regions.Select<Regions, TResult>(selector).ToList<TResult>();
}
}
然后你可以像这样构建你的select语句:
Func<Regions, SelectAllRegion> select = r => new SelectAllRegion
{
RegionID = r.RegionID,
RegionDescription = r.RegionDescription
};
我的SelectAllRegion
:
public class SelectAllRegion
{
public SelectAllRegion()
{
}
public int RegionID { get; set; }
public string RegionDescription { get; set; }
}
北翼的和区域是Region
表。我希望这能帮到你