一个 json 转换工具
在前后端的数据协议(主要指http
和websocket
)的问题上,如果前期沟通好了,那么数据协议上问题会很好解决,前后端商议一种都可以接受的格式即可。但是如果接入的是老系统、第三方系统,或者由于某些奇怪的需求(如为了节省流量,json 数据使用单字母作为key
值,或者对某一段数据进行了加密),这些情况下就无法商议,需要在前端做数据转换,如果不转换,那么奔放的数据格式可读性差,也会造成项目难以维护。
这也正是我在项目种遇到的问题,网上也找了一些方案,要么过于复杂,要么有些功能不能很好的支持,于是有了这个工具[class-converter](https://www.npmjs.com/package/class-converter)。
下面我们用例子来说明下:
面对如下的Server
返回的一个用户user
数据:
{
"i": 1234,
"n": "name",
"a": "1a2b3c4d5e6f7a8b"
}
或者这个样的:
{
"user_id": 1234,
"user_name": "name",
"u_avatar": "1a2b3c4d5e6f7a8b"
}
数据里的 avatar
字段在使用时,可能需要拼接成一个 url
,例如 https://xxx.cdn.com/1a2b3c4d5e6f7a8b.png
。
当然可以直接这么做:
const json = {
"i": 1234,
"n": "name",
"a": "1a2b3c4d5e6f7a8b",
};
const data = {};
const keyMap = {
i: 'id',
n: 'name',
a: 'avatar',
}
Object.entries(json).forEach(([key, value]) => {
data[keyMap[key]] = value;
});
// data = { id: 1234, name: 'name', avatar: '1a2b3c4d5e6f7a8b' }
然后我们进一步就可以把这个抽象成一个方法,像下面这个样:
const jsonConverter = (json, keyMap) => {
const data = {};
Object.entries(json).forEach(([key, value]) => {
data[keyMap[key]] = value;
});
return data;
}
如果这个数据扩展了,添加了教育信息,user
数据结构看起来这个样:
{
"i": 1234,
"n": "name",
"a": "1a2b3c4d5e6f7a8b",
"edu": {
"u": "South China Normal University",
"ea": 1
}
}
此时的 jsonConverter
方法已经无法正确转换 edu
字段的数据,需要做一些修改:
const json = {
"i": 1234,
"n": "name",
"a": "1a2b3c4d5e6f7a8b",
"edu": {
"u": "South China Normal University",
"ea": 1
}
};
const data = {};
const keyMap = {
i: 'id',
n: 'name',
a: 'avatar',
edu: {
key: 'education',
keyMap: {
u: 'universityName',
ea: 'attainment'
}
},
}
随着数据复杂度的上升,keyMap 数据结构会变成一个臃肿的配置文件,此外 jsonConverter
方法会越来越复杂,以至于后面同样难以维护。但是转换后的数据格式,对于项目来说,数据的可读性是很高的。所以,这个转换必须做,但是方式可以更优雅一点。
写这个工具的初衷也是为了更优雅的进行数据转换。
工具用法
还是上面的例子(这里使用typescript
写法):
import { toClass, property } from 'class-converter';
// 待解析的数据
const json = {
"i": 1234,
"n": "name",
"a": "1a2b3c4d5e6f7a8b",
};
class User {
@property('i')
id: number; @property('n')
name: string; @property('a')
avatar: string;
}
const userIns = toClass(json, User);
你可以轻而易举的获得下面的数据:
// userIns 是 User 的一个实例
const userIns = {
id: 1234,
name: 'name',
avatar: '1a2b3c4d5e6f7a8b',
}
userIns instanceof User // true
Json
类既是文档又是类似于上文说的与keyMap
类似的配置文件,并且可以反向使用。
import { toPlain } from 'class-converter';
const user = toPlain(userIns, User);
// user 数据结构
{
i: 1234,
n: 'name',
a: '1a2b3c4d5e6f7a8b',
};
这是一个最简单的例子,我们来一个复杂的数据结构:
{
"i": 10000,
"n": "name",
"user": {
"i": 20000,
"n": "name1",
"email": "zqczqc",
// {"i":1111,"n":"department"}
"d": "eyJpIjoxMTExLCJuIjoiZGVwYXJ0bWVudCJ9",
"edu": [
{
"i": 1111,
"sn": "szzx"
},
{
"i": 2222,
"sn": "scnu"
},
{
"i": 3333
}
]
}
}
这是后端返回的一个叫package
的json对象,字段意义在文档中这么解释:
- i:package 的 id
- n:package 的名字
- user:package 的所有者,一个用户
- i:用户 id
- n:用户名称
- email:用户email,但是只有邮箱前缀
- d:用户的所在部门,使用了base64编码了一个json字符串
- i:部门 id
- n:部门名称
- edu:用户的教育信息,数组格式
- i:学校 id
- sn:学校名称
我们的期望是将这一段数据解析成,不看文档也能读懂的一个json
对象,首先我们经过分析得出上面一共有4类实体对象:package、用户信息、部门信息、教育信息。
下面是代码实现:
import {
toClass, property, array, defaultVal,
beforeDeserialize, deserialize, optional
} from 'class-converter';
// 教育信息
class Education {
@property('i')
id: number; // 提供一个默认值
@defaultVal('unknow')
@prperty('sn')
schoolName: string;
}
// 部门信息
class Department {
@property('i')
id: number; @prperty('n')
name: string;
}
// 用户信息
class User {
@property('i')
id: number;
@property('n')
name: string; // 保留一份邮箱前缀数据
@optional()
@property()
emailPrefix: string; @optional()
// 这里希望自动把后缀加上去
@deserialize(val => `${val}@xxx.com`)
@property()
email: string; @beforeDeserialize(val => JSON.parse(atob(val)))
@typed(Department)
@property('d')
department: Department; @array()
@typed(Education)
@property('edu')
educations: Education[];
}
// package
class Package {
@property('i')
id: number; @property('n')
name: string; @property('user', User)
owner: User;
}
数据已经定义完毕,这时只要我们执行toClass
方法就可以得到我们想要的数据格式:
{
id: 10000,
name: 'name',
owner: {
id: 20000,
name: 'name1',
emailPrefix: 'zqczqc',
email: "zqczqc@xxx.com",
department: {
id: 1111,
name: 'department'
},
educations: [
{
id: 1111,
schoolName: 'szzx'
},
{
id: 2222,
schoolName: 'scnu'
},
{
id: 3333,
schoolName: 'unknow'
}
]
}
}
上面这一份数据,相比后端返回的数据格式,可读性大大提升。这里的用法出现了@deserialize
、@beforeDeserialize
、@yped
的装饰器,这里对这几个装饰器是管道方式调用的(前一个的输出一个的输入),这里做一个解释:
beforeDeserialize
第一个参数可以最早拿到当前属性值,这里可以做一些解码操作typed
这个是转换的类型,入参是一个类,相当于自动调用toClass
,并且调动时的第一个参数是beforeDeserialize
的返回值或者当前属性值(如果没有@beforeDeserialize
装饰器)。如果使用了@array
装饰器,则会对每一项数组元素都执行这个转换deserialize
这个装饰器是最后执行的,第一个参数是beforeDeserialize
返回值,@typed
返回值,或者当前属性值(如果前面两个装饰器都没设置的话)。在这个装饰器里可以做一些数据订正的操作
这三个装饰器是在执行toClass
时才会调用的,同样的,当调用toPlain
时也会有对应的装饰器@serialize
、@fterSerialize
,结合@typed
进行一个相反的过程。下面将这两个转换过程的流程绘制出来。
调用 toClass
的过程:
调用 toPlain
的过程是调用 toClass
的逆过程,但是有些许不一样,有一个注意点就是:在调用 toClass
时允许出现一对多的情况,就是一个属性可以派生出多个属性,所以调用调用 toPlain
时需要使用 @serializeTarget
来标记使用哪一个值作为逆过程的原始值,具体用法可以参考文档。
一个 json 转换工具的更多相关文章
- Json转换工具
import java.util.List; import com.fasterxml.jackson.core.JsonProcessingException; import com.fasterx ...
- Json转换工具类(基于google的Gson和阿里的fastjson)
在项目之中我们经常会涉及到字符串和各种对象的转换,为此特地整理了一下常用的转换方法 一.基于com.google.code.gson封装的json转换工具类 1. 在pom.xml文件里面引入gson ...
- java json转换工具类
在java项目中,通常会用到json类型的转换,常常需要对 json字符串和对象进行相互转换. 在制作自定义的json转换类之前,先引入以下依赖 <!--json相关工具--><de ...
- AEM上的一个图片转换工具
目的: 不同情况下,同样一张图片,需要不一样大小/背景/尺寸显示. 例子: dam下面有一张940 x 300 的图片: http://localhost:4502/content/dam/geome ...
- SpringMVC整合FastJson:用"最快的json转换工具"替换SpringMVC的默认json转换
2017年11月23日 09:18:03 阅读数:306 一.环境说明 Windows 10 1709 Spring 4.3.12.RELEASE FastJson 1.2.40 IDEA 2017. ...
- json转换工具类:json<===>list或者对象
public class JsonTools { /** * POJO 转 JSON */ public static String createJsonString(Object object) { ...
- json转换工具类
using System;using System.Collections.Generic;using System.Text;using Newtonsoft.Json;using System.I ...
- json转换工具——fastjson的使用
1.maven依赖<dependency> <groupId>com.alibaba</groupId> <artifactId>fastjson< ...
- java util - json转换工具 gson
需要 gson-2.7.jar 包 package cn.java.gson; import com.google.gson.JsonElement; import com.google.gson.J ...
随机推荐
- 用纯css、JavaScript、jQuery简单的轮播图
完成一个可以自动切换或点击数字的轮播图 HTML代码只需要一个div 包含着一个图片和一个列表,我们主要的思路就是通过点击相应的数字,改变图片的 路径. 有4张图片都在img文件夹里,名称为 img ...
- 搜索+简单dp
前言:即使是简单的递归,在复杂度过高时也可以使用简单的dp. 一般有两种情况,一是利用dp思想求最优子结构进行搜索剪枝,二是利用搜索进行dp数组的填充. 例题一.hdu1978 题目大意:这是一个简单 ...
- spring学习笔记(四)我对spring中bean生命周期的理解
我相信大部分同学对spring中bean的生命周期都不陌生,但是如果要详细的说出每一个步骤,可能能说出来的也不多,我之前也是这样,前几天调了一下spring的源码,看了一点书,突然一下明朗了,理解了s ...
- Coursera课程笔记----C程序设计进阶----Week 4
指针(一) (Week 4) 什么是"指针" 互联网上的资源--地址 当获得一个地址,就能得到该地址对应的资源,所以可以把"网址"称为指向资源的"指针 ...
- Vue + Element-ui实现后台管理系统(3)---面包屑 + Tag标签切换功能
面包屑 + Tag标签切换功能 有关后台管理系统之前写过两遍博客,看这篇之前最好先看下这两篇博客.另外这里只展示关键部分代码,项目代码放在github上: mall-manage-system 1.V ...
- Spark离线日志分析,连接Spark出现报错
首先,我的代码是这样的 import org.apache.spark.rdd.RDD import org.apache.spark.{SparkConf, SparkContext} object ...
- indexDB出坑指南
对于入了前端坑的同学,indexDB绝对是需要深入学习的. 本文针对indexDB的难点问题(事务和数据库升级)做了详细的讲解,而对于indexDB的特点和使用方法只简要的介绍了一下.如果你有一些使用 ...
- java - >IO流_缓冲流(高效流)
缓冲流(高效流) 在我们学习字节流与字符流的时候,大家都进行过读取文件中数据的操作,读取数据量大的文件时,读取的速度会很慢,很影响我们程序的效率,那么,我想提高速度,怎么办? Java中提高了一套缓冲 ...
- Lr运行错误Error: Socket descriptor not found. Hint: the problem might be
在controller中,运行时,报如下错误“Error: Socket descriptor not found. Hint: the problem might be solved applyi ...
- Vue Element-UI使用第三方icon图标(转)
转载自:https://www.jianshu.com/p/59dd28f0b9c9 1.打开阿里icon,注册 >登录>图标管理>我的项目 2.新建项目 3. 添加icon到项 ...