相信大家都了解Json,因为我们通过Json进行“轻量级的文本数据交换”;但是在Asp.Net MVC 中默认的通过javascriptSerializer做Json序列化,尽管这样,微软提供的这套方案却不尽让人满意;诸如不好用(DateTime日期的格式化、循环引用、属性名开头小写等 )。
但是有一个框架(Json.Net(newtonjs))却很优雅的解决了这些问题,而且是.Net中使用频率非常高的Json库。
原理:在新安装的Json库中有一个“ActionNetResult”对象替代“ActionResult”;不过在我使用的过程中发现竟然变为了“ActionNewResult”类,这个倒是无碍,关键是把咱们的活干了,才是最佳的。既然我们知道了通过对象之间的替换来完成我们的需求,那么开始干吧。
public class JsonNewResult : JsonResult//继承类
{
public JsonNewResult()
{
Settings = new JsonSerializerSettings
{
ReferenceLoopHandling = ReferenceLoopHandling.Ignore,//忽略循环引用,如果设置为Error,则遇到循环引用的时候报错(建议设置为Error,这样更规范)
DateFormatString = "yyyy-MM-dd HH:mm:ss",//日期格式化
ContractResolver = new Newtonsoft.Json.Serialization.CamelCasePropertyNamesContractResolver()
};
}
public JsonSerializerSettings Settings { get; private set; }
public override void ExecuteResult(ControllerContext context)
{
if (context == null)
throw new ArgumentNullException("context");
if (this.JsonRequestBehavior == JsonRequestBehavior.DenyGet
&& string.Equals(context.HttpContext.Request.HttpMethod, "GET",
StringComparison.OrdinalIgnoreCase))
throw new InvalidOperationException("JSON GET is not allowed");
HttpResponseBase response = context.HttpContext.Response;
response.ContentType = string.IsNullOrEmpty(this.ContentType) ? "application/json" :
this.ContentType;
if (this.ContentEncoding != null)
response.ContentEncoding = this.ContentEncoding;
if (this.Data == null)
return;
var scriptSerializer = JsonSerializer.Create(this.Settings);
scriptSerializer.Serialize(response.Output, this.Data);
}
}
控制器中方法操作步骤:
public ActionResult Test()
{
Dog dog = new Dog()
{
BirthDate = DateTime.Now,
Id = 5,
Name = "Jasond"
};
//return Json(dog);
return new JsonNewResult() { Data = dog };
}
不过这种操作方式,好像有一些瑕疵,因为我们进行了内部对象的替换,“偷天换日”完成我们的需求,但是这样的操作仅仅适合刚刚开始的项目,变动不会很大,如果针对系统进行维护和升级的话,还是推荐后一种方式;
不知道各位是否还记得过滤器;我们都知道每次启动MVC架构的时候,默认都会执行控制器里面的方法(Action);那么我们的过滤器便是在此处适用的;它能帮我们在方法执行前后,提前过滤(执行)一些方法。详情请见如下:
public class JsonNetActionFilter : IActionFilter
{
public void OnActionExecuted(ActionExecutedContext filterContext)
{
//result就是Action执行完成后返回的ActionResult
if (filterContext.Result is JsonResult&&!(filterContext.Result is JsonNewResult))
{
JsonResult jsonResult = (JsonResult)filterContext.Result;
JsonNewResult jsonNetResult = new JsonNewResult();
jsonNetResult.ContentEncoding = jsonResult.ContentEncoding;
jsonNetResult.ContentType = jsonResult.ContentType;
jsonNetResult.Data = jsonResult.Data;
jsonNetResult.JsonRequestBehavior = jsonResult.JsonRequestBehavior;
jsonNetResult.MaxJsonLength = jsonResult.MaxJsonLength;
jsonNetResult.RecursionLimit = jsonResult.RecursionLimit;
filterContext.Result = jsonNetResult;
}
}
public void OnActionExecuting(ActionExecutingContext filterContext)
{
}
}
控制器中方法操作步骤:
public ActionResult Test()
{
Dog dog = new Dog()
{
BirthDate = DateTime.Now,
Id = 5,
Name = "Jasond"
};
return Json(dog);
//return new JsonNewResult() { Data = dog };
}
注意:一定要在Global.asax文件中增添一句:GlobalFilters.Filters.Add(new JsonNetActionFilter ());预加载过滤器,使其发挥作用。
以上便是关于Json在MVC架构后端序列化的过程,相信大家也都有了一部分的了解,后期我们一块来了解IOC 容器“Autofac”;敬请期待!