一、控制器获得前端传来的参数
1、url路径带的参数
形式一:/banner/id (id是参数值 比如:1)
在函数参数中使用@PathVariable Integer id,参考示例代码1
注意:参数名字的定义,要跟@GetMapping("/test1/{id}")中的id一致,如果不一致,要使用@PathVariable(name="id")
2)/banner?id=1&name=liangdm
使用@RequestParam String name,参考示例代码1
示例代码1
假设请求url地址为:http://localhost/api/vi/test1/1?name=deming
@GetMapping("/test1/{id}")
public void test1(@PathVariable Integer id,
@RequestParam String name) {
System.out.println("xxxxx");
}
2、获得RequestBody体传递过来的参数,一般是json对象
{
username:"deming",
password:"123456"
}
1) 方法一 :使用@RequestBody Map<String,Object> person 这种方法需要装箱、解构,比较麻烦,不推荐使用
2)方法二:定义数据传输对象 PersonDTO person,使用@RequestBody PersonDTO person来接受参数
步骤一:创建javabean,如使用@Getter、@Setter,需要使用maven安装lombok插件
public class PersonDTO {
private String username;
private String password;
}
步骤二:
@PostMapping("/test2")
public PersonDTO test2(@RequestBody PersonDTO person) {
PersonDTO dto = PersonDTO.builder().name("liangde").age(18).build();
return dto;
}
二、对获得参数进行校验
使用参数校验,首先在控制器的类上打上@Validated 开启参数校验
@Validated
public class BannerController {
}
1 使用基础内置的验证注解,如:@Range @Length @Max @Min等,适用于url请求的路径参数
@GetMapping("/test1/{id}")
public void test1(@PathVariable @Range(min = 1, max = 10, message = "id号超出1-10") Integer id
, @RequestParam @Length(min = 7) String name) {
System.out.println("this is test1");
}
2、如果是对json对象进行参数校验,需要在@RequestBody 加上@Validated 注解,如下所示
第一步:在javabean对象前,打上@Validated 注解
public PersonDTO test2(@RequestBody @Validated PersonDTO person) {
PersonDTO dto = PersonDTO.builder().name("liangde").age(18).build();
return dto;
}
第二步,因为PersonDTO的成员变量是基础参数,对这些基础参数的校验使用基础验证注解就可以了,如下所示:
public class PersonDTO {
@Length(min=2,max=10,message="姓名的长度应在2~10之间")
private String name;
}
另外,如果说PersonDTO中有SchoolDTO这种对象成员变量,对这种成员变量的校验,需要打上@Valid注解,这种情况叫级连验证,如下所示
public class PersonDTO {
@Length(min=2,max=10,message="姓名的长度应在2~10之间")
private String name;
@Valid
private SchoolDTO schoolDTO;
}
注:schoolDTO下的成员变量参数校验使用普通验证注解
其他基础校验注解
@Null //被注释的元素必须为null@NotNull //被注释的元素不能为null@AssertTrue //被注释的元素必须为true@AssertFalse //被注释的元素必须为false@Min(value) //被注释的元素必须是一个数字,其值必须大于等于指定的最小值@Max(value) //被注释的元素必须是一个数字,其值必须小于等于指定的最大值@DecimalMin(value) //被注释的元素必须是一个数字,其值必须大于等于指定的最小值@DecimalMax(value) //被注释的元素必须是一个数字,其值必须小于等于指定的最大值@Size(max,min) //被注释的元素的大小必须在指定的范围内。@Digits(integer,fraction) //被注释的元素必须是一个数字,其值必须在可接受的范围内@Past //被注释的元素必须是一个过去的日期@Future //被注释的元素必须是一个将来的日期@Pattern(value) //被注释的元素必须符合指定的正则表达式。@Email //被注释的元素必须是电子邮件地址@Length //被注释的字符串的大小必须在指定的范围内@NotEmpty //被注释的字符串必须非空@Range //被注释的元素必须在合适的范围内
2 自定义校验注解做参数校验
第一步:定义一个注解类型PasswordEqual
@Documented
@Retention(RetentionPolicy.RUNTIME)
//ElementType.TYPE 注解应用在类中, ElementType.METHOD 注解应用在方法,,ElementType.filed 注解应用在字段
@Target(ElementType.TYPE)
@Constraint(validatedBy = PasswordValidator.class) //关联类设定PasswordValidator.class,双向关联
public @interface PasswordEqual {
int min() default 2;
int max() default 6;
String message() default "password are not equal"; //出错时,提示消息
// 规范中必须有以下两个模版方法
Class<?> [] groups() default {};
Class<? extends Payload>[] payload() default {};
}
第二步:创建自定义注解的关联类PasswordValidator ,通过实现接口进行注解关联
//第一个参数时自定义注解PasswordEqual,,第二个参数是注解修饰的类型PersonDTO
public class PasswordValidator implements ConstraintValidator<PasswordEqual, PersonDTO> {
private int min;
private int max;
private String message;
@Override
public void initialize(PasswordEqual constraintAnnotation) {
this.min = constraintAnnotation.min();
this.max = constraintAnnotation.max();
this.message = constraintAnnotation.message();
}
@Override
public boolean isValid(PersonDTO value, ConstraintValidatorContext context) {
String password1 = value.getPassword1();
String password2 = value.getPassword2();
boolean match = password1.equals(password2);
System.out.println(this.min);
System.out.println(this.max);
System.out.println(this.message);
return match;
}
}
3、应用注解
@PasswordEqual(min=1,max=100,message="两次密码输入不一致") //注解打在类上面
public class PersonDTO {
private String password1;
private String password2;
}
4、参数校验过程捕获异常,并反馈给前端
前面是使用注解的方法校验参数的异常,一般在后台信息打印异常信息,如果需要把异常信息返回给前端,需要增强框架内容,增加以下方法捕获异常并返回异常信息
1)捕获url路径参数异常
@ExceptionHandler(ConstraintViolationException.class)
@ResponseBody
@ResponseStatus(code=HttpStatus.BAD_REQUEST)
public UnifyResponse handleConstraintException(HttpServletRequest req, ConstraintViolationException e){
String requestURL = req.getRequestURI();
String method = req.getMethod();
String message = e.getMessage();
return new UnifyResponse(10001,message,method + " " + requestURL);
}
2) 捕获DTO参数错误异常,可以同时处理多个参数异常, 状态码统一为参数错误状态码
@ExceptionHandler(MethodArgumentNotValidException.class)
@ResponseStatus(code= HttpStatus.BAD_REQUEST)
@ResponseBody
public UnifyResponse handleBeanValidation(HttpServletRequest req,MethodArgumentNotValidException e) {
String requestURL = req.getRequestURI();
String method = req.getMethod();
List<ObjectError> errors = e.getBindingResult().getAllErrors();
String message = this.FormatAllErrorsMessage(errors);
return new UnifyResponse(10001,message,method + " " +requestURL);
}
private String FormatAllErrorsMessage( List<ObjectError> errors) {
StringBuffer errMsg = new StringBuffer();
errors.forEach(error->
errMsg.append(error.getDefaultMessage()).append(","));
return errMsg.toString();
}