超高性能的json序列化之MVC中使用Json.Net
先不废话,直接上代码
Asp.net MVC自带Json序列化
/// <summary>
/// 加载组件列表
/// </summary>
/// <param name="departmentId">作业部/厂</param>
/// <param name="unitId">组件Id</param>
/// <param name="tag">标签号</param>
/// <param name="pageIndex">当前页码</param>
/// <param name="pageSize">每页条数</param>
/// <returns>返回组件json数据</returns>
public JsonResult ListCom(long departmentId, IEnumberable<long> unitIds, string tag, int pageIndex, int pageSize)
{
var dataEntity = LdarService.ListCom(unitIds, tag, pageIndex + , pageSize);
var dataModel = new Page<LdComModel> {Total = dataEntity.Total};
var data =
dataModel.DataList =
dataEntity.DataList.Select(model => Builder.Builder.Convert<LdComModel>(new object[] {model}));//Entity转ViewModel
dataModel.DataList = data;
return Json(new {
msg = CommonModelBuilder.BuildQuerySuccessMessage("组件信息维护", (int) dataModel.Total),
data = dataModel.DataList,
total = dataModel.Total
});
}
显示到前台界面

接收到的报文中有很多导航属性,前台并不需要这些属性.

LdComModel类中关联了很多外表,也就是导航属性,导航也被序列化,这样不科学,会将所有属性包括导航属性都序列化,还可能会造成循环引用,导致报错。
我只想序列需要的字段,这时可以手写一个匿名类
var data=new {
model.AreaName,
model.AreaId,
......
};
这么写字段少还好,字段多就很不爽吧。
这时我们可以用Json.Net序列化,首先引用newtonsoft.json.dll,使用nuget引用比较方便。在不想序列化的属性上打上[JsonIgnore]特性,序列化就会被忽略。
LdComModel类中的部分代码
/// <summary>
/// 分区
/// </summary>
[JsonIgnore]
public LdAreaModel LdAreaModel { get; set; } /// <summary>
/// 区域名称
/// </summary>
public string AreaName
{
get
{
return LdAreaModel.LdarAreaName;
}
}
使用JsonNet序列化
/// <summary>
/// 加载组件列表
/// </summary>
/// <param name="departmentId">作业部/厂</param>
/// <param name="unitId">组件Id</param>
/// <param name="tag">标签号</param>
/// <param name="pageIndex">当前页码</param>
/// <param name="pageSize">每页条数</param>
/// <returns>返回组件json数据</returns>
public JsonResult ListCom(long departmentId, IEnumberable<long> unitIds, string tag, int pageIndex, int pageSize)
{
var dataEntity = LdarService.ListCom(unitIds, tag, pageIndex + , pageSize);
var dataModel = new Page<LdComModel> {Total = dataEntity.Total};
var data =
dataModel.DataList =
dataEntity.DataList.Select(model => Builder.Builder.Convert<LdComModel>(new object[] {model}));//Entity转ViewModel
dataModel.DataList = data;
var result = new JsonNetResult()
{
Data = new
{
msg = CommonModelBuilder.BuildQuerySuccessMessage("组件信息维护", (int) dataModel.Total),
data = dataModel.DataList,
total = dataModel.Total
}
};
return result;
}
这时返回到前台的json中没有多余的导航属性.

导航属性没有被序列化,速度也快了很多。
这样写,虽然可以实现功能,很每次都要new一个JsonNetResult对象,写起来很是不爽,能不能给Controller写个扩展方法,像Json(...)一样直接写JsonNet(...)?
Controller中Json(...)方法的部分源码
/// <summary>
/// 创建一个将指定对象序列化为 JavaScript 对象表示法 (JSON) 的 <see cref="T:System.Web.Mvc.JsonResult"/> 对象。
/// </summary>
///
/// <returns>
/// 将指定对象序列化为 JSON 格式的 JSON 结果对象。在执行此方法所准备的结果对象时,ASP.NET MVC 框架会将该对象写入响应。
/// </returns>
/// <param name="data">要序列化的 JavaScript 对象图。</param>
protected internal JsonResult Json(object data)
{
return this.Json(data, (string) null, (Encoding) null, JsonRequestBehavior.DenyGet);
} // <summary>
/// 创建 <see cref="T:System.Web.Mvc.JsonResult"/> 对象,该对象使用内容类型、内容编码和 JSON 请求行为将指定对象序列化为 JavaScript 对象表示法 (JSON) 格式。
/// </summary>
///
/// <returns>
/// 将指定对象序列化为 JSON 格式的结果对象。
/// </returns>
/// <param name="data">要序列化的 JavaScript 对象图。</param><param name="contentType">内容类型(MIME 类型)。</param><param name="contentEncoding">内容编码。</param><param name="behavior">JSON 请求行为</param>
protected internal virtual JsonResult Json(object data, string contentType, Encoding contentEncoding, JsonRequestBehavior behavior)
{
return new JsonResult()
{
Data = data,
ContentType = contentType,
ContentEncoding = contentEncoding,
JsonRequestBehavior = behavior
};
}
我们可以仿照Controller中的源码,自己给Controller写个扩展方法JsonNet(...)
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Web; namespace System.Web.Mvc
{
public static class ControllerExt
{
/// <summary>
/// 创建一个将指定对象序列化为 JavaScript 对象表示法 (JSON) 的 <see cref="T:System.Web.Mvc.JsonNetResult"/> 对象。
/// </summary>
///
/// <returns>
/// 将指定对象序列化为 JSON 格式的 JSON 结果对象。在执行此方法所准备的结果对象时,ASP.NET MVC 框架会将该对象写入响应。
/// </returns>
/// <param name="controller">控件器</param>
/// <param name="data">要序列化的 JavaScript 对象图。</param>
public static JsonNetResult JsonNet(this Controller controller, object data)
{
return JsonNet(data, (string) null, (Encoding) null, JsonRequestBehavior.DenyGet);
} /// <summary>
/// 创建一个将指定对象序列化为 JavaScript 对象表示法 (JSON) 格式的 <see cref="T:System.Web.Mvc.JsonNetResult"/> 对象。
/// </summary>
///
/// <returns>
/// 将指定对象序列化为 JSON 格式的 JSON 结果对象。
/// </returns>
/// <param name="controller">控件器</param>
/// <param name="data">要序列化的 JavaScript 对象图。</param>
/// <param name="contentType">内容类型(MIME 类型)。</param>
public static JsonNetResult JsonNet(this Controller controller, object data, string contentType)
{
return JsonNet(data, contentType, (Encoding) null, JsonRequestBehavior.DenyGet);
} /// <summary>
/// 创建一个将指定对象序列化为 JavaScript 对象表示法 (JSON) 格式的 <see cref="T:System.Web.Mvc.JsonNetResult"/> 对象。
/// </summary>
///
/// <returns>
/// 将指定对象序列化为 JSON 格式的 JSON 结果对象。
/// </returns>
/// <param name="controller">控件器</param>
/// <param name="data">要序列化的 JavaScript 对象图。</param>
/// <param name="contentType">内容类型(MIME 类型)。</param>
/// <param name="contentEncoding">内容编码。</param>
public static JsonNetResult JsonNet(this Controller controller, object data, string contentType,
Encoding contentEncoding)
{
return JsonNet(data, contentType, contentEncoding, JsonRequestBehavior.DenyGet);
} /// <summary>
/// 创建 <see cref="T:System.Web.Mvc.JsonNetResult"/> 对象,该对象使用指定 JSON 请求行为将指定对象序列化为 JavaScript 对象表示法 (JSON) 格式。
/// </summary>
///
/// <returns>
/// 将指定对象序列化为 JSON 格式的结果对象。
/// </returns>
/// <param name="controller">控件器</param>
/// <param name="data">要序列化的 JavaScript 对象图。</param>
/// <param name="behavior">JSON 请求行为。</param>
public static JsonNetResult JsonNet(this Controller controller, object data, JsonRequestBehavior behavior)
{
return JsonNet(data, (string) null, (Encoding) null, behavior);
} /// <summary>
/// 创建 <see cref="T:System.Web.Mvc.JsonNetResult"/> 对象,该对象使用指定内容类型和 JSON 请求行为将指定对象序列化为 JavaScript 对象表示法 (JSON) 格式。
/// </summary>
///
/// <returns>
/// 将指定对象序列化为 JSON 格式的结果对象。
/// </returns>
/// <param name="controller">控件器</param>
/// <param name="data">要序列化的 JavaScript 对象图。</param>
/// <param name="contentType">内容类型(MIME 类型)。</param>
/// <param name="behavior">JSON 请求行为</param>
public static JsonNetResult JsonNet(this Controller controller, object data, string contentType,
JsonRequestBehavior behavior)
{
return JsonNet(data, contentType, (Encoding) null, behavior);
} /// <summary>
/// 创建 <see cref="T:System.Web.Mvc.JsonNetResult"/> 对象,该对象使用内容类型、内容编码和 JSON 请求行为将指定对象序列化为 JavaScript 对象表示法 (JSON) 格式。
/// </summary>
///
/// <returns>
/// 将指定对象序列化为 JSON 格式的结果对象。
/// </returns>
/// <param name="controller">控件器</param>
/// <param name="data">要序列化的 JavaScript 对象图。</param>
/// <param name="contentType">内容类型(MIME 类型)。</param>
/// <param name="contentEncoding">内容编码。</param><param name="behavior">JSON 请求行为</param>
public static JsonNetResult JsonNet(this Controller controller, object data, string contentType,
Encoding contentEncoding, JsonRequestBehavior behavior)
{
return JsonNet(data, contentType, contentEncoding, behavior);
} /// <summary>
/// 创建 <see cref="T:System.Web.Mvc.JsonNetResult"/> 对象,该对象使用内容类型、内容编码和 JSON 请求行为将指定对象序列化为 JavaScript 对象表示法 (JSON) 格式。
/// </summary>
/// <param name="data">要序列化的 JavaScript 对象图。</param>
/// <param name="contentType">内容类型(MIME 类型)。</param>
/// <param name="contentEncoding">内容编码。</param>
/// <param name="behavior"></param>
/// <returns>JSON 请求行为</returns>
private static JsonNetResult JsonNet(object data, string contentType, Encoding contentEncoding,
JsonRequestBehavior behavior)
{
return new JsonNetResult()
{
Data = data,
ContentType = contentType,
ContentEncoding = contentEncoding,
JsonRequestBehavior = behavior
};
} }
}
Controller扩展方法
写个JsonNetResult类,继承自JsonResult,重写ExecuteResult()方法,内部使用JsonNet来序列化。
using System.Text;
using Newtonsoft.Json; namespace System.Web.Mvc
{
public class JsonNetResult : JsonResult
{
public Encoding ContentEncoding { get; set; }
public string ContentType { get; set; }
public object Data { get; set; } public JsonSerializerSettings SerializerSettings { get; set; }
public Formatting Formatting { get; set; } public JsonNetResult()
{
SerializerSettings = new JsonSerializerSettings();
} public override void ExecuteResult(ControllerContext context)
{
if (context == null)
throw new ArgumentNullException("context"); HttpResponseBase response = context.HttpContext.Response; response.ContentType = !string.IsNullOrEmpty(ContentType)
? ContentType
: "application/json"; if (ContentEncoding != null)
response.ContentEncoding = ContentEncoding; if (Data != null)
{
var writer = new JsonTextWriter(response.Output) { Formatting = Formatting }; JsonSerializer serializer = JsonSerializer.Create(SerializerSettings);
serializer.Serialize(writer, Data); writer.Flush();
}
}
}
}
JsonNetResult
封装后的JsonNet序列化
/// <summary>
/// 加载组件列表
/// </summary>
/// <param name="departmentId">作业部/厂</param>
/// <param name="unitId">组件Id</param>
/// <param name="tag">标签号</param>
/// <param name="pageIndex">当前页码</param>
/// <param name="pageSize">每页条数</param>
/// <returns>返回组件json数据</returns>
public JsonNetResult ListCom(long departmentId, IEnumberable<long> unitIds, string tag, int pageIndex, int pageSize)
{
var dataEntity = LdarService.ListCom(listUnitId, tag, pageIndex + , pageSize);
var dataModel = new Page<LdComModel> {Total = dataEntity.Total};
var data =
dataModel.DataList =
dataEntity.DataList.Select(model => Builder.Builder.Convert<LdComModel>(new object[] {model}));//Entity转ViewModel
dataModel.DataList = data;
return this.JsonNet(new
{
msg = CommonModelBuilder.BuildQuerySuccessMessage("组件信息维护", (int) dataModel.Total),
data = dataModel.DataList,
total = dataModel.Total
});
}
这样调用起来跟自带的Json(...)一样,非常方便。
由于时间关系,博客就先写到这里。不足及错误之处,敬请批评指正。
超高性能的json序列化之MVC中使用Json.Net的更多相关文章
- 使用Json.Net解决MVC中各种json操作
最近收集了几篇文章,用于替换MVC中各种json操作,微软mvc当然用自家的序列化,速度慢不说,还容易出问题,自定义性也太差,比如得特意解决循环引用的问题,比如datetime的序列化格式,比如性能. ...
- MVC中处理Json和JS中处理Json对象
MVC中处理Json和JS中处理Json对象 ASP.NET MVC 很好的封装了Json,本文介绍MVC中处理Json和JS中处理Json对象,并提供详细的示例代码供参考. MVC中已经很好的封装了 ...
- Json序列化与反序列化(对象与Json字符串的转换)--C#
public class JsonHelper { #region Json序列化与反序列化 /// <summary> /// 将json转化为对象 /// (需要提前构造好结构一致的M ...
- 在MVC中使用Json.Net序列化和反序列化Json对象
在.Net的MVC开发中,经常会使用到Json对象,于是,系统提供了JsonResult这个对象,其本质是调用.Net系统自带的Json序列化类JavaScriptSerializer对数据对象进行序 ...
- 解决MVC Json序列化的循环引用问题/EF Json序列化循引用问题---Newtonsoft.Json
1..Net开源Json序列化工具Newtonsoft.Json中提供了解决序列化的循环引用问题: 方式1:指定Json序列化配置为 ReferenceLoopHandling.Ignore 方式2: ...
- 【转】MVC中处理Json和JS中处理Json对象
事实上,MVC中已经很好的封装了Json,让我们很方便的进行操作,而不像JS中那么复杂了. MVC中: public JsonResult Test() { JsonResult json = new ...
- ASP.NET MVC中的Json Binding和Validate
引子:电子商务网站支付功能页面往往会有很多信息,对于这些信息的保存,往往是分步完成的,那么使用Ajax最合适不过了,比如其中的收货人信息模块.这些信息的新建和编辑保存都是用Ajax来完成的.那么有几种 ...
- IOC容器-Autofac在MVC中实现json方式注入使用
在你阅读时,默认已经了解IOC和autofac的基本用法, 我在最近的我的博客项目中运用了IOC autofac 实现了依赖注入 由于我的项目时asp.net MVC所以我目前向大家展示MVC中如何使 ...
- Asp MVC 中处理JSON 日期类型
注:时间有点忙,直接copy 过来的,要查看原址: http://www.developer.com/net/dealing-with-json-dates-in-asp.net-mvc.html I ...
随机推荐
- makefile函数
http://www.cnblogs.com/tianyajuanke/archive/2013/02/16/2610276.html 通用步骤: 编译时,可以不知钉头文件,如果指定头文件,其作用是当 ...
- module
扯淡的人 http://www.cnblogs.com/yuanchenqi/articles/5732581.html 不被调用的 要写在这下面 1 if __name__=="__ ...
- RMQ 训练 之 codevs 1690 开关灯 已经搞定
思路 懒标记法 记stop[rt] 表示 rt这个线段树节点的下方儿子们需要被更新几次 记住是下方 量纲不要乱 否则写的一堆渣代码 我的代码里面black是维护黑灯的数量 其实做烦了 如果是维 ...
- Sql Server分割字符串函数
-- Description: 分割字符串函数 -- SELECT * FROM dbo.Split('a,b,c,d,e,f,g',',') -- ========================= ...
- vrrp
https://zhidao.baidu.com/question/1498182981731748379.html
- day9mysql操作
#!/usr/bin/env python #coding:utf8 import MySQLdb pip install MySQL-python 先创建一个表 mysql> use xym; ...
- day8-异常
异常处理 1.异常基础 在编程过程中为了增加友好性,在程序出现bug时一般不会将错误信息显示给用户,而是现实一个提示的页面,通俗来说就是不让用户看见大黄页!!! try: pass except Ex ...
- 模块化编程时,#include到底要放在哪里?
结合我自己的经验,谈一谈模块化编程时#include应该出现的位置.总结起来大体有二条规则: 一.规则1:只包含必要的头文件 看下面这个模块: ===foo.c==== #include <st ...
- Access restriction错误解决办法
Access restriction错误, XX方法 is not accessible due to restriction on required library XXlib 解决方案: Ecli ...
- DispatcherServlet作用
DispatcherServlet是前端控制器设计模式的实现,提供Spring Web MVC的集中访问点,而且负责职责的分派,而且与Spring IoC容器无缝集成,从而可以获得Spring的所有好 ...