超高性能的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 ...
随机推荐
- Python入门4
函数 函数其实在日常的编码过程中,你都在使用,比如print().input().len()等函数,只不过这些都是python给你写好的内置函数,供你是用,内置的函数数量有限,想让python为我们做 ...
- Intellij IDEA +MAVEN+Jetty实现SpringMVC简单查询功能
利用 Intellij IDEA +MAVEN+Jetty实现SpringMVC读取数据库数据并显示在页面上的简单功能 1 新建maven项目,配置pom.xml <project xmlns= ...
- 学习SQL的点点滴滴(五)-DELETE小计
惨痛的教训: 某次在执行delete时,一时疏忽忘记写where条件了, 1.删除tb_mobile_cust_micromsg中的内容,前提是c_customer这个字段的值与#datamod表中c ...
- Windows 查看端口占用和关闭进程
支持原创地址 :http://www.cnblogs.com/moodlxs/p/4145384.html 开始--运行--cmd 进入命令提示符 输入netstat -ano 即可看到所有连接的PI ...
- jQuery的编码标准和最佳实践
不知道在哪里看到了这篇关于jQuery编码的文章,挺实用的,恰好最近在研究jQuery的基础知识,今天打开收藏夹来翻译一下,原文的英语不难,但是内容很实用,可能有大神已经翻译过了,大家看精华就行了. ...
- oracle11g dataguard 完全手册
一.前言: 网络上关于dataguard的配置文章很多,但是很多打着oracle11g的文章实际都是只能在9 10 上运行,比如FAL_CLIENT在11g中已经废弃,但是现在网络上的文章都是没有标注 ...
- [python实现设计模式]-2.模板方法模式---把大象关进冰箱.
平时大家上班都很累,为了增加工作中的欢乐气氛,黄页组准备搞个游戏. 游戏的名字是把大象关进冰箱.游戏很简单,需要把指定的物品放进冰箱. 我们都知道,把大象放进冰箱,分3步. 第一步,打开冰箱门,第二步 ...
- FLAG_ACTIVITY_CLEAR_TOP
看了一篇相关的文章,感觉还不错,链接http://www.cnblogs.com/lwbqqyumidi/p/3775479.html
- sql2008r 收缩数据库日志log文件;删除errorlog文件的方法
1.清空log文件,以减少数据库文件log所占的空间 USE dbname1 ; GO ALTER DATABASE dbname1 SET RECOVERY SIMPLE;--设置简单恢复模式 GO ...
- 前端开发根据url进行页面跳转控制以及实现菜单栏手风琴效果
html中的元素:<ul id="accordion" class="accordion"> <li class="licss&q ...