对前端传入的json对象解析成多个对象

本文涉及的产品
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
简介: 如果是多个对象呢?那怎么解决这个问题?此时就可以用到:multiRequestBodyDemo(@MultiRequestBody("dog")Dog dog, @MultiRequestBody("user") User user)这种方式进行接收了。但spring boot是不支持这种方式的。因此,就需要自己写一个解析器来解析这样的传入方式和接收的方式。通常,比如我们有分页和对象时,就可以采用这种方式进行接收。

最近项目中类似:

multiRequestBodyDemo(@MultiRequestBody("dog")

Dog dog, @MultiRequestBody("user") User user)

的注解看到了很多,那么这些注解是Spring MVC自带的吗?当然不是,spring MVC中自带的是@RequestBody的注解,这个注解有什么作用呢?这个注解可以将前端传进来的json数据进行解析成json数据。而如果我们没有采样@MultiRequestBody时,通常的做法是将其首先转成json首先转成json,然后进行json数据解析,然后对相关的属性进行逐一获取。下面的例子来源于网上,同时也是我们常用的解析方式之一。但是,如果我们获取属性过多,必然就会带来一个问题,对于代码会显得很长,不够优雅。

那还有一种方式那就是采样对象去接收,但是对象接收,但是如果是多个对象呢?那怎么解决这个问题?

此时就可以用到:

multiRequestBodyDemo(@MultiRequestBody("dog")

Dog dog, @MultiRequestBody("user") User user)

这种方式进行接收了。但spring boot是不支持这种方式的。因此,就需要自己写一个解析器来解析这样的传入方式和接收的方式。通常,比如我们有分页和对象时,就可以采用这种方式进行接收。

/*** 修改或者新增热门搜索* @param hotSearch* @param request* @return*/@RequestMapping(value="/servehotselectiveajax")
@ResponseBodypublicintservehotselectiveajax(HttpServletRequestreq,HttpServletResponseresp,@RequestBodyJSONObjectobj) {
intcount=0;
intcountAddHotSearch=0;
intcountEditHotSearch=0;
LOGGER.info("data:"+obj.toJSONString());
//data:{"createArr":[{"hotSearchId":"","keyword":"ss","sort":"5","tempid":"21"}],"modifyArr":[{"hotSearchId":"205","keyword":"华为","sort":"2","tempid":"21"},{"hotSearchId":"206","keyword":"游戏本","sort":"3","tempid":"21"},{"hotSearchId":"207","keyword":"平板电视","sort":"3","tempid":"21"},{"hotSearchId":"208","keyword":"连衣裙","sort":"4","tempid":"21"}]}Stringdata=obj.toJSONString();
//解析json数据JSONObjectjson=JSON.parseObject(data);
StringcreateArr=json.getString("createArr");
StringmodifyArr=json.getString("modifyArr");
if(StringUtils.isNotEmpty(createArr)){
JSONArraycreateArray=JSONArray.parseArray(createArr);
for(inti=0;i<createArray.size();i++){
LongtempId=JSONObject.parseObject(JSONObject.toJSONString(createArray.get(i))).getLong("tempId");
Stringkeyword=JSONObject.parseObject(JSONObject.toJSONString(createArray.get(i))).getString("keyword");
Integersort=JSONObject.parseObject(JSONObject.toJSONString(createArray.get(i))).getInteger("sort");
//创建热门搜索对象HotSearchhotSearch=newHotSearch();
hotSearch.setTempid(tempId);
hotSearch.setKeyword(keyword);
hotSearch.setSort(sort);
hotSearch.setCreateDate(newDate());
hotSearch.setDelFlag("0");
//添加热门搜索信息countAddHotSearch=hotSearchService.addHotSearchSelective(hotSearch);       
            }
        }
if(StringUtils.isNotEmpty(modifyArr)){
JSONArraymodifyArray=JSONArray.parseArray(modifyArr);
for(inti=0;i<modifyArray.size();i++){
LonghotSearchId=JSONObject.parseObject(JSONObject.toJSONString(modifyArray.get(i))).getLong("id");
LongtempId=JSONObject.parseObject(JSONObject.toJSONString(modifyArray.get(i))).getLong("tempId");
Stringkeyword=JSONObject.parseObject(JSONObject.toJSONString(modifyArray.get(i))).getString("keyword");
Integersort=JSONObject.parseObject(JSONObject.toJSONString(modifyArray.get(i))).getInteger("sort");
//创建热门搜索对象HotSearchhotSearch=newHotSearch();
hotSearch.setHotSearchId(hotSearchId);
hotSearch.setTempid(tempId);
hotSearch.setKeyword(keyword);
hotSearch.setSort(sort);
//修改热门搜索信息countEditHotSearch=hotSearchService.modifyHostSearchSelectiveById(hotSearch);   
            }
        }
//判断修改或者新增成功if(countAddHotSearch>0||countEditHotSearch>0){
count=1;
        }
returncount;
    }

MultiRequestBody解析器

解决的问题:

1、单个字符串等包装类型都要写一个对象才可以用@RequestBody接收;2、多个对象需要封装到一个对象里才可以用@RequestBody接收。主要优势:1、支持通过注解的value指定JSON的key来解析对象。2、支持通过注解无value,直接根据参数名来解析对象3、支持基本类型的注入4、支持GET和其他请求方式注入5、支持通过注解无value且参数名不匹配JSON串key时,根据属性解析对象。6、支持多余属性(不解析、不报错)、支持参数“共用”(不指定value时,参数名不为JSON串的key)7、支持当value和属性名找不到匹配的key时,对象是否匹配所有属性。

其思路是将前端传入的数据进行获取,也即jsonbody

, 获取请求体JSON字符串。获取之后,将其转成jsonObject。获取自定义元注解@MultiRequestBody中的value,如果@MultiRequestBody注解没有设置value,则取参数名FrameworkServlet作为json解析的key。默认是没有的,因此需要获取controller中的注解value值信息作为key,然后通过key拿到属性信息。进行相应的解析。下面是作者 Wangyang Liu的解析具体实现,具体代码可以到github上去获取,https://github.com/chujianyun/Spring-MultiRequestBody

/*** MultiRequestBody解析器* 解决的问题:* 1、单个字符串等包装类型都要写一个对象才可以用@RequestBody接收;* 2、多个对象需要封装到一个对象里才可以用@RequestBody接收。* 主要优势:* 1、支持通过注解的value指定JSON的key来解析对象。* 2、支持通过注解无value,直接根据参数名来解析对象* 3、支持基本类型的注入* 4、支持GET和其他请求方式注入* 5、支持通过注解无value且参数名不匹配JSON串key时,根据属性解析对象。* 6、支持多余属性(不解析、不报错)、支持参数“共用”(不指定value时,参数名不为JSON串的key)* 7、支持当value和属性名找不到匹配的key时,对象是否匹配所有属性。** @author Wangyang Liu  QQ: 605283073* @date 2018/08/27*/publicclassMultiRequestBodyArgumentResolverimplementsHandlerMethodArgumentResolver {
privatestaticfinalStringJSONBODY_ATTRIBUTE="JSON_REQUEST_BODY";
/*** 设置支持的方法参数类型** @param parameter 方法参数* @return 支持的类型*/@OverridepublicbooleansupportsParameter(MethodParameterparameter) {
// 支持带@MultiRequestBody注解的参数returnparameter.hasParameterAnnotation(MultiRequestBody.class);
    }
/*** 参数解析,利用fastjson* 注意:非基本类型返回null会报空指针异常,要通过反射或者JSON工具类创建一个空对象*/@OverridepublicObjectresolveArgument(MethodParameterparameter, ModelAndViewContainermavContainer, NativeWebRequestwebRequest, WebDataBinderFactorybinderFactory) throwsException {
StringjsonBody=getRequestBody(webRequest);
JSONObjectjsonObject=JSON.parseObject(jsonBody);
// 根据@MultiRequestBody注解value作为json解析的keyMultiRequestBodyparameterAnnotation=parameter.getParameterAnnotation(MultiRequestBody.class);
//注解的value是JSON的keyStringkey=parameterAnnotation.value();
Objectvalue;
// 如果@MultiRequestBody注解没有设置value,则取参数名FrameworkServlet作为json解析的keyif (StringUtils.isNotEmpty(key)) {
value=jsonObject.get(key);
// 如果设置了value但是解析不到,报错if (value==null&&parameterAnnotation.required()) {
thrownewIllegalArgumentException(String.format("required param %s is not present", key));
            }
        } else {
// 注解为设置value则用参数名当做json的keykey=parameter.getParameterName();
value=jsonObject.get(key);
        }
// 获取的注解后的类型 LongClass<?>parameterType=parameter.getParameterType();
// 通过注解的value或者参数名解析,能拿到value进行解析if (value!=null) {
//基本类型if (parameterType.isPrimitive()) {
returnparsePrimitive(parameterType.getName(), value);
            }
// 基本类型包装类if (isBasicDataTypes(parameterType)) {
returnparseBasicTypeWrapper(parameterType, value);
// 字符串类型            } elseif (parameterType==String.class) {
returnvalue.toString();
            }
// 其他复杂对象returnJSON.parseObject(value.toString(), parameterType);
        }
// 解析不到则将整个json串解析为当前参数类型if (isBasicDataTypes(parameterType)) {
if (parameterAnnotation.required()) {
thrownewIllegalArgumentException(String.format("required param %s is not present", key));
            } else {
returnnull;
            }
        }
// 非基本类型,不允许解析所有字段,必备参数则报错,非必备参数则返回nullif (!parameterAnnotation.parseAllFields()) {
// 如果是必传参数抛异常if (parameterAnnotation.required()) {
thrownewIllegalArgumentException(String.format("required param %s is not present", key));
            }
// 否则返回nullreturnnull;
        }
// 非基本类型,允许解析,将外层属性解析Objectresult;
try {
result=JSON.parseObject(jsonObject.toString(), parameterType);
        } catch (JSONExceptionjsonException) {
// TODO:: 异常处理返回null是否合理?result=null;
        }
// 如果非必要参数直接返回,否则如果没有一个属性有值则报错if (!parameterAnnotation.required()) {
returnresult;
        } else {
booleanhaveValue=false;
Field[] declaredFields=parameterType.getDeclaredFields();
for (Fieldfield : declaredFields) {
field.setAccessible(true);
if (field.get(result) !=null) {
haveValue=true;
break;
                }
            }
if (!haveValue) {
thrownewIllegalArgumentException(String.format("required param %s is not present", key));
            }
returnresult;
        }
    }
/*** 基本类型解析*/privateObjectparsePrimitive(StringparameterTypeName, Objectvalue) {
finalStringbooleanTypeName="boolean";
if (booleanTypeName.equals(parameterTypeName)) {
returnBoolean.valueOf(value.toString());
        }
finalStringintTypeName="int";
if (intTypeName.equals(parameterTypeName)) {
returnInteger.valueOf(value.toString());
        }
finalStringcharTypeName="char";
if (charTypeName.equals(parameterTypeName)) {
returnvalue.toString().charAt(0);
        }
finalStringshortTypeName="short";
if (shortTypeName.equals(parameterTypeName)) {
returnShort.valueOf(value.toString());
        }
finalStringlongTypeName="long";
if (longTypeName.equals(parameterTypeName)) {
returnLong.valueOf(value.toString());
        }
finalStringfloatTypeName="float";
if (floatTypeName.equals(parameterTypeName)) {
returnFloat.valueOf(value.toString());
        }
finalStringdoubleTypeName="double";
if (doubleTypeName.equals(parameterTypeName)) {
returnDouble.valueOf(value.toString());
        }
finalStringbyteTypeName="byte";
if (byteTypeName.equals(parameterTypeName)) {
returnByte.valueOf(value.toString());
        }
returnnull;
    }
/*** 基本类型包装类解析*/privateObjectparseBasicTypeWrapper(Class<?>parameterType, Objectvalue) {
if (Number.class.isAssignableFrom(parameterType)) {
Numbernumber= (Number) value;
if (parameterType==Integer.class) {
returnnumber.intValue();
            } elseif (parameterType==Short.class) {
returnnumber.shortValue();
            } elseif (parameterType==Long.class) {
returnnumber.longValue();
            } elseif (parameterType==Float.class) {
returnnumber.floatValue();
            } elseif (parameterType==Double.class) {
returnnumber.doubleValue();
            } elseif (parameterType==Byte.class) {
returnnumber.byteValue();
            }
        } elseif (parameterType==Boolean.class) {
returnvalue.toString();
        } elseif (parameterType==Character.class) {
returnvalue.toString().charAt(0);
        }
returnnull;
    }
/*** 判断是否为基本数据类型包装类*/privatebooleanisBasicDataTypes(Classclazz) {
Set<Class>classSet=newHashSet<>();
classSet.add(Integer.class);
classSet.add(Long.class);
classSet.add(Short.class);
classSet.add(Float.class);
classSet.add(Double.class);
classSet.add(Boolean.class);
classSet.add(Byte.class);
classSet.add(Character.class);
returnclassSet.contains(clazz);
    }
/*** 获取请求体JSON字符串*/privateStringgetRequestBody(NativeWebRequestwebRequest) {
HttpServletRequestservletRequest=webRequest.getNativeRequest(HttpServletRequest.class);
// 有就直接获取StringjsonBody= (String) webRequest.getAttribute(JSONBODY_ATTRIBUTE, NativeWebRequest.SCOPE_REQUEST);
// 没有就从请求中读取if (jsonBody==null) {
try {
jsonBody=IOUtils.toString(servletRequest.getReader());
webRequest.setAttribute(JSONBODY_ATTRIBUTE, jsonBody, NativeWebRequest.SCOPE_REQUEST);
            } catch (IOExceptione) {
thrownewRuntimeException(e);
            }
        }
returnjsonBody;
    }
}

当然可以:


微信图片_20221214020703.jpg

还可以:

微信图片_20221214020710.jpg

还可以:

微信图片_20221214020713.jpg

从测试的结果来看,都可以很方便的转成对应的对象信息,方便使用!


目录
相关文章
|
3月前
|
SQL 存储 JSON
SQL,解析 json
SQL,解析 json
84 8
|
4天前
|
JSON JavaScript 前端开发
一次采集JSON解析错误的修复
两段采集来的JSON格式数据存在格式问题,直接使用PHP的`json_decode`会报错。解决思路包括:1) 手动格式化并逐行排查错误;2) 使用PHP-V8JS扩展在JavaScript环境中解析。具体方案一是通过正则表达式和字符串替换修复格式,方案二是利用V8Js引擎执行JS代码并返回JSON字符串,最终实现正确解析。 简介: 两段采集的JSON数据因掺杂JavaScript代码导致PHP解析失败。解决方案包括手动格式化修复和使用PHP-V8JS扩展在JavaScript环境中解析,确保JSON数据能被正确处理。
|
2月前
|
存储 前端开发 JavaScript
前端中对象的深度应用与最佳实践
前端对象应用涉及在网页开发中使用JavaScript等技术创建和操作对象,以实现动态交互效果。通过定义属性和方法,对象可以封装数据和功能,提升代码的组织性和复用性,是现代Web开发的核心技术之一。
|
2月前
|
SQL Java 数据库连接
canal-starter 监听解析 storeValue 不一样,同样的sql 一个在mybatis执行 一个在数据库操作,导致解析不出正确对象
canal-starter 监听解析 storeValue 不一样,同样的sql 一个在mybatis执行 一个在数据库操作,导致解析不出正确对象
|
3月前
|
JSON 前端开发 数据格式
前端的全栈之路Meteor篇(五):自定义对象序列化的EJSON介绍 - 跨设备的对象传输
EJSON是Meteor框架中扩展了标准JSON的库,支持更多数据类型如`Date`、`Binary`等。它提供了序列化和反序列化功能,使客户端和服务器之间的复杂数据传输更加便捷高效。EJSON还支持自定义对象的定义和传输,通过`EJSON.addType`注册自定义类型,确保数据在两端无缝传递。
|
3月前
|
Python
深入解析 Python 中的对象创建与初始化:__new__ 与 __init__ 方法
深入解析 Python 中的对象创建与初始化:__new__ 与 __init__ 方法
27 1
|
3月前
|
JSON JavaScript API
商品详情数据接口解析返回的JSON数据(API接口整套流程)
商品详情数据接口解析返回的JSON数据是API接口使用中的一个重要环节,它涉及从发送请求到接收并处理响应的整个流程。以下是一个完整的API接口使用流程,包括如何解析返回的JSON数据:
|
2月前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
87 2
|
10天前
|
存储 设计模式 算法
【23种设计模式·全精解析 | 行为型模式篇】11种行为型模式的结构概述、案例实现、优缺点、扩展对比、使用场景、源码解析
行为型模式用于描述程序在运行时复杂的流程控制,即描述多个类或对象之间怎样相互协作共同完成单个对象都无法单独完成的任务,它涉及算法与对象间职责的分配。行为型模式分为类行为模式和对象行为模式,前者采用继承机制来在类间分派行为,后者采用组合或聚合在对象间分配行为。由于组合关系或聚合关系比继承关系耦合度低,满足“合成复用原则”,所以对象行为模式比类行为模式具有更大的灵活性。 行为型模式分为: • 模板方法模式 • 策略模式 • 命令模式 • 职责链模式 • 状态模式 • 观察者模式 • 中介者模式 • 迭代器模式 • 访问者模式 • 备忘录模式 • 解释器模式
【23种设计模式·全精解析 | 行为型模式篇】11种行为型模式的结构概述、案例实现、优缺点、扩展对比、使用场景、源码解析
|
10天前
|
设计模式 存储 安全
【23种设计模式·全精解析 | 创建型模式篇】5种创建型模式的结构概述、实现、优缺点、扩展、使用场景、源码解析
结构型模式描述如何将类或对象按某种布局组成更大的结构。它分为类结构型模式和对象结构型模式,前者采用继承机制来组织接口和类,后者釆用组合或聚合来组合对象。由于组合关系或聚合关系比继承关系耦合度低,满足“合成复用原则”,所以对象结构型模式比类结构型模式具有更大的灵活性。 结构型模式分为以下 7 种: • 代理模式 • 适配器模式 • 装饰者模式 • 桥接模式 • 外观模式 • 组合模式 • 享元模式
【23种设计模式·全精解析 | 创建型模式篇】5种创建型模式的结构概述、实现、优缺点、扩展、使用场景、源码解析

热门文章

最新文章

推荐镜像

更多