EF的 NoTracking 的一些记录
NoTracking官方解释
跟踪与非跟踪查询
跟踪行为可控制 Entity Framework Core 是否将有关实体实例的信息保留在其更改跟踪器中。 如果已跟踪某个实体,则该实体中检测到的任何更改都会在
SaveChanges()
期间永久保存到数据库。Entity Framework Core 还会修正从跟踪查询中获取的实体与先前已加载到 DbContext 实例中的实体两者之间的导航属性。
可以理解为:
DbContext如果配置为TrackAll。
则DbContext会跟踪返回的实体实例,记录增删改查的跟踪记录,并自动整合变化,推送Diff Change保留至数据库。
DbContext如果配置为NoTracking。
则DBContext不会跟踪实体变化(不再自动标记实体变化,但手动标记仍有效),并不自动修正变化。
1.对增删改查的影响
NoTracking 不允许 相同的实体被标记多次
也就是 Attach 的实体必须有不同的标记。
但TrackAll不会有此问题,因为会自动修正2个实体标记
2.查询缓存
public void Test() { var id = st.T_User.Select(o => o.ID).FirstOrDefault(); using (FLFPermissionEntities ent = dbr.CreateContext<FLFPermissionEntities>()) { using (FLFPermissionEntities sub1 = new FLFPermissionEntities()) { var jm = sub1.FlexUpdate<T_User>(o => o.Name = "lcc", o => o.ID == id); if (jm.HasMsg()) Assert.Fail(jm.msg); } var user21 = ent.T_User.FirstOrDefault(o => o.ID == id); Console.WriteLine($"TrackAllDBContext 第一次读取:{user21.Name}"); using (FLFPermissionEntities sub2 = new FLFPermissionEntities()) { Console.WriteLine($"改变Name的值为lcc2"); var jm = sub2.FlexUpdate<T_User>(o => o.Name = "lcc2", o => o.ID == id); if (jm.HasMsg()) Assert.Fail(jm.msg); } var user22 = ent.T_User.FirstOrDefault(o => o.ID == id); Console.WriteLine($"TrackAllDBContext 第二次读取:{user22.Name}"); } using (FLFPermissionEntities ent = dbr.CreateReadContext<FLFPermissionEntities>()) { using (FLFPermissionEntities sub1 = new FLFPermissionEntities()) { var jm = sub1.FlexUpdate<T_User>(o => o.Name = "lcc", o => o.ID == id); if (jm.HasMsg()) Assert.Fail(jm.msg); } var user21 = ent.T_User.FirstOrDefault(o => o.ID == id); Console.WriteLine($"NoTrackDBContext 第一次读取:{user21.Name}"); using (FLFPermissionEntities sub2 = new FLFPermissionEntities()) { Console.WriteLine($"改变Name的值为lcc2"); var jm = sub2.FlexUpdate<T_User>(o => o.Name = "lcc2", o => o.ID == id); if (jm.HasMsg()) Assert.Fail(jm.msg); } var user22 = ent.T_User.FirstOrDefault(o => o.ID == id); Console.WriteLine($"NoTrackDBContext 第二次读取:{user22.Name}"); } }
NoTrack的结果是正确的,TrackAll的结果与实际结果不同
会发现 TrackAll的 2次查询 结果是一样的,而且TrackAll的第二次读取时已经访问了数据,但出来的结果 还是lcc 并不是lcc2。
所以TrackAll 会从数据库读取数据后,根据跟踪记录里的数据信息,覆盖数据库中读取的结果
DbContext 认为 跟踪记录的数据必须统一且数据已跟踪记录为准