使用SpringBoot通過自定義注解+AOP+全局異常處理實現(xiàn)參數(shù)統(tǒng)一非空校驗

一树灶、前言

????????在我們寫后臺接口時,難免對參數(shù)進(jìn)行非空校驗糯而,如果一兩個還好天通,但如果需要寫大量的接口,及必填參數(shù)太多的時候熄驼,會給我們開發(fā)帶來大量的重復(fù)工作像寒,及很多相似代碼。而sping自帶的@RequestParam注解并不能完全滿足我們的需求瓜贾,因為這個注解只會校驗請求中是否存在該參數(shù)诺祸,而不會校驗這個參數(shù)的值是nulll還是空字符串(""),如果參數(shù)不存在則會拋出org.springframework.web.bind.MissingServletRequestParameterException異常祭芦。雖然目前已經(jīng)有許多成熟的校驗框架筷笨,功能豐富,但是我們只需要做一個非空校驗即可龟劲。

????????因此我們可以自定義 一個注解用于校驗參數(shù)是否為空胃夏。

使用的的框架

  • spring boot:1.5.9.RELEASE
  • JDK:1.8

二、準(zhǔn)備工作

????????首先需要創(chuàng)建一個spring boot項目昌跌,并引入相關(guān)maven依賴(主要是spring-boot-starter-web與aspectjweaver),pom文件如下:

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <groupId>com.beauxie</groupId>
    <artifactId>param-demo</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <packaging>jar</packaging>

    <name>param-demo</name>
    <description>param-demo for Spring Boot</description>

    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>1.5.9.RELEASE</version>
        <relativePath/> <!-- lookup parent from repository -->
    </parent>

    <properties>
        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
        <project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>
        <java.version>1.8</java.version>
    </properties>

    <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter</artifactId>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>

        <!-- 添加支持web -->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <!--引入AOP相應(yīng)的注解-->
        <dependency>
            <groupId>org.aspectj</groupId>
            <artifactId>aspectjweaver</artifactId>
            <version>1.8.5</version>
        </dependency>
    </dependencies>

    <build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
            </plugin>
        </plugins>
    </build>

</project>

說明:

  • spring-boot-starter-web用于spring boot WEB支持
  • aspectjweaver 用于引入aop的相關(guān)的注解仰禀,如@Aspect@Pointcut

三蚕愤、自定義注解實現(xiàn)統(tǒng)一校驗

總體思路:自定義一個注解答恶,對必填的參數(shù)加上該注解囊榜,然后定義一個切面,校驗該參數(shù)是否為空亥宿,如果為空則拋出自定義的異常,該異常被自定義的異常處理器捕獲砂沛,然后返回相應(yīng)的錯誤信息烫扼。

1. 自定義注解

創(chuàng)建一個名為'ParamCheck'的注解,代碼如下:

package com.beauxie.param.demo.annotation;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

/**
 * "參數(shù)不能為空"注解碍庵,作用于方法參數(shù)上映企。
 * 
 * @author Beauxie
 * @date Created on 2017/1/6
 */
@Target(ElementType.PARAMETER)
@Retention(RetentionPolicy.RUNTIME)
public @interface ParamCheck {
    /**
     * 是否非空,默認(rèn)不能為空
     */
    boolean notNull() default true;
}

說明:

  • @Target(ElementType.PARAMETER)表示該注解作用于方法參數(shù)上
  • 該類可以拓展,比如增加length校驗

2. 自定義異常類

這個異常類與自定義注解配合一起使用静浴,當(dāng)加上'@ParamCheck'的參數(shù)為空時堰氓,拋出該異常,代碼如下:

package com.beauxie.param.demo.exception;

/**
 * @author Beauxie
 * @date Created on 2017/1/6
 */
public class ParamIsNullException extends RuntimeException {
    private final String parameterName;
    private final String parameterType;

    public ParamIsNullException(String parameterName, String parameterType) {
        super("");
        this.parameterName = parameterName;
        this.parameterType = parameterType;
    }

    @Override
    public String getMessage() {
        return "Required " + this.parameterType + " parameter \'" + this.parameterName + "\' must be not null !";
    }

    public final String getParameterName() {
        return this.parameterName;
    }

    public final String getParameterType() {
        return this.parameterType;
    }
}

說明:

  • 該異常繼承RuntimeException苹享,并定義了兩個成員屬性双絮、重寫了getMessage()方法
  • 之所以自定義該異常,而不用現(xiàn)有的org.springframework.web.bind.MissingServletRequestParameterException類得问,是因為MissingServletRequestParameterException為Checked異常,在動態(tài)代理過程中囤攀,很容易引發(fā)java.lang.reflect.UndeclaredThrowableException異常。

3. 自定義AOP

代碼如下:

package com.beauxie.param.demo.aop;

import com.beauxie.param.demo.annotation.ParamCheck;
import com.beauxie.param.demo.exception.ParamIsNullException;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.*;
import org.aspectj.lang.reflect.MethodSignature;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Component;

import java.lang.annotation.Annotation;
import java.lang.reflect.Method;

/**
 * @author Beauxie
 * @date Created on 2017/1/6
 */
@Component
@Aspect
public class ParamCheckAop {
    private Logger logger = LoggerFactory.getLogger(this.getClass());


    /**
     * 定義有一個切入點宫纬,范圍為web包下的類
     */
    @Pointcut("execution(public * com.beauxie.param.demo.web..*.*(..))")
    public void checkParam() {
    }

    @Before("checkParam()")
    public void doBefore(JoinPoint joinPoint) {
    }

    /**
     * 檢查參數(shù)是否為空
     */
    @Around("checkParam()")
    public Object doAround(ProceedingJoinPoint pjp) throws Throwable {

        MethodSignature signature = ((MethodSignature) pjp.getSignature());
        //得到攔截的方法
        Method method = signature.getMethod();
        //獲取方法參數(shù)注解焚挠,返回二維數(shù)組是因為某些參數(shù)可能存在多個注解
        Annotation[][] parameterAnnotations = method.getParameterAnnotations();
        if (parameterAnnotations == null || parameterAnnotations.length == 0) {
            return pjp.proceed();
        }
        //獲取方法參數(shù)名
        String[] paramNames = signature.getParameterNames();
        //獲取參數(shù)值
        Object[] paranValues = pjp.getArgs();
        //獲取方法參數(shù)類型
        Class<?>[] parameterTypes = method.getParameterTypes();
        for (int i = 0; i < parameterAnnotations.length; i++) {
            for (int j = 0; j < parameterAnnotations[i].length; j++) {
                //如果該參數(shù)前面的注解是ParamCheck的實例,并且notNull()=true,則進(jìn)行非空校驗
                if (parameterAnnotations[i][j] != null && parameterAnnotations[i][j] instanceof ParamCheck && ((ParamCheck) parameterAnnotations[i][j]).notNull()) {
                    paramIsNull(paramNames[i], paranValues[i], parameterTypes[i] == null ? null : parameterTypes[i].getName());
                    break;
                }
            }
        }
        return pjp.proceed();
    }

    /**
     * 在切入點return內(nèi)容之后切入內(nèi)容(可以用來對處理返回值做一些加工處理)
     *
     * @param joinPoint
     */
    @AfterReturning("checkParam()")
    public void doAfterReturning(JoinPoint joinPoint) {
    }

    /**
     * 參數(shù)非空校驗漓骚,如果參數(shù)為空蝌衔,則拋出ParamIsNullException異常
     * @param paramName
     * @param value
     * @param parameterType
     */
    private void paramIsNull(String paramName, Object value, String parameterType) {
        if (value == null || "".equals(value.toString().trim())) {
            throw new ParamIsNullException(paramName, parameterType);
        }
    }

}

4. 全局異常處理器

該異常處理器捕獲在ParamCheckAop類中拋出的ParamIsNullException異常,并進(jìn)行處理,代碼如下:

package com.beauxie.param.demo.exception;

import com.beauxie.param.demo.common.Result;
import com.beauxie.param.demo.enums.EnumResultCode;
import com.beauxie.param.demo.utils.ResponseMsgUtil;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.web.bind.MissingServletRequestParameterException;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.RestControllerAdvice;

import javax.servlet.http.HttpServletRequest;

/**
 * 全局異常處理.
 * 一般情況下蝌蹂,方法都有異常處理機(jī)制噩斟,但不能排除有個別異常沒有處理,導(dǎo)致返回到前臺叉信,因此在這里做一個異常攔截亩冬,統(tǒng)一處理那些未被處理過的異常
 *
 * @author Beauxie
 * @date Created on 2017/1/6
 */
@RestControllerAdvice
public class GlobalExceptionHandler {
    private static final Logger LOGGER = LoggerFactory.getLogger(GlobalExceptionHandler.class);


    /**
     * 參數(shù)為空異常處理
     *
     * @param ex
     * @return
     */
    @ExceptionHandler({MissingServletRequestParameterException.class, ParamIsNullException.class})
    public Result<String> requestMissingServletRequest(Exception ex) {
        LOGGER.error("request Exception:", ex);
        return ResponseMsgUtil.builderResponse(EnumResultCode.FAIL.getCode(), ex.getMessage(), null);
    }

    /**
     * 特別說明: 可以配置指定的異常處理,這里處理所有
     *
     * @param request
     * @param e
     * @return
     */
    @ExceptionHandler(value = Exception.class)
    public Result<String> errorHandler(HttpServletRequest request, Exception e) {
        LOGGER.error("request Exception:", e);
        return ResponseMsgUtil.exception();
    }
}

說明:

  • requestMissingServletRequest()方法上加上@ExceptionHandler({MissingServletRequestParameterException.class, ParamIsNullException.class})注解,表明只處理處理MissingServletRequestParameterExceptionParamIsNullException異常
  • errorHandler()方法則處理其他的異常

四硼身、測試

com.beauxie.param.demo.web包下新建一個名為HelloController的類,用于測試,代碼如下:

package com.beauxie.param.demo.web;

import com.beauxie.param.demo.annotation.ParamCheck;
import com.beauxie.param.demo.common.Result;
import com.beauxie.param.demo.enums.EnumResultCode;
import com.beauxie.param.demo.utils.ResponseMsgUtil;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.RestController;

/**
 * @author Beauxie
 * @date Created on 2018/1/6
 */
@RestController
public class HelloController {
    /**
     *測試@RequestParam注解
     * @param name
     * @return
     */
    @GetMapping("/hello1")
    public Result<String> hello1(@RequestParam String name) {
        return ResponseMsgUtil.builderResponse(EnumResultCode.SUCCESS.getCode(), "請求成功", "Hello," + name);
    }

    /**
     * 測試@ParamCheck注解
     * @param name
     * @return
     */
    @GetMapping("/hello2")
    public Result<String> hello2(@ParamCheck String name) {
        return ResponseMsgUtil.builderResponse(EnumResultCode.SUCCESS.getCode(), "請求成功", "Hello," + name);
    }

    /**
     * 測試@ParamCheck與@RequestParam一起時
     * @param name
     * @return
     */
    @GetMapping("/hello3")
    public Result<String> hello3(@ParamCheck @RequestParam String name) {
        return ResponseMsgUtil.builderResponse(EnumResultCode.SUCCESS.getCode(), "請求成功", "Hello," + name);
    }

最后運(yùn)行ParamDemoApplicatio的main方法,打開瀏覽器進(jìn)行測試硅急。

1. 測試@RequestParam注解

  • 參數(shù)名為空測試

    在瀏覽器的地址欄輸入:http://localhost:8080/hello1,結(jié)果如下:

    RequestParam1

    后臺錯誤信息輸出:
    Error1

    此時后臺會報org.springframework.web.bind.MissingServletRequestParameterException: Required String parameter 'name' is not present錯誤信息佳遂,提示參數(shù)'name'不存在

  • 參數(shù)名不為空营袜,值為空測試

    在瀏覽器的地址欄輸入:http://localhost:8080/hello1?name=,結(jié)果如下:

    RequestParam2


    此時丑罪,name的值為空荚板,但請求結(jié)果正常返回凤壁。

  • 參數(shù)名與值都不為空測試

    在瀏覽器的地址欄輸入:http://localhost:8080/hello1?name=Beauxie,結(jié)果如下:

    RequestParam3

2. 測試@ParamCheck注解

  • 參數(shù)名為空測試

    在瀏覽器的地址欄輸入:http://localhost:8080/hello2跪另,結(jié)果如下:
    [圖片上傳失敗...(image-f11449-1515228018125)]
    后臺錯誤信息輸出:

    Error2

  • 參數(shù)名不為空拧抖,值為空測試

    在瀏覽器的地址欄輸入:http://localhost:8080/hello2?name=,結(jié)果如下:
    [圖片上傳失敗...(image-8d0648-1515228018125)]
    此時免绿,name的值為空唧席,請求結(jié)果y提示參數(shù)name的值不能為空。
    后臺錯誤信息輸出:

    Error3

  • 參數(shù)名與值都不為空測試

    在瀏覽器的地址欄輸入:http://localhost:8080/hello2?name=Beauxie嘲驾,結(jié)果如下:

    ParamCheck3

3. 測試總結(jié)

  • 當(dāng)參數(shù)名為空時淌哟,分別添加兩個注解的接口都會提示參數(shù)不能為空
  • 當(dāng)參數(shù)名不為空,值為空時辽故,@RequestParam注解不會報錯徒仓,但@ParamCheck注解提示參數(shù)'name'的值為空

五、總結(jié)

  • 經(jīng)過以上的測試也驗證了@RequestParam只會驗證對應(yīng)的參數(shù)是否存在誊垢,而不會驗證值是否為空
  • ParamCheck還可以進(jìn)行拓展掉弛,比如參數(shù)值長度、是否含有非法字符等校驗





六彤枢、源碼下載地址

由于csdn下載需要積分狰晚,因此添加github源碼地址:

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市缴啡,隨后出現(xiàn)的幾起案子壁晒,更是在濱河造成了極大的恐慌,老刑警劉巖业栅,帶你破解...
    沈念sama閱讀 206,311評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件秒咐,死亡現(xiàn)場離奇詭異,居然都是意外死亡碘裕,警方通過查閱死者的電腦和手機(jī)携取,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,339評論 2 382
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來帮孔,“玉大人雷滋,你說我怎么就攤上這事∥木ぃ” “怎么了晤斩?”我有些...
    開封第一講書人閱讀 152,671評論 0 342
  • 文/不壞的土叔 我叫張陵,是天一觀的道長姆坚。 經(jīng)常有香客問我澳泵,道長,這世上最難降的妖魔是什么兼呵? 我笑而不...
    開封第一講書人閱讀 55,252評論 1 279
  • 正文 為了忘掉前任兔辅,我火速辦了婚禮腊敲,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘维苔。我一直安慰自己碰辅,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 64,253評論 5 371
  • 文/花漫 我一把揭開白布介时。 她就那樣靜靜地躺著乎赴,像睡著了一般。 火紅的嫁衣襯著肌膚如雪潮尝。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,031評論 1 285
  • 那天饿序,我揣著相機(jī)與錄音勉失,去河邊找鬼。 笑死原探,一個胖子當(dāng)著我的面吹牛乱凿,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播咽弦,決...
    沈念sama閱讀 38,340評論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼徒蟆,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了型型?” 一聲冷哼從身側(cè)響起段审,我...
    開封第一講書人閱讀 36,973評論 0 259
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎闹蒜,沒想到半個月后寺枉,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,466評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡绷落,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 35,937評論 2 323
  • 正文 我和宋清朗相戀三年姥闪,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片砌烁。...
    茶點故事閱讀 38,039評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡筐喳,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出函喉,到底是詐尸還是另有隱情避归,我是刑警寧澤,帶...
    沈念sama閱讀 33,701評論 4 323
  • 正文 年R本政府宣布函似,位于F島的核電站槐脏,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏撇寞。R本人自食惡果不足惜顿天,卻給世界環(huán)境...
    茶點故事閱讀 39,254評論 3 307
  • 文/蒙蒙 一堂氯、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧牌废,春花似錦咽白、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,259評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至懂从,卻和暖如春授段,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背番甩。 一陣腳步聲響...
    開封第一講書人閱讀 31,485評論 1 262
  • 我被黑心中介騙來泰國打工侵贵, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人缘薛。 一個月前我還...
    沈念sama閱讀 45,497評論 2 354
  • 正文 我出身青樓窍育,卻偏偏與公主長得像,于是被迫代替她去往敵國和親宴胧。 傳聞我的和親對象是個殘疾皇子漱抓,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,786評論 2 345

推薦閱讀更多精彩內(nèi)容

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理乍赫,服務(wù)發(fā)現(xiàn)故河,斷路器,智...
    卡卡羅2017閱讀 134,599評論 18 139
  • Spring Boot 參考指南 介紹 轉(zhuǎn)載自:https://www.gitbook.com/book/qbgb...
    毛宇鵬閱讀 46,748評論 6 342
  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 171,504評論 25 707
  • 這世界上最可怕的付出就是:我都是為了你。 我的朋友小斯敦间,最近辭職了瓶逃。 她從剛畢業(yè)就在這家公司服務(wù),14年了廓块。從職場...
    江來閱讀 208評論 0 2
  • 當(dāng)時間被回顧時會發(fā)現(xiàn)時光荏苒厢绝,不過彈指一揮間,特別是當(dāng)自己忙忙碌碌卻沒有多大實際作為時带猴,一種憤懣昔汉、悲涼的情緒涌...
    天空里的飛鳥閱讀 2,993評論 1 50