Asp.Net Core 轻松学-实现跨平台的自定义Json数据包
前言
在前后端分离的业务开发中,我们总是需要返回各种各样的数据包格式,一个良好的 json 格式数据包是我们一贯奉行的原则,下面就利用 Json.Net 来做一个简单具有跨平台的序列化数据包实现类。
1. 应用 Json.Net
- 1.1 首先在项目中引用 NuGet 包

- 1.2 编写一个 JsonReturn 结果包装类,继承自 ContentResult ,并重写 ContentResult 方法 ExecuteResult(ActionContext context)
public partial class JsonReturn : ContentResult
{
public int Code { get; protected set; }
public string Message { get; protected set; }
public Hashtable Data { get; protected set; } = new Hashtable();
public bool Success { get { return this.Code == 0; } }
public JsonReturn(int code, string message) { this.Code = code; this.SetMessage(message); }
public JsonReturn SetMessage(string value) { this.Message = value; return this; }
public JsonReturn SetData(params object[] value)
{
this.Data.Clear();
return this.AppendData(value);
}
public JsonReturn AppendData(params object[] value)
{
if (value?.Length < 2)
return this;
for (int a = 0; a < value.Length; a += 2)
{
if (value[a] == null) continue;
this.Data[value[a]] = a + 1 < value.Length ? value[a + 1] : null;
}
return this;
}
private void ToJson(ActionContext context)
{
this.ContentType = "text/json;charset=utf-8;";
this.Content = JsonConvert.SerializeObject(this);
}
public override Task ExecuteResultAsync(ActionContext context)
{
ToJson(context);
return base.ExecuteResultAsync(context);
}
public override void ExecuteResult(ActionContext context)
{
ToJson(context);
base.ExecuteResult(context);
}
/// <summary>
/// 成功 0
/// </summary>
public static JsonReturn 成功 { get { return new JsonReturn(0, "成功"); } }
/// <summary>
/// 失败 500
/// </summary>
public static JsonReturn 失败 { get { return new JsonReturn(500, "失败"); } }
}
在 JsonReturn 类中,定义了一个存储业务数据对象的 Hashtable 对象,在接口中可以往该对象中写入需要序列化的数据,并重写了 ContentResult 的 ExecuteResultAsync 和 ExecuteResult 方法,在方法内实现 JsonResult 对象的序列化,最后提供了两个静态属性方便调用;在 JsonReutrn 类中,最重要的是定义了成功和失败的 Code ,默认 0 =成功,500=失败,这样就约定了所有客户端都强制使用该协议,完成了标准的统一。
1.3 在控制器中将此对象返回
[HttpGet]
public ActionResult Get()
{
UserInfo info = new UserInfo()
{
Age = 22,
Gender = true,
Name = "Ron.lang",
RegTime = DateTime.Now
};
return JsonReturn.成功.SetData("detail", info);
}
- 1.4 运行程序,得到如下内容
{
"Code": 0,
"Message": "成功",
"Data": {
"detail": {
"Name": "Ron.lang",
"Gender": true,
"Age": 22,
"RegTime": "2018-12-02T16:27:17.3289028+08:00"
}
}
}
2. 改造
2.1 上面的结果还可以接受,只是有一点小瑕疵,比如 bool 类型和字段名称大小写的问题,以及时间格式,都不是太友好,对于跨平台来说,会存在一些问题,下面我们改造一下,使得输出的字段名称全部消息,bool 类型转换为数字 0/1,时间转换为 Unix 格式;首先创建 3 个自定义 json 序列化类
2.2 LowercaseContractResolver.cs 转换字段名称为小写,该类非常简单,仅有一行核心代码
public class LowercaseContractResolver : DefaultContractResolver
{
protected override string ResolvePropertyName(string propertyName)
{
return propertyName.ToLower();
}
}
- 2.3 BooleanConverter.cs 将 bool 类型转换为数字 0/1
public class BooleanConverter : JsonConverter
{
public override bool CanConvert(Type objectType)
{
return objectType == typeof(bool) || objectType == typeof(Nullable<bool>);
}
public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
{
if (reader.Value == null)
return null;
return Convert.ToBoolean(reader.Value);
}
public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
{
if (value == null)
writer.WriteNull();
else
{
UInt32 val = Convert.ToUInt32(Convert.ToBoolean(value));
writer.WriteValue(val);
}
}
}
- 2.4 DateTimeConverter.cs Unix 时间格式转换类
public class DateTimeConverter : DateTimeConverterBase
{
public static DateTime Greenwich_Mean_Time = TimeZoneInfo.ConvertTime(new DateTime(1970, 1, 1), TimeZoneInfo.Local);
public override bool CanConvert(Type objectType)
{
return objectType == typeof(DateTime) || objectType == typeof(Nullable<DateTime>);
}
public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
{
if (reader.Value == null)
return null;
if (CanConvert(objectType))
{
if (string.IsNullOrEmpty(reader.Value.ToNullOrString()))
return reader.Value;
if (reader.Value is string)
{
if (DateTime.TryParse(reader.Value.ToString(), out DateTime dt))
return dt;
else
return reader.Value;
}
else
return new DateTime(Greenwich_Mean_Time.Ticks + Convert.ToInt64(reader.Value) * 10000).ToLocalTime();
}
else
return reader.Value;
}
public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
{
if (value == null)
writer.WriteNull();
else
{
long val = 0;
if (value.GetType() == typeof(DateTime))
{
DateTime dt = Convert.ToDateTime(value);
val = (dt.ToUniversalTime().Ticks - Greenwich_Mean_Time.Ticks) / 10000;
}
else
val = Convert.ToInt64(value);
writer.WriteValue(val);
}
}
}
- 2.5 最后一步,全局注册 JsonSettings 到系统中,打开 Startup.cs 文件,在 Startup 方法中写入以下内容
public Startup(IConfiguration configuration, IHostingEnvironment env)
{
JsonConvert.DefaultSettings = () =>
{
var st = new JsonSerializerSettings
{
Formatting = Formatting.Indented
};
st.Converters.Add(new BooleanConverter());
st.Converters.Add(new DateTimeConverter());
st.ContractResolver = new LowercaseContractResolver();
return st;
};
}
- 2.6 运行程序,接口输出以下内容,完成
{
"code": 0,
"message": "成功",
"data": {
"detail": {
"name": "Ron.lang",
"gender": 1,
"age": 22,
"regtime": 1543739815980
}
}
}
结语
通过继承 ContentResult 实现自定义的序列化数据包,这是刚需;为了实现跨平台的要求,我们还自定义 JsonSettings 实现各种类型的自定义转换,在实际项目开发中,这是非常有用的。
代码下载
https://github.com/lianggx/EasyAspNetCoreDemo/tree/master/Ron.JsonTest
Asp.Net Core 轻松学-实现跨平台的自定义Json数据包的更多相关文章
- Asp.Net Core 轻松学系列-1阅读指引目录
https://www.cnblogs.com/viter/p/10474091.html 目录 前言 1. 从安装到配置 2. 业务实现 3. 日志 4. 测试 5. 缓存使用 6.网络和通讯 7. ...
- Asp.Net Core 轻松学-使用MariaDB/MySql/PostgreSQL和支持多个上下文对象
前言 在上一篇文章中(Asp.Net Core 轻松学-10分钟使用EFCore连接MSSQL数据库)[https://www.cnblogs.com/viter/p/10243577.html],介 ...
- Asp.Net Core 轻松学-多线程之Task(补充)
前言 在上一章 Asp.Net Core 轻松学-多线程之Task快速上手 文章中,介绍了使用Task的各种常用场景,但是感觉有部分内容还没有完善,在这里补充一下. 1. 任务的等待 在使用 ...
- Asp.Net Core 轻松学-利用文件监视进行快速测试开发
前言 在进行 Asp.Net Core 应用程序开发过程中,通常的做法是先把业务代码开发完成,然后建立单元测试,最后进入本地系统集成测试:在这个过程中,程序员的大部分时间几乎都花费在开发.运行 ...
- 如何从40亿整数中找到不存在的一个 webservice Asp.Net Core 轻松学-10分钟使用EFCore连接MSSQL数据库 WPF实战案例-打印 RabbitMQ与.net core(五) topic类型 与 headers类型 的Exchange
如何从40亿整数中找到不存在的一个 前言 给定一个最多包含40亿个随机排列的32位的顺序整数的顺序文件,找出一个不在文件中的32位整数.(在文件中至少确实一个这样的数-为什么?).在具有足够内存的情况 ...
- WebAPI调用笔记 ASP.NET CORE 学习之自定义异常处理 MySQL数据库查询优化建议 .NET操作XML文件之泛型集合的序列化与反序列化 Asp.Net Core 轻松学-多线程之Task快速上手 Asp.Net Core 轻松学-多线程之Task(补充)
WebAPI调用笔记 前言 即时通信项目中初次调用OA接口遇到了一些问题,因为本人从业后几乎一直做CS端项目,一个简单的WebAPI调用居然浪费了不少时间,特此记录. 接口描述 首先说明一下,基于 ...
- C# 中一些类关系的判定方法 C#中关于增强类功能的几种方式 Asp.Net Core 轻松学-多线程之取消令牌
1. IsAssignableFrom实例方法 判断一个类或者接口是否继承自另一个指定的类或者接口. public interface IAnimal { } public interface ID ...
- Asp.Net Core 轻松学-一行代码搞定文件上传 JSONHelper
Asp.Net Core 轻松学-一行代码搞定文件上传 前言 在 Web 应用程序开发过程中,总是无法避免涉及到文件上传,这次我们来聊一聊怎么去实现一个简单方便可复用文件上传功能:通过创建 ...
- Asp.Net Core 轻松学系列-2从安装环境开始
Asp.Net Core 介绍 Asp.Net Core是微软新一代的跨平台开发框架,基于 C# 语言进行开发,该框架的推出,意味着微软从系统层面正式进击 Linux 服务器平台:从更新速度开 ...
随机推荐
- BZOJ_5118_Fib数列2_矩阵乘法+欧拉定理
BZOJ_5118_Fib数列2_矩阵乘法+欧拉定理 Description Fib定义为Fib(0)=0,Fib(1)=1,对于n≥2,Fib(n)=Fib(n-1)+Fib(n-2) 现给出N,求 ...
- B20J_1419_Red Is Good_期望DP
B20J_1419_red is good_期望DP 题意:有R张红牌和B张黑牌,一张一张地翻牌,翻到红牌得到1美元,黑牌则付出1美元.可以随时停止翻牌,在最优策略下平均能得到多少钱. 分析:期望DP ...
- Dubbo原理和源码解析之服务引用
一.框架设计 在官方<Dubbo 开发指南>框架设计部分,给出了引用服务时序图: 另外,在官方<Dubbo 用户指南>集群容错部分,给出了服务引用的各功能组件关系图: 本文将根 ...
- Linux的文件类型
在Linux中,文件的颜色都是有含义的.其中, Linux中文件名颜色不同,代表文件类型不一样. 浅蓝色:表示链接文件: 灰色:表示其他文件: 绿色:表示可执行文件: 红色:表示压缩文件: 蓝色:表示 ...
- 数字类型——python3
今天我为各位小伙伴准备了python3中数字类型,希望能够帮助到你们! Python 数字数据类型用于存储数值. 数据类型是不允许改变的,这就意味着如果改变数字数据类型的值,将重新分配内存空间. 以下 ...
- java基于BasicPlayer调用 播放音乐
无聊中想想用java调用下听音乐的api.晚上很多文章用的比较老大方法了,都是用原生的代码写,而且不支持mp3格式,BasicPlayer第三方包提供了很好的api调用,简单的3行代码就可以调用mp3 ...
- Uiautomator--出现报错“urllib3.exceptions.ProtocolError:<'Connection aborted.',error<10054,''>>”的解决方式!
在运行uiautomator时,出现报错"urllib3.exceptions.ProtocolError:<'Connection aborted.',error<10054, ...
- ASP.NET Core微服务实战系列
希望给你3-5分钟的碎片化学习,可能是坐地铁.等公交,积少成多,水滴石穿,码字辛苦,如果你吃了蛋觉得味道不错,希望点个赞,谢谢关注. 前言 这里记录的是个人奋斗和成长的地方,该篇只是一个系列目录和构想 ...
- java接口与抽象类
本片随笔讲讲java中接口与抽象类. 一,接口 1.什么是接口? 那在日常生活中接口是什么呢?就是两个对象之间进行连接的部分就是接口,就比如热水器与水管的接口一样,他可以确保不同的东西之间的顺利连接, ...
- postgresql 使用pg_restore时显示role "root" does not exist的解决办法
在docker里恢复bakcup格式的数据库,结果提示role "root" does not exist 解决方法: 切换用户: su - postgres 然后再次运行命令: ...