FreeSql 支持导航属性延时加载,即当我们需要用到的时候才进行加载(读取),支持1对1、多对1、1对多、多对多关系的导航属性。
当我们希望浏览某条订单信息的时候,才显示其对应的订单详细记录时,我们希望使用延迟加载来实现,这样不仅加快的了 读取的效率,同时也避免加载不需要的数据。延迟加载通常用于foreach循环读取数据时。
那么我们在定义Model的时候,需要在属性前面添加virtual关键字。如下
public class Order {
[Column(IsPrimary = true)] public int OrderID { get; set; } public string OrderTitle { get; set; } public string CustomerName { get; set; } public DateTime TransactionDate { get; set; } public virtual List<OrderDetail> OrderDetails { get; set; }
}public class OrderDetail {
[Column(IsPrimary = true)] public int DetailId { get; set; } public int OrderId { get; set; } public virtual Order Order { get; set; }
}
延时加载功能默认被关闭的,使用此功能请时,请在申明处开启;
延时加载功能,依赖 FreeSql.Extensions.LazyLoading 包,请前往 nuget 下载;
static IFreeSql fsql = new FreeSql.FreeSqlBuilder()
.UseConnectionString(FreeSql.DataType.MySql, "Data Source=127.0.0.1;Port=3306;User ID=root;Password=root;Initial Catalog=cccddd;Charset=utf8;SslMode=none;Max pool size=10")
.UseLazyLoading(true) //开启延时加载功能
.UseMonitorCommand(
cmd => Console.WriteLine(cmd.CommandText)) //监听SQL命令对象,在执行前
.Build(); //请务必定义成 Singleton 单例模式var order = fsql.Select<Order>().Where(a => a.OrderID == 1).ToOne(); //查询订单表var orderDetail1 = order.OrderDetails; //第一次访问,查询数据库var orderDetail2 = order.OrderDetails; //第二次访问,不查var order1 = orderDetail1.FirstOrDefault(); //访问导航属性,此时不查数据库,因为 OrderDetails 查询出来的时候已填充了该属性
控制台输出内容:
SELECT a.`OrderID`, a.`OrderTitle`, a.`CustomerName`, a.`TransactionDate`
FROM `Order` a
WHERE (a.`OrderID` = 1)
limit 0,1SELECT a.`DetailId`, a.`OrderId`
FROM `OrderDetail` a
WHERE (a.`OrderId` = 1)
FreeSql延时加载支持1对1、多对1、1对多、多对多关系的导航属性,前三者大小同异,以下我们单独介绍多对多关系。
多对多延时加载
public partial class Song {
[Column(IsIdentity = true)] public int Id { get; set; } public DateTime? Create_time { get; set; } public bool? Is_deleted { get; set; } public string Title { get; set; } public string Url { get; set; } public virtual ICollection<Tag> Tags { get; set; }
}public partial class Song_tag { public int Song_id { get; set; } public virtual Song Song { get; set; } public int Tag_id { get; set; } public virtual Tag Tag { get; set; }
}public partial class Tag {
[Column(IsIdentity = true)] public int Id { get; set; } public int? Parent_id { get; set; } public virtual Tag Parent { get; set; } public decimal? Ddd { get; set; } public string Name { get; set; } public virtual ICollection<Song> Songs { get; set; }
}
如上有三个表,音乐、标签,以及他们的关系表。
var songs = fsql.Select<Song>().Limit(10).ToList(); //取10条音乐var songs1 = songs.First().Tags; //第一次访问,查询数据库var songs2 = Songs.First().Tags; //第二次访问,不查
控制台输出内容:
SELECT a.`Id`, a.`Create_time`, a.`Is_deleted`, a.`Title`, a.`Url`
FROM `Song` a
limit 0,10SELECT a.`Id`, a.`Parent_id`, a.`Ddd`, a.`Name`
FROM `Tag` a
WHERE (exists(SELECT 1
FROM `Song_tag` b
WHERE (b.`Song_id` = 2 AND b.`Tag_id` = a.`Id`)
limit 0,1))
总结
优点:只在需要的时候加载数据,不需要预先计划,避免了各种复杂的外连接、索引、视图操作带来的低效率问题。
缺陷:多次与DB交互,性能降低。
如果要在循环中使用数据,请使用贪婪加载,否则使用懒加载。