java中的jpa
几天前,我在处理JPA中的LocalDateTime属性时遇到问题。 在这篇博客文章中,我将尝试创建一个样本问题来说明该问题以及我使用的解决方案。
考虑以下实体,该实体为特定公司的员工建模–
@Entity
@Getter
@Setter
public class Employee {@Id@GeneratedValueprivate Long id;private String name;private String department;private LocalDateTime joiningDate;
}
我使用的是Spring Data JPA,因此创建了以下存储库–
@Repository
public interface EmployeeRepository extends JpaRepository<Employee, Long> {}
我想找到在特定日期加入公司的所有员工。 为此,我从
JpaSpecificationExecutor –
@Repository
public interface EmployeeRepository extends JpaRepository<Employee, Long>,JpaSpecificationExecutor<Employee> {}
并写了如下查询
@SpringBootTest
@RunWith(SpringRunner.class)
@Transactional
public class EmployeeRepositoryIT {@Autowiredprivate EmployeeRepository employeeRepository;@Testpublic void findingEmployees_joiningDateIsZeroHour_found() {DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss");LocalDateTime joiningDate = LocalDateTime.parse("2014-04-01 00:00:00", formatter);Employee employee = new Employee();employee.setName("Test Employee");employee.setDepartment("Test Department");employee.setJoiningDate(joiningDate);employeeRepository.save(employee);// Query to find employeesList<Employee> employees = employeeRepository.findAll((root, query, cb) ->cb.and(cb.greaterThanOrEqualTo(root.get(Employee_.joiningDate), joiningDate),cb.lessThan(root.get(Employee_.joiningDate), joiningDate.plusDays(1))));assertThat(employees).hasSize(1);}
}
以上测试顺利通过。 但是,以下测试失败了(应该通过了)–
@Test
public void findingEmployees_joiningDateIsNotZeroHour_found() {DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss");LocalDateTime joiningDate = LocalDateTime.parse("2014-04-01 08:00:00", formatter);LocalDateTime zeroHour = LocalDateTime.parse("2014-04-01 00:00:00", formatter);Employee employee = new Employee();employee.setName("Test Employee");employee.setDepartment("Test Department");employee.setJoiningDate(joiningDate);employeeRepository.save(employee);List<Employee> employees = employeeRepository.findAll((root, query, cb) ->cb.and(cb.greaterThanOrEqualTo(root.get(Employee_.joiningDate), zeroHour),cb.lessThan(root.get(Employee_.joiningDate), zeroHour.plusDays(1))));assertThat(employees).hasSize(1);
}
与之前的测试唯一不同的是,在先前的测试中,我将零小时作为加入日期,而在这里,我使用了上午8点。 起初,我觉得很奇怪。 只要将员工的入职日期设置为一天的零小时,测试似乎就会通过,但是如果将其设置为任何其他时间,则测试都会失败。
为了调查问题,我打开了Hibernate日志记录,以查看实际查询和发送到数据库的值,并在日志中注意到了类似的内容–
2017-03-05 22:26:20.804 DEBUG 8098 --- [ main] org.hibernate.SQL:selectemployee0_.id as id1_0_,employee0_.department as departme2_0_,employee0_.joining_date as joining_3_0_,employee0_.name as name4_0_fromemployee employee0_whereemployee0_.joining_date>=?and employee0_.joining_dateHibernate:selectemployee0_.id as id1_0_,employee0_.department as departme2_0_,employee0_.joining_date as joining_3_0_,employee0_.name as name4_0_fromemployee employee0_whereemployee0_.joining_date>=?and employee0_.joining_date2017-03-05 22:26:20.806 TRACE 8098 --- [ main] o.h.type.descriptor.sql.BasicBinder : binding parameter [1] as [VARBINARY] - [2014-04-01T00:00]2017-03-05 22:26:20.807 TRACE 8098 --- [ main] o.h.type.descriptor.sql.BasicBinder : binding parameter [2] as [VARBINARY] - [2014-04-02T00:00]
显然,JPA 并未将joinDate属性视为日期或时间,而是视为VARBINARY类型。 这就是与实际日期比较失败的原因。
我认为这不是一个很好的设计。 它没有抛出UnsupportedAttributeException之类的东西,而是默默地尝试将值转换为其他值,从而导致随机比较失败(不是完全随机)。 除非您有很强的自动化测试套件,否则在应用程序中很难找到这种类型的错误,幸运的是,我的情况是这样。
现在回到问题。 JPA无法正确转换LocalDateTime的原因非常简单。 JPA规范的最新版本(即2.1)是在Java 8之前发布的,因此它无法处理新的Date and Time API。
为了解决该问题,我创建了一个自定义转换器实现,该实现将LocalDateTime转换为
将java.sql.Timestamp保存到数据库之前,反之亦然。 那解决了问题–
@Converter(autoApply = true)
public class LocalDateTimeConverter implements AttributeConverter<LocalDateTime, Timestamp> {@Overridepublic Timestamp convertToDatabaseColumn(LocalDateTime localDateTime) {return Optional.ofNullable(localDateTime).map(Timestamp::valueOf).orElse(null);}@Overridepublic LocalDateTime convertToEntityAttribute(Timestamp timestamp) {return Optional.ofNullable(timestamp).map(Timestamp::toLocalDateTime).orElse(null);}
}
每当我尝试保存LocalDateTime属性时,上述转换器将自动应用。 我还可以使用来显式标记要显式转换的属性
javax.persistence.Convert注释–
@Convert(converter = LocalDateTimeConverter.class)
private LocalDateTime joiningDate;
完整的代码可以在Github上找到 。
翻译自: https://www.javacodegeeks.com/2017/03/dealing-javas-localdatetime-jpa.html
java中的jpa