借助JSR 310,Java 8终于为我们带来了不错的日期和时间API。 对于仍在使用Java 7的那些人(例如我目前在我的当前项目中),有很好的反向移植,请访问www.threeten.org了解更多详细信息。 但是,由于有关该主题的博客已经很多,因此我将不涉及使用新API的任何细节。 在本文中,我将向您展示的是如何通过编写自己的自定义批注将Date / Time API与JSR 303 Bean验证API结合使用。

如果您同时使用bean验证和新的日期/时间API,则可能需要结合使用它们。 API和Hibernate Validator之类的实现仅提供少数约束,例如NotEmpty@Pattern 。 但是,到目前为止,JSR 310还没有现成的约束。幸运的是,创建自己的约束非常容易。 作为示例,我将演示如何编写自己的@Past批注以验证java.time.LocalDate字段。

为了进行测试,我们将从一个非常简单的类开始,该类包含一个日期和一个dateTime。 这些字段应该代表过去的日期。 因此,它们使用@Past注释进行注释:

ClassWithPastDates

package it.jdev.example.jsr310.validator;import java.time.LocalDate;
import java.time.LocalDateTime;public class ClassWithPastDates {@Pastprivate LocalDate date;@Pastprivate LocalDateTime dateTime;public LocalDate getDate() {return date;}public void setDate(LocalDate date) {this.date = date;}public LocalDateTime getDateTime() {return dateTime;}public void setDateTime(LocalDateTime dateTime) {this.dateTime = dateTime;}}

接下来,我们将为@Past约束编写一个非常基本的单元测试,以证明我们的意图:显然,除了过去的日期之外,我们还希望null引用有效,但将来的日期无效,甚至今天也算作无效。

过去测试

package it.jdev.example.jsr310.validator;import static org.junit.Assert.assertEquals;import java.time.LocalDate;
import java.time.LocalDateTime;
import java.util.Set;import javax.validation.ConstraintViolation;
import javax.validation.Validation;
import javax.validation.Validator;
import javax.validation.ValidatorFactory;import org.junit.Before;
import org.junit.Test;public class PastTest {private ClassWithPastDates classUnderTest;@Beforepublic void setup() {classUnderTest = new ClassWithPastDates();}@Testpublic void thatNullIsValid() {Set<ConstraintViolation<ClassWithPastDates>> violations = validateClass(classUnderTest);assertEquals(violations.size(), 0);}@Testpublic void thatYesterdayIsValid() throws Exception {classUnderTest.setDate(LocalDate.now().minusDays(1));classUnderTest.setDateTime(LocalDateTime.now().minusDays(1));Set<ConstraintViolation<ClassWithPastDates>> violations = validateClass(classUnderTest);assertEquals(violations.size(), 0);}@Testpublic void thatTodayIsInvalid() throws Exception {classUnderTest.setDate(LocalDate.now());classUnderTest.setDateTime(LocalDateTime.now());Set<ConstraintViolation<ClassWithPastDates>> violations = validateClass(classUnderTest);assertEquals(violations.size(), 2);}@Testpublic void thatTomorrowIsInvalid() throws Exception {classUnderTest.setDate(LocalDate.now().plusDays(1));classUnderTest.setDateTime(LocalDateTime.now().plusDays(1));Set<ConstraintViolation<ClassWithPastDates>> violations = validateClass(classUnderTest);assertEquals(violations.size(), 2);}private Set<ConstraintViolation<ClassWithPastDates>> validateClass(ClassWithPastDates myClass) {ValidatorFactory factory = Validation.buildDefaultValidatorFactory();Validator validator = factory.getValidator();Set<ConstraintViolation<ClassWithPastDates>> violations = validator.validate(myClass);return violations;}}

现在我们已经建立了基本测试,我们可以实现约束本身。 这包括两个步骤。 首先,我们必须编写注释,然后必须实现ConstraintValidator 。 从注释开始:

@interface过去

package it.jdev.example.jsr310.validator;import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;import javax.validation.Constraint;
import javax.validation.Payload;@Target({ ElementType.FIELD })
@Retention(RetentionPolicy.RUNTIME)
@Constraint(validatedBy = PastValidator.class)
@Documented
public @interface Past {String message() default "it.jdev.example.jsr310.validator.Past.message";Class<?>[] groups() default {};Class<? extends Payload>[] payload() default {};}

如您所见, @Past注释不是很壮观。 要注意的主要事情是@Constraint批注,在其中我们指定将使用哪个类执行实际验证。

PastValidator

package it.jdev.example.jsr310.validator;import java.time.LocalDate;
import java.time.temporal.Temporal;import javax.validation.ConstraintValidator;
import javax.validation.ConstraintValidatorContext;public class PastValidator implements ConstraintValidator<Past, Temporal> {@Overridepublic void initialize(Past constraintAnnotation) {}@Overridepublic boolean isValid(Temporal value, ConstraintValidatorContext context) {if (value == null) {return true;}LocalDate ld = LocalDate.from(value);if (ld.isBefore(LocalDate.now())) {return true;}return false;}}

PastValidator是所有魔术发生的地方。 通过实现ConstraintValidator接口,我们不得不提供两种方法,但对于我们的示例,仅使用isValid()方法,这是我们执行实际验证的地方。

请注意,我们使用java.time.temporal.Temporal作为类型,因为它是LocalDate和LocalDateTime类共同拥有的接口。 这使我们可以对LocalDate和LocalDateTime字段使用相同的@Past

真正的全部就是它。 在这个非常基本的示例中,我展示了创建自己的定制JSR 303 bean验证约束是多么容易。

翻译自: https://www.javacodegeeks.com/2014/09/custom-jsr-303-bean-validation-constraints-for-the-jsr-310-new-datetime-api.html

JSR 310新日期/时间API的自定义JSR 303 Bean验证约束相关推荐

  1. jsr303自定义验证_JSR 310新日期/时间API的自定义JSR 303 Bean验证约束

    jsr303自定义验证 借助JSR 310,Java 8终于为我们带来了不错的日期和时间API. 对于仍在使用Java 7的那些人(就像我目前在我的当前项目中一样),有很好的反向移植,请访问www.t ...

  2. 高级JAVA开发必备技能:java8 新日期时间API((一)JSR-310:ZoneId 时区和偏移量)(JAVA 小虚竹)

    技术活,该赏 点赞再看,养成习惯 大家好,我是小虚竹.之前有粉丝私聊我,问能不能把JAVA8 新的日期时间API(JSR-310)知识点梳理出来.答案是肯定的,谁让我宠粉呢.由于内容偏多(超十万字了) ...

  3. JDK8新特性:新日期时间API全面学习

    旧版jdk对时间和日期的设计非常差 1.设计不合理 在java.util和Java.sql 包中都有日期类,java.util.Data同时包含日期和时间 而Java.sql.Date仅仅包含日期 此 ...

  4. 重学Java8新特性(四) : 日期时间API、LocalDateTime、DateTimeFormatter、开发中时间工具类(常用)

    文章目录 一.JDK8中日期时间API的介绍 1.1.LocalDate.LocalTime.LocalDateTime的使用 2.2.Instant类的使用 2.3.DateTimeFormatte ...

  5. Java日期时间API

    日期时间API 参考:https://lw900925.github.io/java/java8-newtime-api.html 旧日期时间API System java.lang.System类提 ...

  6. 6 日期字符串转日期_Java日期时间API系列6-----Jdk8中java.time包中的新的日期时间API类...

    因为Jdk7及以前的日期时间类的不方便使用问题和线程安全问题等问题,2005年,Stephen Colebourne创建了Joda-Time库,作为替代的日期和时间API.Stephen向JCP提交了 ...

  7. java 包结构 枚举类_Java日期时间API系列6-----Jdk8中java.time包中的新的日期时间API类...

    因为Jdk7及以前的日期时间类的不方便使用问题和线程安全问题等问题,2005年,Stephen Colebourne创建了Joda-Time库,作为替代的日期和时间API.Stephen向JCP提交了 ...

  8. java8 日期api_我们多么想要新的Java日期/时间API?

    java8 日期api 当前的Java.net 民意测验问题是:" 对于用Java 8实现的JSR-310(新的日期和时间API)有多重要? "在我撰写本文时,将近150位受访者投 ...

  9. 我们多么想要新的Java日期/时间API?

    当前的Java.net 民意测验问题是:" 对于用Java 8实现的JSR-310(新的日期和时间API)有多重要? "在我撰写本文时,将近150位受访者投了赞成票,绝大多数人回答 ...

最新文章

  1. Python机器学习——Agglomerative层次聚类
  2. ubuntu18.04.4 录制麦克风音频
  3. 高并发场景下的httpClient优化使用
  4. 且看且学Gradle--(1)Gradle入门
  5. MyBatis动态SQL,写SQL更爽
  6. C/Cpp / static 关键字
  7. Angular Jasmine单元测试用例spec.ts的加载逻辑
  8. MSSQL-字符串分离与列记录合并成一行混合使用
  9. css flexbox模型_Flexbox和CSS Grid之间的主要区别
  10. android 焦点分发,Android TV 焦点分发原理解析
  11. js-JavaScript高级程序设计学习笔记2
  12. springboot之提高maven的速度
  13. asp,jsp,php等web开发必懂技术---什么时候使用静态页,什么时候必须使用动态页...
  14. flask实现mysql连接池_mysql-Flask-SQLAlchemy-快速连接到多个数据库
  15. python反编译命令_Python 反编译模块dis
  16. 关注手机病毒:重点手机安全事件盘点
  17. 计算机方法学,浅谈计算机教学的方法
  18. 狄拉克函数和广义函数 | 线性泛函
  19. Nginx配置中的if判断
  20. mysql + excel 校正线上数据

热门文章

  1. JSP 统计网站访问人数
  2. 179. 最大数---LeetCode---JAVA
  3. 八皇后问题---回溯
  4. 基本属性---Linux
  5. ubuntu 13.04 mysql_Ubuntu13.04 下MySQL5.6安装过程
  6. datagridview绑定数据源不显示_sharding-jdbc系列之 数据源配置(一)
  7. java登录界面命令_Java命令行界面(第8部分):Argparse4j
  8. cloud foundry_将Spring Boot应用程序绑定到Cloud Foundry中的服务的方法
  9. Cassandra中的数据建模
  10. 带有NetBeans 10的Java EE