JSON不对称反序列化映射方案
源码Git地址: https://github.com/git-simm/simm-framework.git (欢迎大家提交优化代码 ^_^)
- 用底层权限服务提供的数据格式,把业务代码中不规范的引用都改一遍。影响面实在太广,放弃;
- 加一个数据适配层,从底层权限服务请求到json数据,定义一套匹配的pojo类型进行接收。之后再用适配方法,进行对象转换,得到适合系统使用的业务对象。这种搞法比较传统,代理层、适配器都需要自己人工处理。代码量还是较大,不够优雅;
- 利用反射+注解的方式,让程序自动去匹配不对等的属性,自行完成数据适配的过程。这种搞法就便捷多了,以后遇到名称不匹配的属性,我就直接添加个注解就行了。接下来就开撸吧。
- 简单对象,属性都是简单类型,能够自动映射;
- 复杂的Class,要能递归进行自动映射;
- 复杂的List对象,要能递归进行自动映射;

三、实现方案

- 核心转换实现类 :ProxyJsonUtil
- 映射关系解析实现类: ProxyResolveUtil
- 反射赋值工具类:ReflectUtil
- 权限平台请求代理:AuthorityUtil
1. 新建两个注解 ProxyEntity、ProxyProp
/**
* 代理实体(用于json序列化&反序列化)
*/
@Retention(value = RetentionPolicy.RUNTIME)
@Target({ElementType.TYPE})
public @interface ProxyEntity {
//代理实体名
String value() default "";
} /**
* 代理属性(用于json序列化&反序列化)
*/
@Retention(value = RetentionPolicy.RUNTIME)
@Target({ElementType.FIELD})
public @interface ProxyProp {
//属性名
String value();
}
2、核心代码 反射+递归 进行字段映射
package simm.framework.jsonext; import org.springframework.data.util.Pair;
import simm.framework.jsonext.annotations.ProxyEntity;
import simm.framework.jsonext.annotations.ProxyProp;
import simm.framework.jsonext.entity.ProxyField;
import simm.framework.jsonext.entity.ProxyMsg; import java.lang.reflect.Field;
import java.util.*; /**
* 代理信息解析工具
*/
public class ProxyResolveUtil {
/**
* 同步锁
*/
private static Object _lock = new Object();
/**
* 代理类型缓存
*/
//对类型缓存做线程安全处理
private static Map<String,ProxyMsg> clazzCache = Collections.synchronizedMap(new HashMap<String,ProxyMsg>()); /**
* 获取代理信息
* @param clazz
* @return
*/
public static ProxyMsg getProxyMsg(Class clazz){
String key = clazz.getName();
if(clazzCache.containsKey(key)) return clazzCache.get(key);
synchronized (_lock){
//双重检查
if(clazzCache.containsKey(key)) return clazzCache.get(key);
//开始解析
clazzCache.put(key,getClazzProxyMsg(clazz));
}
return clazzCache.get(key);
} /**
* 获取类型代理信息
* @param clazz
* @return
*/
private static ProxyMsg getClazzProxyMsg(Class clazz){
ProxyEntity proxyEntity = (ProxyEntity) clazz.getAnnotation(ProxyEntity.class);
if(proxyEntity==null) return null;
ProxyMsg proxyMsg = new ProxyMsg();
proxyMsg.setClazz(clazz);
proxyMsg.setProxyName(proxyEntity.value());
//解析字段信息
List<ProxyField> propList = new ArrayList<>();
List<Pair<String, String>> listConfig= new ArrayList<>();
HashMap<String, String> mapConfig= new HashMap<>();
for (Field field: clazz.getDeclaredFields()){
ProxyProp proxyProp = field.getAnnotation(ProxyProp.class);
if(proxyProp == null){
mapConfig.put(field.getName(),field.getName());
continue;
}else{
ProxyField pField = new ProxyField();
pField.setField(field);
pField.setFieldName(field.getName());
pField.setProxyName(proxyProp.value());
propList.add(pField);
//beanutils 做属性拷贝时,使用该参数
listConfig.add(Pair.of(pField.getProxyName(),pField.getFieldName()));
mapConfig.put(pField.getProxyName(),pField.getFieldName());
}
}
proxyMsg.setFields(propList);
proxyMsg.setListConfig(listConfig);
proxyMsg.setMapConfig(mapConfig);
return proxyMsg;
}
}
package simm.framework.jsonext; import java.lang.reflect.*;
import java.util.Collections;
import java.util.HashMap;
import java.util.Map; public class ReflectUtil {
/**
* 同步锁
*/
private static Object _lock = new Object();
/**
* 代理字段缓存
*/
//对类型缓存做线程安全处理
private static Map<String,Field> fieldCache = Collections.synchronizedMap(new HashMap<String,Field>()); /**
* 获取字段信息
* @param entity
* @param fieldName
* @param <E>
* @return
*/
public static <E> Field getField(E entity,String fieldName){
String key = entity.getClass().getName()+"@"+fieldName;
if(fieldCache.containsKey(key)) return fieldCache.get(key);
synchronized (_lock){
//双重检查
if(fieldCache.containsKey(key)) return fieldCache.get(key);
//开始解析
Field f = null;
try {
f = entity.getClass().getDeclaredField(fieldName);
f.setAccessible(true);
} catch (NoSuchFieldException e) {
e.printStackTrace();
}
fieldCache.put(key,f);
}
return fieldCache.get(key);
}
/**
* 获取list 属性的 泛型类型
* @param entity
* @param fieldName
* @return
*/
public static <E> Class<?> getActClazz(E entity, String fieldName){
Field f = getField(entity,fieldName);
if(f.getType() == java.util.List.class){
// 如果是List类型,得到其Generic的类型
Type genericType = f.getGenericType();
if(genericType == null) return null;
// 如果是泛型参数的类型
if(genericType instanceof ParameterizedType){
ParameterizedType pt = (ParameterizedType) genericType;
//得到泛型里的class类型对象
Class<?> genericClazz = (Class<?>)pt.getActualTypeArguments()[0];
return genericClazz;
}
}
return (Class<?>) f.getGenericType();
}
/**
* 获取字段值
* @param target
* @param fieldName
* @param <E>
* @return
* @throws Exception
*/
public static <E> Object getFieldVal(E target, String fieldName){
try {
return getField(target,fieldName).get(target);
} catch (IllegalAccessException e) {
e.printStackTrace();
return null;
}
} /**
* 设置字段值
* @param target
* @param fieldName
* @param <E>
* @return
*/
public static <E> void setFieldVal(E target, String fieldName,Object value) {
try {
getField(target,fieldName).set(target,value);
} catch (IllegalAccessException e) {
e.printStackTrace();
}
}
}
package simm.framework.jsonext; import com.alibaba.fastjson.JSONArray;
import com.alibaba.fastjson.JSONObject;
import com.alibaba.fastjson.parser.ParserConfig;
import com.alibaba.fastjson.util.TypeUtils;
import org.springframework.data.util.Pair;
import simm.framework.jsonext.entity.ProxyMsg; import java.lang.reflect.Type;
import java.util.ArrayList;
import java.util.List;
import java.util.Map; public class ProxyJsonUtil {
/**
* 批量拷贝
* @param source
* @param clazz
* @param <E>
* @param deepCopy 是否需要递归处理
* @return
*/
public static <E> List<E> cast(List<JSONObject> source, Class<E> clazz,boolean deepCopy) {
List<E> result = new ArrayList<>();
for (JSONObject t : source) {
E object = cast(t, clazz, deepCopy);
result.add(object);
}
return result;
} /**
* 单条拷贝
* @param source
* @param clazz
* @param <E>
* @return
*/
public static <E> E cast(JSONObject source, Class<E> clazz, boolean deepCopy) {
return copyProperties(source, clazz, deepCopy);
} /**
* 拷贝属性
* @param source
* @param clazz
* @param <E>
* @param deepCopy
* @return
*/
private static <E> E copyProperties(JSONObject source, Class<E> clazz, boolean deepCopy) {
try {
E object = TypeUtils.castToJavaBean(source, clazz, ParserConfig.getGlobalInstance());
copyProperties(source, object, deepCopy);
return object;
} catch (Exception e) {
e.printStackTrace();
return null;
}
} /**
* JsonObject属性值拷贝
* @param source
* @param target
* @param <E>
* @param deepCopy
*/
private static <E> void copyProperties(JSONObject source,E target, boolean deepCopy) throws Exception {
if(deepCopy){
//深层递归,处理子对象赋值
proxyEntityCast(source, target, deepCopy);
}
//代理字段赋值
proxyFieldCast(source, target);
} /**
* 深层递归,处理子对象赋值
* @param source
* @param target
* @param deepCopy
* @param <E>
* @throws Exception
*/
private static <E> void proxyEntityCast(JSONObject source, E target, boolean deepCopy) throws Exception {
ProxyMsg msg = ProxyResolveUtil.getProxyMsg(target.getClass());
Map<String,String> map = msg.getMapConfig();
for (Map.Entry<String, Object> entry : source.entrySet()){
//映射实体不包含该属性,则退出代理赋值
if(!map.containsKey(entry.getKey()))continue;
//获取映射实体的字段名
String fieldName = map.get(entry.getKey());
Object value = entry.getValue();
if(value instanceof JSONArray){
jsonArrayCast(target, deepCopy, fieldName, (JSONArray) value);
}else if(value instanceof JSONObject){
jsonObjectCast(target, deepCopy, fieldName, (JSONObject) value);
}
}
} /**
* JSONObject转换处理
* @param target
* @param deepCopy
* @param fieldName
* @param value
* @param <E>
* @throws Exception
*/
private static <E> void jsonObjectCast(E target, boolean deepCopy, String fieldName, JSONObject value) throws Exception {
//属性是一个JSONObject 对象
Object fieldTarget = ReflectUtil.getFieldVal(target, fieldName);
if(fieldTarget == null){
Type fieldType = ReflectUtil.getField(target, fieldName).getGenericType();
ReflectUtil.setFieldVal(target, fieldName,copyProperties(value,(Class)fieldType,deepCopy));
}else{
//递归为JsonObject属性赋值
copyProperties(value,fieldTarget,deepCopy);
}
} /**
* JSONArray 字段处理
* @param target
* @param deepCopy
* @param fieldName
* @param value
* @param <E>
* @throws Exception
*/
private static <E> void jsonArrayCast(E target, boolean deepCopy, String fieldName, JSONArray value) throws Exception {
//属性是一个 JSONArray 的列表对象
//获取需要被赋值的目标对象
Object fieldTarget = ReflectUtil.getFieldVal(target, fieldName);
//目标对象为空,退出运行
Class<?> fieldGenericClazz = ReflectUtil.getActClazz(target,fieldName);
if(fieldTarget == null){
fieldTarget = new ArrayList();
ReflectUtil.setFieldVal(target, fieldName,fieldTarget);
}
JSONArray tempList = value;
if(fieldTarget instanceof List){
// 如果是List类型,得到其Generic的类型
List<Object> temps = (List<Object>)fieldTarget;
for(int i=0;i<tempList.size();i++){
//递归为JsonObject属性赋值
if(temps.size()<=i){
//自动创建新对象
temps.add(copyProperties((JSONObject)tempList.get(i),fieldGenericClazz,deepCopy));
}else{
copyProperties((JSONObject)tempList.get(i),temps.get(i),deepCopy);
}
}
}
}
/**
* 代理字段赋值
* @param source
* @param target
* @param <E>
*/
private static <E> void proxyFieldCast(JSONObject source, E target) {
//代理字段赋值
ProxyMsg msg = ProxyResolveUtil.getProxyMsg(target.getClass());
for (Pair<String, String> pair : msg.getListConfig()) {
String key = pair.getFirst();
if(!source.containsKey(key)) continue;
Object value = source.get(key);
if(value instanceof JSONArray || value instanceof JSONObject) continue;
ReflectUtil.setFieldVal(target, pair.getSecond(),value);
}
}
}
JSON不对称反序列化映射方案的更多相关文章
- C# JSON 序列化和反序列化——JavaScriptSerializer实现
一. JavaScriptSerializer 类由异步通信层内部使用,用于序列化和反序列化在浏览器和 Web 服务器之间传递的数据.您无法访问序列化程序的此实例.但是,此类公开了公共 API.因此, ...
- 序列化对象C++对象的JSON序列化与反序列化探索
新手发帖,很多方面都是刚入门,有错误的地方请大家见谅,欢迎批评指正 一:背景 作为一名C++开发人员,我始终很期待能够像C#与JAVA那样,可以省力的进行对象的序列化与反序列化,但到现在为止,还没有找 ...
- C++对象的JSON序列化与反序列化探索
一:背景 作为一名C++开发人员,我一直很期待能够像C#与JAVA那样,可以轻松的进行对象的序列化与反序列化,但到目前为止,尚未找到相对完美的解决方案. 本文旨在抛砖引玉,期待有更好的解决方案:同时向 ...
- Net中JSON序列化和反序列化处理(日期时间特殊处理)
0 缘由 笔者最近在web api端使用Json.Net进行序列化处理,而在调用端使用DataContractSerializer进行反序列化,遇到日期时间处理反序列化不成功[备注:笔者使用Net ...
- C++对象的JSON序列化与反序列化探索完结-列表的序列化与反序列化
在前两篇文章中,我们已经完成对普通对象以及复杂对象嵌套的序列化与反序列化,见如下地址: C++对象的JSON序列化与反序列化探索 C++对象的JSON序列化与反序列化探索续-复杂对象的序列化与反序列化 ...
- C++对象的JSON序列化与反序列化探索续-复杂对象的序列化与反序列化
本文是基本上一篇博文进行改进而成,上一篇请见: C++对象的JSON序列化与反序列化探索 此处就不多说了,直接上代码. 1. 序列化基类 #pragma once #include <strin ...
- 迄今为止 .Net 平台功能最强大,性能最佳的 JSON 序列化和反序列化库。
Swifter.Json 这是迄今为止 .Net 平台功能最强大,性能最佳的 JSON 序列化和反序列化库. Github : https://github.com/Dogwei/Swifter.Js ...
- Newtonsoft.Json C# Json序列化和反序列化工具的使用、类型方法大全 C# 算法题系列(二) 各位相加、整数反转、回文数、罗马数字转整数 C# 算法题系列(一) 两数之和、无重复字符的最长子串 DateTime Tips c#发送邮件,可发送多个附件 MVC图片上传详解
Newtonsoft.Json C# Json序列化和反序列化工具的使用.类型方法大全 Newtonsoft.Json Newtonsoft.Json 是.Net平台操作Json的工具,他的介绍就 ...
- DotNet的JSON序列化与反序列化
JSON(JavaScript Object Notation)JavaScript对象表示法,它是一种基于文本,独立于语言的轻量级数据交换格式.在现在的通信中,较多的采用JSON数据格式,JSON有 ...
随机推荐
- 为什么 I2C(IIC)需要上拉电阻
源鑫问: I2C时钟线和数据线为什么要接上拉电阻? 因为 I2C 的 IO 是开漏的,所以需要上拉电阻. 延伸: 之前 hippo曾经说过有人将 IO 设置为 PP,可能会烧 IO. 之前以为 I2C ...
- Eclipse中调试Jar包的源码(调试Struts2源码)
首先在Eclipse中创建一个新的项目,加入运行Struts2所需要的JAR文件,并将它们加到项目的CLASSPATH中(在Lisbs中右击 build path 如下图: ),成功后的界面如图 1- ...
- php-fpm.conf 配置文件详解
php-fpm.conf 配置文件详解 [global] pid = run/php-fpm.pid error_log = log/php-fpm.log log_level = notice # ...
- windows下通过.bat运行java程序
在windows下运行Java项目,单独的jar可以使用,java -jar xxx.jar 运行,如果是一个zip包,里面包含了class文件和所依赖的jar的时候,可以使用 (也可以以看看这里): ...
- 【转】Jmeter基础之——jmeter基础概念
JMeter 介绍:一个非常优秀的开源的性能测试工具. 优点:你用着用着就会发现它的重多优点,当然不足点也会呈现出来. 从性能工具的原理划分: Jmeter工具和其他性能工具在原理上完全一致,工具包含 ...
- java图形用户界面BorderLayout布局。冲突
总结:在使用边界布局发现,把所有的按钮组件都放入了panel.但是在中部的按钮组件找不到了.发现自己重复用了组件 1.this.add(bt4,BorderLayout.North); 2.panel ...
- Spring缓存源码剖析:(一)工具选择
从本篇开始对Spring 4.3.6版本中Cache部分做一次深度剖析.剖析过程中会对其中使用到的设计模式以及原则进行分析.相信对设计内功修炼必定大有好处. 一.环境及工具 IntelliJ IDEA ...
- 【转】分布式存储和一致性hash
本文我将对一致性算法作介绍,同时谈谈自己对一致性hash和一般意义上的hash算法的区别 hash是什么 hash即hash算法,又称为散列算法,百度百科的定义是 哈希算法将任意长度的二进制值映射为较 ...
- python---Redis 学习笔记
缓存 前言: 大家都听过缓存,缓存是干啥的呢?我们可以和json和pickle来说,两个程序之间实现信息交互,可以通过在A程序中把数据改成json ,然后传给B程序,通过文件这个介质.文件这个效率很低 ...
- 十分钟学会Fiddler
一.Fiddler介绍 Fiddler是一个http抓包改包工具,fiddle英文中有“欺骗.伪造”之意,与wireshark相比它更轻量级,上手简单,因为只能抓http和https数据包,所以在针对 ...