公司转java开发也有一段时间了,在实际开发过程当中仍是会遇到一些问题的,本篇主要记录下接口服务中参数验证相关的开发过程和一些知识点。
在接口服务开发中,不免会校验传入方的参数校验,尤为在post请求时,验证字符长度,字符类型是否知足数据库中字段的最大长度及类型,若是不符合条件应及时拦截并返回,避免后续的流程。java
先了解下提供的注解,基本上经常使用的都提供了,在代码编写时仍是比较方便的,一个注解解决了验证逻辑。git
/**Bean Validation 中内置的 constraint**/ @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(regex=,flag=) //被注释的元素必须符合指定的正则表达式 /**Hibernate Validator 附加的 constraint**// @NotBlank(message =) //验证字符串非null,且长度必须大于0 @Email //被注释的元素必须是电子邮箱地址 @Length(min=,max=) //被注释的字符串的大小必须在指定的范围内 @NotEmpty //被注释的字符串的必须非空 @Range(min=,max=,message=) //被注释的元素必须在合适的范围内
/** * 订单号 */ @Range(min=1,message = "不是正确的订单号") private Long e_order_id; /** * 产品code */ @NotBlank(message = "不是正确的产品code") private String product_code;
在controller中,咱们经过BindingResult来接收对应的验证信息正则表达式
@ApiOperation(value = "修改订单状态", notes = "若找不到结果则返回 null。") @RequestMapping(value = "/status", method = RequestMethod.PUT) @ResponseBody public String PutOrderStatus(@RequestBody @Validated @NotNull OrderStatusReq req, BindingResult bindingResult) { String validResult = assertParameterValid(bindingResult); if (validResult != null) { return validResult; } return iOrderStatusService.putOrderStatus(req).toString(); }
protected String assertParameterValid(BindingResult bindingResult) { if (bindingResult.hasErrors()) { FieldError error = bindingResult.getFieldError(); return new Response<>(BusinessReturnCode.VALIDATION_FAILURE, String.format("[%s] %s.", error.getField(), error.getDefaultMessage()), null).toString(); } return null; }
惋惜的是,Hibernate validation
中没有提供枚举相关的校验,而实际业务场景中会有不少校验类型、状态等,这里咱们只能自定义了。数据库
首先咱们须要自定义一个annotation
来标记你的验证字段,由于Validator框架里面的基础annotation已经不够用。c#
而后自定义一个Validator
(继承ConstraintValidator),并将annotation
类型给到ConstraintValidator
的泛型列表,至关于作了一个绑定。而后implement ConstraintValidator的两个方法,在isValid
方法里面用要验证的枚举验证参数。app
能够看下一个简单的demo:框架
@Target({ElementType.FIELD, ElementType.METHOD, ElementType.ANNOTATION_TYPE}) @Retention(RetentionPolicy.RUNTIME) @Constraint(validatedBy = {EnumValidAnnotation.EnumValidtor.class}) @Documented public @interface EnumValidAnnotation { String message() default "枚举不在范围内"; Class<?>[] groups() default {}; Class<? extends Payload>[] payload() default {}; Class<?>[] target() default {}; public class EnumValidtor implements ConstraintValidator<EnumValidAnnotation, Integer> { Class<?>[] cls; //枚举类 @Override public void initialize(EnumValidAnnotation constraintAnnotation) { cls = constraintAnnotation.target(); } @Override public boolean isValid(Integer value, ConstraintValidatorContext context) { System.out.println("枚举值" + value); if (cls.length > 0) { for (Class<?> cl : cls) { try { if (cl.isEnum()) { //枚举类验证 Object[] objs = cl.getEnumConstants(); Method method = cl.getMethod("getCode"); for (Object obj : objs) { Object code = method.invoke(obj); if (value.equals(code)) { return true; } } } } catch (NoSuchMethodException e) { e.printStackTrace(); } catch (IllegalAccessException e) { e.printStackTrace(); } catch (InvocationTargetException e) { e.printStackTrace(); } } } return false; } }
这样的话在你要验证的字段上加上对应的注解便可:ide
/** * 更新类型 */ @EnumValidAnnotation(target = OrderStatusEnum.class) private int stype;
java中注解仍是挺有意思的,相似c#中的attribute,但java中各类框架、方法的注解真的不少,不一直使用或作对应的笔记真的很容易忘记,仍是须要多多积累和记录的。post