延迟加载与序列化
如果使用了延迟加载(Lazy Load),那么,我们就会在序列化的时候碰到 延迟加载 变成了 “立即加载” 的问题。这是为什么呢,因为序列化器会去访问对象的属性,这就会导致属性的 get 方法内的代码被执行起来。
比如,类型:
class Test
{
public string Name {get; private set;}
public PaperStrategy Paper
{
get
{
if (paper == null)
{
paper = PaperStrategy.GetPaperByTest(this);
}return paper;
}
}
}
当返回给前台的时候,我们只需要返回 Name 属性就可以,但是,如果我们使用 JavaScriptSerializer (即 ASP.NET MVC 默认的 JSON 序列化器)的时候,序列化器会默认去遍历全部的属性,这就会导致业务上并不需要加载的 Paper 被序列化器自动加载了。
这是不能容忍的。解决方案有:
一:为属性加入 [ScriptIgnore]
即:
[ScriptIgnore]
public PaperStrategy Paper
Attribute ScriptIgnore 会通知 JavaScriptSerializer 不去序列化此属性,这样,就不会执行 get 中的代码;
不过,这带来一个问题,如果在别的请求中,又是需要这个属性的 Value ,该怎么办。所以,通过加 Attribute 来指导序列化并不可取。
二:构筑匿名类型
或者,我们在控制器中构筑匿名类型,如下,这就相当于存在一个转换过程,如果属性较多的话,就相对的编码烦多。
public class HomeController : SessionController
{
public ActionResult Test(int id)
{
var test = Session.Get<Test>(id);return Json(new
{
test.Name
}, JsonRequestBehavior.AllowGet);
}
}
三:使用 ViewModel
如果我们觉得以上两种不合意,则可以强迫自己使用 ViewModel,即创建一个 TestDto,只包含需要序列化的字段,当然,这仍然需要一个类似 二 中的转换。不过,过多的实体类,不是我喜欢的,所以并不建议此种做法(如想减少实体类,请参看:减少到处衍生的实体类)。所以,大部分情况下,推荐的做法还是 构筑匿名类型 来达到 延迟加载 和 序列化 之间的平衡,除非我们有十分强烈的使用 ViewModel 的理由,比如:使用绑定。
参考: