FreeSql (二十六)貪婪加載 Include、IncludeMany


1、導航屬性 ManyToOne

ManyToOne 導航屬性通過 ToList(includeNestedMembers: false) 加載,參數說明:

false: 返回 2級 Join 的導航數據(默認);

true: 返回所有層級深度 Join 的導航數據(未使用的導航數據不會返回);

Select<Tag>().Include(a => a.Parent.Parent).ToList(true);

Select<Tag>().Where(a => a.Parent.Parent.Name == "1").ToList(true);
//這樣寫,不需要再標記 Join,解析表達式時自動處理成 LeftJoin

2、導航屬性 OneToMany/ManyToMany

IncludeMany 貪婪加載集合的導航屬性,其實是分兩次查詢,在 ToList 后進行了數據重裝。

Select<Tag>().IncludeMany(a => a.Songs).ToList();

IncludeMany 有第二個參數,可以進行二次查詢前的修飾工作。

Select<Tag>().IncludeMany(a => a.Songs, 
    then => then.Where(song => song.User == "admin")).ToList();

其實在 then 那里,還可以繼續進行向下 Include/IncludeMany。只要你喜歡,向下 100 層都沒問題。

3、變異

沒有配置導航關系,也可以貪婪加載。

Select<Tag>().IncludeMany(a => a.TestManys.Where(b => b.TagId == a.Id));

只查詢每項子集合的前幾條數據,避免像EfCore加載所有數據導致IO性能低下(比如某商品下有2000條評論)。

Select<Tag>().IncludeMany(a => a.TestManys.Take(10));

子集合返回部分字段,避免字段過多的問題。

Select<Tag>().IncludeMany(a => a.TestManys.Select(b => new TestMany { Title = b.Title ... }));

4、IncludeMany 擴展方法

當主數據已存在內存中,子數據怎么加載?所以我們增加了 List<T> 擴展方法,示例如下:

new List<Song>(new[] { song1, song2, song3 }).IncludeMany(fsql, a => a.Tags);

5、IncludeMany 兩種方式對比

方式一(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 Dto {
    public int TypeId { get; set; }
    public List<Vod> Vods { get; set; }
}

var dto = new [] { 1,2,3 }.Select(a => new Dto { TypeId = a }).ToList();
dto.IncludeMany(d => d.Vods.Take(10).Where(vod => vod.TypeId == d.TypeId));

//執行后,dto 每個元素.Vods 將只有 10條記錄

參考資料

系列文章導航


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM