偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

必讀!SpringBoot接口參數(shù)校驗(yàn)N種實(shí)用技巧大揭秘

開發(fā) 前端
我們通過 AOP 技術(shù)實(shí)現(xiàn)了參數(shù)統(tǒng)一處理,但是這樣輸出錯(cuò)誤信息很不友好,接下來我們來完善下,通過全局異常通知攔截處理。這里的異常信息我們可以通過全局異常處理下格式。

環(huán)境:SpringBoot2.6.12

實(shí)際的開發(fā)工作中大部分的接口都是需要進(jìn)行參數(shù)有效性校驗(yàn)的,參數(shù)可能是簡(jiǎn)單的基本數(shù)據(jù)類型,也可能是對(duì)象類型,基本上所有接收參數(shù)的接口都是需要對(duì)這些參數(shù)進(jìn)行校驗(yàn)的,你對(duì)這些參數(shù)是怎么校驗(yàn)的?接下來帶你一起見識(shí)下我在實(shí)際項(xiàng)目中都應(yīng)用過哪些校驗(yàn)姿勢(shì)!。該案例會(huì)詳細(xì)介紹如下 7 方面的內(nèi)容。

  1. 簡(jiǎn)單參數(shù)校驗(yàn)
  2. 參數(shù)校驗(yàn)分組
  3. 單個(gè)參數(shù)校驗(yàn)
  4. 嵌套參數(shù)校驗(yàn)
  5. 自定義工具類參數(shù)校驗(yàn)
  6. 國(guó)際化支持
  7. AOP 驗(yàn)證參數(shù)統(tǒng)一處理

在正式介紹主體內(nèi)容前我們還是先要了解學(xué)習(xí)一些規(guī)范 JSR303。

JSR 是什么?

JSR 是 Java Specification Requests 的縮寫,意思是 Java 規(guī)范提案。是指向 JCP(Java Community Process)提出新增一個(gè)標(biāo)準(zhǔn)化技術(shù)規(guī)范的正式請(qǐng)求。任何人都可以提交 JSR,以向 Java 平臺(tái)增添新的 API 和服務(wù)。JSR 已成為 Java 界的一個(gè)重要標(biāo)準(zhǔn)。JSR-303 是 JAVA EE 6 中的一項(xiàng)子規(guī)范,叫做 Bean Validation,Hibernate Validator 是 Bean Validation 的參考實(shí)現(xiàn) . Hibernate Validator 提供了 JSR 303 規(guī)范中所有內(nèi)置 constraint 的實(shí)現(xiàn),除此之外還有一些附加的 constraint。相關(guān)注解如下:

圖片圖片

在Spring中提供了SpringValidation驗(yàn)證框架對(duì)參數(shù)的驗(yàn)證機(jī)制提供了@Validated(Spring'sJSR-303規(guī)范,是標(biāo)準(zhǔn)JSR-303的一個(gè)變種),javax提供了@Valid(標(biāo)準(zhǔn)JSR-303規(guī)范),結(jié)合BindingResult對(duì)象可以直接獲取錯(cuò)誤信息。在本案中這兩種是等效的,但是也有區(qū)別,在接下來的案例中將會(huì)說明。

1. 配置依賴

<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjrt</artifactId>
</dependency>
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
<scope>runtime</scope>
</dependency>
</dependencies>

org.aspectj 依賴是在最后我們要通過 AOP 技術(shù)來實(shí)現(xiàn)統(tǒng)一參數(shù)的校驗(yàn)。

2. 參數(shù)驗(yàn)證

  • 簡(jiǎn)單參數(shù)校驗(yàn)
public class Users {
  @NotEmpty(message = "姓名必需填寫")
  private String name ;
  @Min(value = 10, message = "年齡不能小于 10")
  private Integer age ;
  @Length(min = 6, max = 18, message = "郵箱介于 6 到 18 之間")
  private String email ;
  @NotEmpty(message = "電話必需填寫")
  private String phone ;
  // 這里的2個(gè)接口在下面的案例中會(huì)使用到
  public static interface G1 {}
  public static interface G2 {}
}

這里對(duì)需要校驗(yàn)的字段都應(yīng)用了不同的注解來約束。接下來就是在Controller接口上添加相應(yīng)的注解即可:

@ResponseBody
public class UsersController extends BaseController {
  @RequestMapping(value = "/valid/save1", method = RequestMethod.POST)
  public Object save1(@RequestBody @Validated Users user, BindingResult result) {
    Optional<List<String>> op = valid(result) ;
    if (op.isPresent()) {
      return op.get() ;
    }
    return "success" ;
  }
}
public class BaseController {
  protected Optional<List<String>> valid(BindingResult result) {
    if (result.hasErrors()) {
      return Optional.of(result.getAllErrors().stream().map(err -> err.getDefaultMessage()).collect(Collectors.toList())) ;
    }
    return Optional.empty() ;
  }
}

接收參數(shù)的 Users 對(duì)象前面要是用@Validated 注解,并且通過 BindingResult 來收集錯(cuò)誤信息(可判斷是否有錯(cuò)誤信息);測(cè)試如下:

圖片圖片

正確情況

圖片圖片

  • 參數(shù)校驗(yàn)分組

有些時(shí)候我們這一個(gè)對(duì)象可能會(huì)應(yīng)用到不同的場(chǎng)景,出現(xiàn)不同的校驗(yàn)規(guī)則該怎么做呢?這時(shí)候我們就可以應(yīng)用分組功能,不同的應(yīng)用場(chǎng)景指明不同的分組即可,開始擼。注意:JSR303 是沒有分組功能的。

public class Users {
  @NotEmpty(message = "姓名不能為空", groups = G1.class)
  private String name ;
  @Min(value = 10, message = "年齡不能小于 10", groups = G1.class)
  @Min(value = 20, message = "年齡不能小于 20", groups = G2.class)
  private Integer age ;
  @Length(min = 6, max = 18, message = "郵箱介于 6 到 18 之間", groups = {G1.class, G2.class})
  private String email ;
  @NotEmpty(message = "電話必需填寫")
  private String phone ;
  public static interface G1 {}
  public static interface G2 {}
}

這里不同的字段上加了 groups 屬性,指明屬于哪個(gè)分組。注意在該實(shí)體類中我們又定義了 2 個(gè)類 G1,G2 就是為了分組用的(具體指明哪個(gè)分組)。接口處理:

@RequestMapping(value = "/valid/save1", method = RequestMethod.POST)
public Object save1(@RequestBody @Validated(Users.G1.class) Users user, BindingResult result) {
  Optional<List<String>> op = valid(result) ;
  if (op.isPresent()) {
    return op.get() ;
  }
  return "success" ;
}
@RequestMapping(value = "/valid/save2", method = RequestMethod.POST)
public Object save2(@RequestBody @Validated(Users.G2.class) Users user, BindingResult result) {
  Optional<List<String>> op = valid(result) ;
  if (op.isPresent()) {
    return op.get() ;
  }
  return "success" ;
}

在這個(gè)兩個(gè)接口中 @Validated(Users.G2.class)分別指明了自己的分組,接下來測(cè)試看看效果。

分組 G1 測(cè)試:

圖片圖片

從這里返回的信息來看我們的 phone 雖然寫了@NotEmpty 但是并沒有起作用,因?yàn)槲覀儾]有指明他的分組,并且接口上我們指明了是用 G1 分組。

分組 G2 測(cè)試:

圖片圖片

在這個(gè)接口中發(fā)現(xiàn) name 驗(yàn)證是 G1 的,所以這里不會(huì)進(jìn)行校驗(yàn),并且年齡的判斷是不能小于 20 了。

  • 單個(gè)參數(shù)校驗(yàn)

單個(gè)參數(shù)的校驗(yàn)不需要實(shí)體對(duì)象,一般就是吧 JSR303 相關(guān)的注解直接應(yīng)用到接口參數(shù)上即可。同時(shí)還需要在 Controller 類上添加@Validated 注解。

@Validated
public class UsersController extends BaseController {
  @PackMapping("/valid/find")
  public Object find(@NotEmpty(message = "參數(shù) Id 不能為空") String id) {
    return "查詢到參數(shù)【" + id + "】" ;
  }
}

該接口中直接將注解應(yīng)用到參數(shù)上。測(cè)試:

圖片圖片

同時(shí)控制臺(tái)會(huì)輸出如下異常:

圖片圖片

你也發(fā)現(xiàn)這種異常信息提示很不友好,接下來我們做個(gè)簡(jiǎn)單的局部異常處理。

我們只需要在Controller添加如下方法即可:

@ExceptionHandler(ConstraintViolationException.class)
@ResponseBody
public Object ConstraintViolationExceptionHandler(ConstraintViolationException e) {
  String message = e.getConstraintViolations().stream().map(ConstraintViolation::getMessage).collect(Collectors.joining());
  return message ;
}

在該 Controller 中我們添加了一個(gè)異常處理句柄(簡(jiǎn)單吧將錯(cuò)誤信息輸出)。

圖片圖片

  • 嵌套參數(shù)校驗(yàn)

在實(shí)際的工作中往往參數(shù)對(duì)象比這復(fù)雜的多,Users 對(duì)象中可能還嵌套有其他的對(duì)象,這個(gè)其他的對(duì)象也可能需要參數(shù)的校驗(yàn)。接下來我們就來看看這種嵌套參數(shù)是如何校驗(yàn)的。

public class Users {
  @NotEmpty(message = "姓名不能為空", groups = G1.class)
  private String name ;
  @Min(value = 10, message = "年齡不能小于 10", groups = G1.class)
  @Min(value = 20, message = "年齡不能小于 20", groups = G2.class)
  private Integer age ;
  @Length(min = 6, max = 18, message = "郵箱介于 6 到 18 之間", groups = {G1.class, G2.class})
  private String email ;
  @NotEmpty(message = "電話必需填寫")
  private String phone ;
  @Valid
  private Address address;
}

注意:嵌套對(duì)象 Address 的校驗(yàn)需要在上面加@Valid 注解。

public class Address {
  @NotEmpty(message = "地址信息必需填寫")
  private String addr ;
}

測(cè)試接口:

@RequestMapping(value = "/valid/save3", method = RequestMethod.POST)
public Object save3(@RequestBody @Validated Users user, BindingResult result) {
  Optional<List<String>> op = valid(result) ;
  if (op.isPresent()) {
    return op.get() ;
  }
  return "success" ;
}

接口上沒有什么特別的與之前的一模一樣。注意:這里的校驗(yàn)沒有設(shè)定分組,所以校驗(yàn)時(shí)都是校驗(yàn)的沒有設(shè)置分組的字段。

測(cè)試:

圖片圖片

這里發(fā)現(xiàn)我們的地址信息根本就沒有進(jìn)行校驗(yàn)。接著我們吧參數(shù)變動(dòng)下

圖片圖片

參數(shù)中我們吧 address 字段設(shè)置上后 參數(shù)進(jìn)行校驗(yàn)了。接下來修改 Users 實(shí)體,吧 Address 默認(rèn) new 出來再進(jìn)行測(cè)試

@Valid
private Address address = new Address();

圖片圖片

發(fā)現(xiàn)即便我們的入?yún)]有 address 字段也能進(jìn)行校驗(yàn)了,這里大家需要注意下。

  • 自定義參數(shù)校驗(yàn)

請(qǐng)查看【技巧】API接口參數(shù)驗(yàn)證的必備神器,讓你的代碼更高效!

  • 國(guó)際化支持
public class Users {
  @NotEmpty(message = "{name.notempty}", groups = G1.class)
  private String name ;
  @Min(value = 10, message = "年齡不能小于 10", groups = G1.class)
  @Min(value = 20, message = "年齡不能小于 20", groups = G2.class)
  private Integer age ;
  @Length(min = 6, max = 18, message = "郵箱介于 6 到 18 之間", groups = {G1.class,
  G2.class})
  private String email ;
  @NotEmpty(message = "電話必需填寫")
  private String phone ;
  @Valid
  private Address address = new Address();
}

注意這里的 name 字段中的 message 屬性我們使用了表達(dá)式的方式,而 name.notempty 為我們?cè)谫Y源文件中定義的 key。接下來,在 resources/下新建如下屬性文件:

圖片圖片

屬性文件必須是 ValidationMessages 開頭。默認(rèn)文件及 zh_CN 內(nèi)容:

name.notempty=姓名必需填寫

en_US 內(nèi)容:

name.notempty=name is require

測(cè)試:

圖片圖片

為了模擬英文環(huán)境,我們需要設(shè)置請(qǐng)求頭 Accept-Language:en-US

圖片圖片

顯示了 en_US.properties 中定義的消息,到此國(guó)際化完成。

  • AOP 驗(yàn)證參數(shù)統(tǒng)一處理

自定義注解標(biāo)記需要進(jìn)行統(tǒng)一參數(shù)校驗(yàn)處理的接口。

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface EnableValidate {}

AOP切面類

@Component
@Aspect
public class ValidateAspect {
  @Pointcut("@annotation(com.pack.params.valid.EnableValidate)")
  public void valid() {}
  @Before("valid()")
  public void validateBefore(JoinPoint jp) {
    Object[] args = jp.getArgs() ;
    for (Object arg : args) {
      if (arg instanceof BindingResult) {
        BindingResult result = (BindingResult) arg ;
        if (result.hasErrors()) {
          String messages = result.getAllErrors().stream().map(err -> err.getDefaultMessage()).collect(Collectors.joining(",")) ;
          throw new ParamsException(messages) ;
        }
      }
    }
  }
}

定義了一個(gè)前置通知,攔截標(biāo)記有@EnableValidate 注解的接口。如果有異常信息收集錯(cuò)誤信息然后拋出異常信息。測(cè)試:

@PackMapping(value = "/valid/save1", method = RequestMethod.POST)
@EnableValidate
public Object save1(@RequestBody @Validated(Users.G1.class) Users user, BindingResult result) {
  Optional<List<String>> op = valid(result) ;
  if (op.isPresent()) {
    return op.get() ;
  }
  return "success" ;
}

圖片圖片

到此我們通過 AOP 技術(shù)實(shí)現(xiàn)了參數(shù)統(tǒng)一處理,但是這樣輸出錯(cuò)誤信息很不友好,接下來我們來完善下,通過全局異常通知攔截處理。這里的異常信息我們可以通過全局異常處理下格式。

完畢?。?!

責(zé)任編輯:武曉燕 來源: Spring全家桶實(shí)戰(zhàn)案例源碼
相關(guān)推薦

2024-11-29 10:48:54

IDEA技巧Mac

2009-12-21 15:50:39

2009-09-04 10:27:28

Linux實(shí)用技巧linux操作系統(tǒng)linux

2022-03-23 09:18:10

Git技巧Linux

2009-01-03 09:34:30

ASP.NET.NET性能優(yōu)化

2011-04-08 15:40:01

Oracle認(rèn)證

2022-11-03 10:28:59

PandasSAC機(jī)制

2022-10-11 08:00:47

多線程開發(fā)技巧

2024-05-17 08:52:43

SQL實(shí)用技巧行列轉(zhuǎn)換

2022-12-30 08:49:41

SpringBoot@Validated

2023-03-16 08:23:33

2010-10-27 14:17:19

UI設(shè)計(jì)布局

2009-12-23 17:32:35

Linux構(gòu)建軟路由

2019-10-10 16:31:51

PyCharmPythonWindows

2010-10-08 15:44:17

vim

2019-12-22 23:10:19

LinuxSSH加密

2009-12-09 11:21:30

Linux實(shí)用技巧

2019-11-25 10:12:59

Python技巧工具

2010-09-14 10:41:24

DIV+CSS排版

2009-07-17 09:47:41

Linux RootLinux使用Linux開發(fā)
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)