背景
之前写过两篇文章介绍模型的合法性:
今天讨论的问题其实是关于“”的,只是其中还涉及一种决策:“允许模型处于非常状态”。
测试代码
1 public static void Do() 2 { 3 Database.SetInitializer(new DropCreateDatabaseAlways ()); 4 5 using (var context = new MyDbContext()) 6 { 7 /****************添加一个Note*****************/ 8 var note = new Note { Name = "合法名字" }; 9 context.Set ().Add(note);10 context.SaveChanges();11 /****************添加一个Note*****************/12 13 try14 {15 /****************让Note处于非法状态*****************/16 var firstNote = context.Set ().First();17 firstNote.Name = "非法名称";18 if (firstNote.Name == "非法名称")19 {20 throw new InvalidOperationException("非法名称");21 }22 /****************让Note处于非法状态*****************/23 }24 catch25 {26 //这里会出现BUG,显示的还是非法名字。27 Console.WriteLine(context.Set ().First().Name);28 29 //清空DbContext以后就对了。30 foreach (var entity in context.ChangeTracker.Entries())31 {32 entity.State = EntityState.Detached;33 }34 Console.WriteLine(context.Set ().First().Name);35 }36 }37 }
分析
第一个输出之所以不是期望的结果是因为EntityFramework内置了主键映射模式,内存状态还是处于非法状态,虽然First会导致一次数据库往返。
第二个输出之所以正确是因为清空了主键映射,这样会导致重新用数据库的内容填充主键映射。
结论
出现异常最好终止线程或程序的执行,上边这种BUG是因为使用了一种异常反模式:“把异常作为正常的逻辑处理流程”。
备注
这个错误我犯过,后来的朋友也犯过。