Bean Validation完结篇:你必须关注的边边角角(约束级联、自定义约束、自定义校验器、国际化失败消息…)【享学Spring】(上)

前言

一般来说,对于web项目我们都有必要对请求参数进行校验,有的前端使用JavaScript校验,但是为了安全spring翻译起见后端的校验都是必须的。因此数据校验不仅仅是在web下&#安全教育平台登录入口xff0c;在方方面面都是一个重要的点。前端校验有它的JS校验框架(比如我之前用的jQuer安全教育平台登录y Validation Plugin),后端自然也少不了。

前面洋洋洒洒已经把数据校验Bean Validation讲了很多了,如果你已经运用在你的项目数据库连接中,势必将大大提高生产力吧,本文作为完结篇(不java是什么意思是总结篇)就不用再系统性的介绍Bean Validation他了,而是旨在介绍你在使用过程中不得不关心的周边、细节~

如果说前面是用机,那么本文就有点玩机的意思~

BV(Bean Validation)的使用范围

本次再次强java环境变量配置调了这一点(springboot设计思想是我认为特别重要的存在):使用范围。

Bean Validation并不局限于应用程序的某一层或者哪种编程模型, 它可以被用在任何一层, 除了web程序&#安全教育平台登录入口xff0c;也可以是像Swing这样的富客户端程序中&#javascript高级程序设计xff08;GUI编程)。javascript

我抄了一副业界著名的图给大家:

Bean Validation完结篇:你必须关注的边边角角(约束级联、自定义约束、自定义校验器、国际化失败消息...)【享学Spring】(上)

Bean Validatiapi接口on的目标是简化Bean校spring验,将以往重复的校验逻辑进行抽象和spring标准化,形程序员计算器成统一API规范;

说到抽象统一API,它可不是乱来前端开发和后端开发有什么区别的,java环境变量配置只有当你能最大程度的得到公有,这个动作才有意义,至少它一般都是与业务无关的。抽象能力是对程序员分级的最重要标准之一安全教育日

约束继承

如果子类继承自他的父类,除了校验子类javascript面试题,同时还会校验父类,这就是约束继承(同样适用于接口)。

// child和person上标注的约束都会被执行
public class Child extends Person {
    ...
}

注意:如果子类覆盖了父类的方法,那么子类和父类的约束都会被校验。

约束级联(级联校验)

如果要验证属性关联的对象,那么需要在属性上添加@Valid注解,如果一个对象被校验,那么它的所有的标注了@Valid的关联对象都会被校验阿飘是什么意思f0c;这些对象也可以是数组、集合、安全教育日Map等,这时会验证他们持有的所有元素。

Demospring面试题f1a;

@Getter
@Setter
@ToString
public class Person {
    @NotNull
    private String name;
    @NotNull
    @Positive
    private Integer age;
    @Valid
    @NotNull
    private InnerChild child;
    @Valid // 让它校验List里面所有的属性
    private List<InnerChild> childList;
    @Getter
    @Setter
    @ToString
    public static class InnerChild {
        @NotNull
        private String name;
        @NotNull
        @Positive
        private Integer age;
    }
}

校验程序:

    public static void main(String[] args) {
        Person person = new Person();
        person.setName("fsx");
        Person.InnerChild child = new Person.InnerChild();
        child.setName("fsx-age");
        child.setAge(-1);
        person.setChild(child);
        // 设置childList
        person.setChildList(new ArrayList<Person.InnerChild>(){{
            Person.InnerChild innerChild = new Person.InnerChild();
            innerChild.setName("innerChild1");
            innerChild.setAge(-11);
            add(innerChild);
            innerChild = new Person.InnerChild();
            innerChild.setName("innerChild2");
            innerChild.setAge(-12);
            add(innerChild);
        }});
        Validator validator = Validation.byProvider(HibernateValidator.class).configure().failFast(false)
                .buildValidatorFactory().getValidator();
        Set<ConstraintViolation<Person>> result = validator.validate(person);
        // 输出错误消息
        result.stream().map(v -> v.getPropertyPath() + " " + v.getMessage() + ": " + v.getInvalidValue())
                .forEach(System.out::println);
    }

打印校验失败的消息:

age 不能为null: null
childList[0].age 必须是正数: -11
child.age 必须是正数: -1
childList[1].age 必须是正数: -12

约束失败消息message自定义

程序员那么可爱免费观看个约束定义中都包含有一个用于提示验证结果的消阿飘重生五年前军人老公息模版message,啊拼音并且在声明一个约束条件的时候,你可以通过这个约束注解中的me阿飘军人老公二爷ssage属性来重写默认的数据库连接运算消息模版(这是自定义message最简单的一javascriptdownload种方式&#前端开发入门薪水xff09;。

如果在校验的时候,这个约束条件没有通过,那么你配置的M数据库连接池essageInterpolator插值器会被用来当成解析器来解析这个约束中定义的消息模版, 从而得到最终的验证spring翻译失败提示信息。

默认使用的插值器是org.前端开发入门薪水hibernate.validator.m阿飘essageinterpolation.ResourceBundleMessageInterpolator,它借助org.hibernate.validator.spi.resourceloading.ResourceBundleLocator来获取到国际化资源属性文件从而填充模版内容~

资源解析器默认java怎么读使用的实spring翻译现是Pl程序员怎么学atforapi接口mResourceBundlespringboot面试题Locator,在配置Config数据库连接语句uration初始化的时候默认程序员工作一年后工资被赋值:

    private ConfigurationImpl() {
        this.validationBootstrapParameters = new ValidationBootstrapParameters();
        // 默认的国际化资源文件加载器USER_VALIDATION_MESSAGES值为:ValidationMessages
        // 这个值就是资源文件的文件名~~~~
        this.defaultResourceBundleLocator = new PlatformResourceBundleLocator(
                ResourceBundleMessageInterpolator.USER_VALIDATION_MESSAGES
        );
        this.defaultTraversableResolver = TraversableResolvers.getDefault();
        this.defaultConstraintValidatorFactory = new ConstraintValidatorFactoryImpl();
        this.defaultParameterNameProvider = new DefaultParameterNameProvider();
        this.defaultClockProvider = DefaultClockProvider.INSTANCE;
    }

这个解析器会尝试解析模版中的占位符( 大括号括起来的字符串,形如这样{xxx})。

它解析mspring框架essage的核心代码如下&#x安全教育手抄报ff08;比如此处message模版是{javax.vali安全教育平台登录入口dation.constraints.NotNull前端开发需要学什么.message}为例):

public abstract class AbstractMessageInterpolator implements MessageInterpolator {
    ...
    private String interpolateMessage(String message, Context context, Locale locale) throws MessageDescriptorFormatException {
        // 如果message消息木有占位符,那就直接返回  不再处理了~
        // 这里自定义的优先级是最高的~~~
        if ( message.indexOf( '{' ) < 0 ) {
            return replaceEscapedLiterals( message );
        }
        // 调用resolveMessage方法处理message中的占位符和el表达式
        if ( cachingEnabled ) {
            resolvedMessage = resolvedMessages.computeIfAbsent( new LocalizedMessage( message, locale ), lm -> resolveMessage( message, locale ) );
        } else {
            resolvedMessage = resolveMessage( message, locale );
        }
        ...
    }
    private String resolveMessage(String message, Locale locale) {
        String resolvedMessage = message;
        // 获取资源ResourceBundle三部曲
        ResourceBundle userResourceBundle = userResourceBundleLocator.getResourceBundle( locale );
        ResourceBundle constraintContributorResourceBundle = contributorResourceBundleLocator.getResourceBundle( locale );
        ResourceBundle defaultResourceBundle = defaultResourceBundleLocator.getResourceBundle( locale );
        ...
    }
}

对如上me程序员需要什么学历ssage的处理步骤大致总结如下:

  1. 若没占位符符号{需要处理,直接返回(比如我们自定义message属性值全是文字,就直接返回了)~有占位符或者EL,交给resolveMessage()方法从资源文件里拿内容来处程序员理~
  2. 拿取资源文件,按数据库连接失败照如下三个步骤寻找数据库连接池的作用f1a;1. userResourceBjavaundleLocator:去用户自己的classpath里面去找资源文件(默认名字是ValidationMessages.proapi是什么perties,当然你也可以使用国际化名&#x程序员计算器ff09;2. contributor程序员工资一般多少ResourceBundleLocator:加载贡献的资源包3. defaultResourceBundle:默认的策略。去这里于/org/hibernate/validator加载ValidationMessages.properties
  3. 需要注意的是,如上是加载资源的顺序。无论怎么样,这三处的资源文件都会加载进内存的(并无短路逻辑)。javascript是干什么的进行占位符匹配的时候,依旧遵守这规律:1. 最先用自己当前项目classpath下的资源去匹配资源占位符,若没匹配上再用下一级api接口别的资源~~~2. 规律同上&javascript是干什么的#xff0c;依次数据库连接类推,递归javascript面试题的匹配所有的占位符(若占apink位符没匹配上,原样输出,并不是输出null哦~)

需要注意的是,因为{在此处是特殊字符,若你就想输出{,请转义:\{