发生异常的代码如下:
1: public JsonResult GetSaleByNo1(string id)
2: {
3: SaleMvcUI.Helper.saleDBEntities saleDB = new Helper.saleDBEntities();
4:
5: var saleF = (from sf in saleDB.t_saleform
6: where sf.f_saleform_no == id
7: select sf).First();
8:
9: return this.Json(saleF, JsonRequestBehavior.AllowGet);
10: }
运行后错误信息如下:
1: 序列化类型为“SaleMvcUI.Helper.t_saleform”的对象时检测到循环引用。
2: 说明: 执行当前 Web 请求期间,出现未经处理的异常。请检查堆栈跟踪信息,以了解有关该错误以及代码中导致错误的出处的详细信息。
3:
4: 异常详细信息: System.InvalidOperationException: 序列化类型为“SaleMvcUI.Helper.t_saleform”的对象时检测到循环引用。
5:
6: 源错误:
7:
8: 执行当前 Web 请求期间生成了未经处理的异常。可以使用下面的异常堆栈跟踪信息确定有关异常原因和发生位置的信息。
问题分析:
这里在数据库中设计的表有两个,分别结构如下图所示:
这两个表有1对多的实体映射关系。在生成的实体模型中,他们是可以通过这个一对多的映射关系查找相互之间的数据的。
在上面的第5行代码里面,这只是定义了一个Linq查询语句并且找出第一条数据,这是没有什么问题。在这个过程中,会自动查找与之有映射关系的数据。
第9行代码是在MVC里面返回一个json对象的数据,在这个过程中将我们找到的这条数据进行序列化为json对象的一个过程。在这个过程的时候,由于这个对象有映射关系,那么它在序列化t_saleform对象的时候会序列化该对象的属性t_saleform_detail对象,而这个属性t_saleform_detail对象又有属性t_saleform对象对象,依次反复。就导致了这个问题的产生。
解决方案:
将该实体对象的访问导航属性时是否自动加载相关对象属性关闭即可解决该问题。相关信息参考(MSDN),代码如下:
saleDB.ContextOptions.LazyLoadingEnabled = false;
或者规避json序列化的时候直接序列化该t_saleform对象,改为序列化其它没有这种映射关系的对象。代码如下:
1: public JsonResult GetSaleByNo(string id)
2: {
3: SaleMvcUI.Helper.saleDBEntities saleDB = new Helper.saleDBEntities();
4:
5: var saleF = (from sf in saleDB.t_saleform
6: where sf.f_saleform_no == id
7: select new
8: {
9: f_saleform_no = sf.f_saleform_no,
10: f_saleform_date = sf.f_saleform_date,
11: f_customer = sf.f_customer,
12: f_sales = sf.f_sales,
13: f_remark = sf.f_remark
14: }).First();
15: //此处为了好转换日期格式,多定义了一个临时变量。
16: var tm = new{
17: f_saleform_no = saleF.f_saleform_no,
18: f_saleform_date = saleF.f_saleform_date.ToString("yyyy-MM-dd"),
19: f_customer = saleF.f_customer,
20: f_sales = saleF.f_sales,
21: f_remark = saleF.f_remark
22: };
23: return this.Json(tm, JsonRequestBehavior.AllowGet);
24: }