-
Notifications
You must be signed in to change notification settings - Fork 849
贪婪加载
贪婪加载顾名思议就是把所有要加载的东西一次性读取。
本节内容为了配合【延时加载】而诞生,贪婪加载和他本该在一起介绍,开发项目的过程中应该双管齐下,才能写出高质量的程序。
Select<Tag>().Limit(10).ToList(a => new TestDto());
Select<Tag>().Limit(10).ToList(a => new TestDto { });
Select<Tag>().Limit(10).ToList(a => new TestDto() { });
Select<Tag>().Limit(10).ToList<TestDto>();
Select<Tag>().Limit(10).ToList(a => new TestDto { id = a.Id, name = a.Title });
//相当于先映射 TestDto,再映射 a.Id, a.Title
//注意:v0.11.6 以前的版本是只映射 a.Id, a.Title
这种映射支持单表/多表。
查找规则,查找属性名,会循环内部对象 _tables(join 查询后会增长),以 主表优先查,直到查到相同的字段。
如:
A, B, C 都有 id,Dto { id, a1, a2, b1, b2 },A.id 被映射。也可以指定 id = C.id 映射。
友情提醒:在 dto 可以直接映射一个导航属性
ManyToOne/OneToOne 导航属性通过 ToList() 加载,这个方法有一个参数:includeNestedMembers。
参数说明:
false: 返回 2级 Join 的数据;
true: 返回所有层级深度 Join 的导航数据;
如果查询中已经使用了 a.Parent.Parent 类似表达式,则可以无需 LeftJoin 等操作。
如:
Select<Tag>().Where(a => a.Parent.Name == "1").ToList();
//这样写,不需要再标记 Join,解析表达式时自动处理成 LeftJoin
如果导航属性没有使用,又想加载,可使用 Include 方法。
Select<Tag>().Include(a => a.Parent).ToList();
IncludeMany 贪婪加载集合的导航属性,其实是分两次查询,在 ToList 后进行了数据重装。
Select<Tag>().IncludeMany(a => a.Songs).ToList();
//这是 ManyToMany 关系的贪婪加载
OneToMany 的使用方法相同
IncludeMany 有第二个参数,可以进行二次查询前的修饰工作。
Select<Tag>().IncludeMany(a => a.Songs,
then => then.Where(song => song.User == "admin")).ToList();
然后,其实在 then 那里,还可以继续进行向下 Include/IncludeMany。只要你喜欢,向下 100 层都没问题。
变异的 IncludeMany,即使选择的不是导航属性,也可以贪婪加载。
Select<Tag>().IncludeMany(a => a.TestManys.Where(b => b.TagId == a.Id));
支持联合键关系指定
比如 EFCore include 吧,如何只查询每项子集合的前几条数据,它只可以加载所有导致IO性能低下(如果某些子集合,有100条,200条),FreeSql 可以解决这个问题。
Select<Tag>().IncludeMany(a => a.TestManys.Take(10));
前面介绍 ISelect 中进行贪婪加载集合属性数据,主数据与子数据查询必须在一个代码逻辑内完成。
当主数据已存在内存中,子数据怎么加载?所以我们增加了 List<T> 扩展方法,示例如下:
new List<Song>(new[] { song1, song2, song3 }).IncludeMany(fsql, a => a.Tags);
这是一个扩展方法(IncludeMany),方法名与 ISelect.IncludeMany 同名,参数基本一致(除了需要额外传递 IFreeSql 对象参数),功能也一致(包括前面提到的变异)。
方式一(IncludeMany 扩展方法):
var list111 = fsql.Select<SysModule>()
.Page(1, 10)
.ToList(a => new { Id = a.Id }) //查询数据 id
.Select(a => new SysModule { Id = a.Id }).ToList() //内存操作
.IncludeMany(fsql, a => a.Permissions, then => then.Include(a => a.Button));
SELECT a."Id" as1
FROM "SysModule" a
limit 0,10
SELECT a."Id", a."SysModuleId", a."SysModuleButtonId", a."Status",
a__Button."Id" as5, a__Button."Name", a__Button."EventName", a__Button."EnCode", a__Button."Icon", a__Button."Sort", a__Button."CreateTime"
FROM "SysModulePermission" a
LEFT JOIN "SysModuleButton" a__Button ON a__Button."Id" = a."SysModuleButtonId"
WHERE ((a."SysModuleId") in ('menu1','menu2'))
方式二(直接 IncludeMany + ToList):
var list222 = fsql.Select<SysModule>()
.IncludeMany(m => m.Permissions, then => then.Include(a => a.Button))
.Page(1, 10)
.ToList();
SELECT a."Id", a."ParentId", a."Name", a."Icon", a."UrlAddress", a."IsShow", a."Sort", a."Description", a."CreateTime"
FROM "SysModule" a
limit 0,10
SELECT a."Id", a."SysModuleId", a."SysModuleButtonId", a."Status",
a__Button."Id" as5, a__Button."Name", a__Button."EventName", a__Button."EnCode", a__Button."Icon", a__Button."Sort", a__Button."CreateTime"
FROM "SysModulePermission" a
LEFT JOIN "SysModuleButton" a__Button ON a__Button."Id" = a."SysModuleButtonId"
WHERE ((a."SysModuleId") in ('menu1','menu2'))
案例:查询 Vod 表,分类1、分类2、分类3 各10条数据
class Vod {
public Guid Id { get; set; }
public int TypeId { get; set; }
}
//定义临时类,也可以是 Dto 类
class TempType {
public int TypeId { get; set; }
public List<Vod> Vods { get; set; }
}
var types = new [] { 1,2,3 }.Select(a => new TempType { TypeId = a }).ToList();
types.IncludeMany(type => type.Vods.Take(10).Where(vod => vod.TypeId == type.TypeId));
//执行后,types 每个元素.Vods 将只有 10条记录