欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

基于@RequestBody注解只能注入對象和map的解決

 更新時間:2021年10月22日 14:54:45   作者:WoddenFish  
這篇文章主要介紹了@RequestBody注解只能注入對象和map的解決方案,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教

@RequestBody注解只能注入對象和map的問題

前后端分離開發(fā)模式下,前后端數(shù)據(jù)交互全部采用json,所以在后端在采用spring框架的時候都會使用@ResponseBody(后端返回參數(shù)封裝為json格式)和@RequestBody(前端請求攜帶json參數(shù)解析)注解。

但是在實際開發(fā)中,往往@RequestBody的使用會比較令人難受(超級難受),因為它spring官方只支持到將json解析為一個定義好的對象或者是一個通用性的map,而我們實際項目中經(jīng)常傳遞的參數(shù)僅僅是一個或者是兩個參數(shù),這樣的參數(shù)封裝程對象總是有點大材小用的感覺,并且還消耗性能,而使用map又感覺操作比較繁瑣,那這樣的話可不可以使用簡單的類似于@PathVariable(value="")這樣的模式取請求數(shù)據(jù)呢????當(dāng)然可以?。?/p>

1、自定義一個適應(yīng)于這種情況的注解@RequestJson

package cn.annotation; 
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target; 
/**
 * Title:      RequestJson
 * @date       2018年9月10日
 * @version    V1.0
 * Description: 自定義請求json數(shù)據(jù)解析注解(主要解決單參數(shù)傳遞)
 */
@Target(ElementType.PARAMETER)//使用在參數(shù)上
@Retention(RetentionPolicy.RUNTIME)//運行時注解
public @interface RequestJson {
    /**
     * 是否必須出現(xiàn)的參數(shù)
     */
    boolean required() default true;
 
    /**
     * 當(dāng)value的值或者參數(shù)名不匹配時,是否允許解析最外層屬性到該對象
     */
    boolean parseAllFields() default true;
 
    /**
     * 解析時用到的JSON的key
     */
    String value() default "";
}

2、自定義RequestJsonHandlerMethodArgumentResolver

實現(xiàn)HandlerMethodArgumentResolver(spring解析數(shù)據(jù)的接口)

package cn.config; 
import com.alibaba.fastjson.JSON;
import com.alibaba.fastjson.JSONObject; 
import cn.annotation.RequestJson; 
import org.apache.commons.io.IOUtils;
import org.apache.commons.lang3.StringUtils;
import org.springframework.core.MethodParameter;
import org.springframework.web.bind.support.WebDataBinderFactory;
import org.springframework.web.context.request.NativeWebRequest;
import org.springframework.web.method.support.HandlerMethodArgumentResolver;
import org.springframework.web.method.support.ModelAndViewContainer; 
import javax.servlet.http.HttpServletRequest;
import java.io.IOException;
import java.lang.reflect.Field;
import java.util.HashSet;
import java.util.Set;
 
/**
 * Title:      RequestJsonHandlerMethodArgumentResolver
 * @date       2018年9月10日
 * @version    V1.0
 * Description: 自定義解析json數(shù)據(jù)
 */
public class RequestJsonHandlerMethodArgumentResolver implements HandlerMethodArgumentResolver {
 private static final String JSONBODY_ATTRIBUTE = "JSON_REQUEST_BODY";
  
    /**
     * 設(shè)置支持的方法參數(shù)類型
     * @param parameter 方法參數(shù)
     * @return 支持的類型
     */
    @Override
    public boolean supportsParameter(MethodParameter parameter) {
        // 支持帶@RequestJson注解的參數(shù)
        return parameter.hasParameterAnnotation(RequestJson.class);
    }
 
    /**
     * 參數(shù)解析,利用fastjson
     * 注意:非基本類型返回null會報空指針異常,要通過反射或者JSON工具類創(chuàng)建一個空對象
     */
    @Override
    public Object resolveArgument(MethodParameter parameter, ModelAndViewContainer mavContainer, NativeWebRequest webRequest, WebDataBinderFactory binderFactory) throws Exception {
 
        String jsonBody = getRequestBody(webRequest);
 
        JSONObject jsonObject = JSON.parseObject(jsonBody);
        // 根據(jù)@RequestJson注解value作為json解析的key
        RequestJson parameterAnnotation = parameter.getParameterAnnotation(RequestJson.class);
        //注解的value是JSON的key
        String key = parameterAnnotation.value();
        Object value = null;
        // 如果@RequestJson注解沒有設(shè)置value,則取參數(shù)名FrameworkServlet作為json解析的key
        if (StringUtils.isNotEmpty(key)) {
            value = jsonObject.get(key);
            // 如果設(shè)置了value但是解析不到,報錯
            if (value == null && parameterAnnotation.required()) {
                throw new IllegalArgumentException(String.format("required param %s is not present", key));
            }
        } else {
            // 注解為設(shè)置value則用參數(shù)名當(dāng)做json的key
            key = parameter.getParameterName();
            value = jsonObject.get(key);
        }
 
 
        Class<?> parameterType = parameter.getParameterType();
        // 通過注解的value或者參數(shù)名解析,能拿到value進行解析
        if (value != null) {
            if (isBasicDataTypes(parameterType)) {
                return value;
            }
            return JSON.parseObject(value.toString(), parameterType);
        }
 
        // 解析不到則將整個json串解析為當(dāng)前參數(shù)類型
        if (isBasicDataTypes(parameterType)) {
            if (parameterAnnotation.required()) {
                throw new IllegalArgumentException(String.format("required param %s is not present", key));
            } else {
                return null;
            }
        }
 
        Object result = parameterType.newInstance();
        // 非基本類型,不允許解析所有字段,返回null
        if (!parameterAnnotation.parseAllFields()) {
            // 如果是必傳參數(shù)拋異常
            if (parameterAnnotation.required()) {
                throw new IllegalArgumentException(String.format("required param %s is not present", key));
            }
            // 否則返回空對象
            return result;
        }
       // 非基本類型,允許解析,將外層屬性解析
        result = JSON.parseObject(jsonObject.toString(), parameterType);
        // 如果非必要參數(shù)直接返回,否則如果沒有一個屬性有值則報錯
        if (!parameterAnnotation.required()) {
                return result;
        }else{
            boolean haveValue = false;
            Field[] declaredFields = parameterType.getDeclaredFields();
            for(Field field : declaredFields){
                field.setAccessible(true);
                if(field.get(result) != null){
                    haveValue = true;
                    break;
                }
            }
            if(!haveValue){
                throw new IllegalArgumentException(String.format("required param %s is not present", key));
            }
            return result;
        }
    }
 
    /**
     * 基本數(shù)據(jù)類型直接返回
     */
    @SuppressWarnings("rawtypes")
    private boolean isBasicDataTypes(Class clazz) {
  Set<Class> classSet = new HashSet<>();
        classSet.add(String.class);
        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(Character.class);
        return classSet.contains(clazz);
    }
 
    /**
     * 獲取請求體JSON字符串
     */
    private String getRequestBody(NativeWebRequest webRequest) {
        HttpServletRequest servletRequest = webRequest.getNativeRequest(HttpServletRequest.class);
 
        // 有就直接獲取
        String jsonBody = (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 (IOException e) {
                throw new RuntimeException(e);
            }
        }
        return jsonBody;
 }
}

3、將上述配置應(yīng)用到spring項目中

重寫addArgumentResolvers方法

package cn.config; 
import java.nio.charset.Charset;
import java.util.List; 
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.http.converter.HttpMessageConverter;
import org.springframework.http.converter.StringHttpMessageConverter;
import org.springframework.web.method.support.HandlerMethodArgumentResolver;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurerAdapter;
 
/**
 * Title:      WebConfig
 * @date       2018年9月10日
 * @version    V1.0
 * Description: 將自定義注解配置到spring
 */
 
@Configuration
@SuppressWarnings("deprecation")
public class WebConfig extends WebMvcConfigurerAdapter{
 @Override
    public void addArgumentResolvers(List<HandlerMethodArgumentResolver> argumentResolvers) {
        argumentResolvers.add(new RequestJsonHandlerMethodArgumentResolver());
    }
 
    @Bean
    public HttpMessageConverter<String> responseBodyConverter() {
        return new StringHttpMessageConverter(Charset.forName("UTF-8"));
    }
 
    @Override
    public void configureMessageConverters(List<HttpMessageConverter<?>> converters) {
        super.configureMessageConverters(converters);
        converters.add(responseBodyConverter());
    }
}

4、配置完成了,簡單使用

單參數(shù):(不加value默認為參數(shù)名)

或者:

多參數(shù):

@RequestBody注解的使用問題

今天遇到的問題:@RequestBody的使用問題

先看一下@RequestBody的作用

在這里插入圖片描述

我想獲取json字符串某個字段值,看截圖:

在這里插入圖片描述

看一下控制臺的輸出信息:

在這里插入圖片描述

what ?這什么情況,為什么拿到的是整個json字符串,然后我繼續(xù)測試

在這里插入圖片描述

給了一個400

在這里插入圖片描述

what ?這又是什么情況 (好像只能有一個@RequestBody)我想?yún)?shù)如果是整形的話能不能獲取,我繼續(xù)進行測試代碼:

在這里插入圖片描述

傳參:

在這里插入圖片描述

又給了一個400 (好像只能是String類型) 測試引用類型對象

在這里插入圖片描述

代碼:

在這里插入圖片描述

傳參:

在這里插入圖片描述

控制臺打?。?/p>

在這里插入圖片描述

測試成功。

個人總結(jié):

1) 一個方法只能有一個@RequestBody

2) 如果接收參數(shù)是字符串類型的,獲取的是整個json字符串

3) 如果接受的參數(shù)是引用對象,@requestBody User user 會將json字符串中的值賦予user中對應(yīng)的屬性上

需要注意的是,json字符串中key必須和User對象的屬性名對應(yīng)

以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。

相關(guān)文章

  • Hibernate中獲取Session的兩種方式代碼示例

    Hibernate中獲取Session的兩種方式代碼示例

    這篇文章主要介紹了Hibernate中獲取Session的兩種方式代碼示例,具有一定借鑒價值,需要的朋友可以參考下。
    2017-12-12
  • Spring高階用法之自定義業(yè)務(wù)對象組件化

    Spring高階用法之自定義業(yè)務(wù)對象組件化

    這篇文章主要介紹了Spring高階用法之自定義業(yè)務(wù)對象組件化,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-03-03
  • Springboot注解之@EnableAutoConfiguration詳解

    Springboot注解之@EnableAutoConfiguration詳解

    這篇文章主要介紹了Springboot注解之@EnableAutoConfiguration詳解,@EnableAutoConfiguration是一個加載Starter目錄包之外的需要Spring自動生成bean對象,本文對其進行總結(jié),需要的朋友可以參考下
    2023-08-08
  • java實現(xiàn)將域名解析成ip示例

    java實現(xiàn)將域名解析成ip示例

    這篇文章主要介紹了java實現(xiàn)將域名解析成ip示例,需要的朋友可以參考下
    2014-04-04
  • Spring計時器StopWatch的具體使用

    Spring計時器StopWatch的具體使用

    本文主要介紹了Spring計時器StopWatch的具體使用,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-06-06
  • SpringMVC攔截器和異常處理器使用示例超詳細講解

    SpringMVC攔截器和異常處理器使用示例超詳細講解

    攔截器(Interceptor)是一種動態(tài)攔截方法調(diào)用的機制,在SpringMVC中動態(tài)攔截控制器方法的執(zhí)行。本文將詳細講講SpringMVC中攔截器參數(shù)及攔截器鏈配置,感興趣的可以嘗試一下
    2022-09-09
  • Java 自定義Spring框架與Spring IoC相關(guān)接口分析

    Java 自定義Spring框架與Spring IoC相關(guān)接口分析

    Spring框架是由于軟件開發(fā)的復(fù)雜性而創(chuàng)建的。Spring使用的是基本的JavaBean來完成以前只可能由EJB完成的事情。然而,Spring的用途不僅僅限于服務(wù)器端的開發(fā)
    2021-10-10
  • 詳解Java設(shè)計模式之橋接模式

    詳解Java設(shè)計模式之橋接模式

    橋接,顧名思義,就是用來連接兩個部分,使得兩個部分可以互相通訊。橋接模式將系統(tǒng)的抽象部分與實現(xiàn)部分分離解耦,使他們可以獨立的變化。本文通過示例詳細介紹了橋接模式的原理與使用,需要的可以參考一下
    2022-06-06
  • JAVA內(nèi)存模型(JMM)詳解

    JAVA內(nèi)存模型(JMM)詳解

    這篇文章主要介紹了JAVA內(nèi)存模型(JMM)詳解的相關(guān)資料,需要的朋友可以參考下
    2022-12-12
  • Java8中使用一行代碼讀取文件

    Java8中使用一行代碼讀取文件

    這篇文章主要介紹了Java8中使用一行代碼讀取文件,要注意,本文介紹的方法不適合讀取很大的文件,因為可能存在內(nèi)存空間不足的問題,需要的朋友可以參考下
    2015-03-03

最新評論