原文: https://howtodoinjava.com/hibernate/immutable-and-naturalid-hibernate-specific-annotations/
在上一篇文章中,我们了解了Hiberate中最常用的 JPA 注解,还了解了与这些 JPA 注解互补的 Hiberate 注解。 使用 JPA 注解使您的应用代码可移植到其他 JPA 实现中,这是一件好事。 除了 JPA 注解之外,Hiberate 还具有一些自己的注解,您可以使用它们在应用代码中具有某些功能。 但是您必须记住,将来的日期可能很难使您的代码可移植。
阅读更多: JPA2 持久化注解教程
在本文中,我们将学习两个特定于 Hibernate 的此类注解。
1)@Immutable注解
@Immutable注解将一个实体标记为不可变。 这在您的实体表示参考数据的情况下非常有用,例如状态列表,性别或其他很少突变的数据。
由于状态之类的东西很少会更改,因此通常有人会通过 SQL 或管理应用手动更新数据。 Hibernate 可以积极地缓存此数据,需要将其考虑在内。 如果参考数据发生变化,则需要确保已通知使用该数据的应用(可以使用refresh()方法)或以某种方式重新启动。
@Immutable注解告诉 Hibernate,对不可变实体的任何更新都不应传递给数据库而不会给出任何错误。 @Immutable也可以放在集合上; 在这种情况下,对集合的更改(添加或删除)将引发HibernateException。
EmployeeEntity.java
import org.hibernate.annotations.Immutable;@Immutable@Entity@Table(name = "Employee")public class EmployeeEntity implements Serializable{private static final long serialVersionUID = -1798070786993154676L;@Id@Column(name = "ID", unique = true, nullable = false)private Integer employeeId;@Column(name = "FIRST_NAME", unique = false, nullable = false, length = 100)private String firstName;@Column(name = "LAST_NAME", unique = false, nullable = false, length = 100)private String lastName;//Setters and Getters}
ImmutableAnnotationExample.java
public class ImmutableAnnotationExample{public static void main(String[] args){setupTestData();Session sessionOne = HibernateUtil.getSessionFactory().openSession();sessionOne.beginTransaction();//Load the employee in another sessionEmployeeEntity employee = (EmployeeEntity) sessionOne.load(EmployeeEntity.class, 1);//Update the first nameemployee.setFirstName("Alex");sessionOne.flush();sessionOne.close();Session sessionTwo = HibernateUtil.getSessionFactory().openSession();sessionTwo.beginTransaction();//Load the employee in another sessionEmployeeEntity employeeUpdated = (EmployeeEntity) sessionTwo.load(EmployeeEntity.class, 1);//Verify the first nameSystem.out.println(employeeUpdated.getFirstName());sessionTwo.flush();sessionTwo.close();HibernateUtil.shutdown();}private static void setupTestData(){Session session = HibernateUtil.getSessionFactory().openSession();session.beginTransaction();//Create EmployeeEmployeeEntity emp = new EmployeeEntity();emp.setEmployeeId(1);emp.setFirstName("Lokesh");emp.setLastName("Gupta");session.save(emp);session.getTransaction().commit();session.close();}}Output:Hibernate: insert into Employee (FIRST_NAME, LAST_NAME, ID) values (?, ?, ?)Hibernate: select employeeen0_.ID as ID1_1_0_, employeeen0_.FIRST_NAME as FIRST_NA2_1_0_, employeeen0_.LAST_NAME as LAST_NAM3_1_0_from Employee employeeen0_ where employeeen0_.ID=?Hibernate: select employeeen0_.ID as ID1_1_0_, employeeen0_.FIRST_NAME as FIRST_NA2_1_0_, employeeen0_.LAST_NAME as LAST_NAM3_1_0_from Employee employeeen0_ where employeeen0_.ID=?Lokesh //Value didn't updated in database i.e. immutable
2)@NaturalId注解
在过去的教程中,我们了解了很多有关@Id和@GeneratedValue注解以为数据库中的记录创建主键的知识。 在大多数实际应用中,这些主键是“人工主键”,并且仅在应用运行时内部引用。 但是,还存在“自然 ID”的概念,除了人工或复合主键之外,它还提供了另一种方便且合乎逻辑的方式来引用实体。
自然 ID 的示例可能是美国的社会安全号码或税号,而印度则是 PAN 号码。 实体(是个人或公司)可能具有由 Hibernate 生成的人为主键,但也可能具有唯一的税标识符。 Hibernate 还允许您基于这些自然 ID 搜索和加载实体。
对于自然 ID,有两种形式的加载机制: 一种使用简单自然 ID(其中自然 ID 是一个且仅一个字段),另一种使用命名属性作为复合自然 ID 的一部分。
简单的自然 ID 示例
@Entity@Table(name = "Employee")public class EmployeeEntity implements Serializable{private static final long serialVersionUID = -1798070786993154676L;@Id@Column(name = "ID", unique = true, nullable = false)private Integer employeeId;@Column(name = "FIRST_NAME", unique = false, nullable = false, length = 100)private String firstName;@Column(name = "LAST_NAME", unique = false, nullable = false, length = 100)private String lastName;//Natural id can be SSN as well@NaturalIdInteger SSN;//Setters and Getters}
SimpleNaturalIdExample.java
public class SimpleNaturalIdExample{public static void main(String[] args){setupTestData();Session sessionOne = HibernateUtil.getSessionFactory().openSession();sessionOne.beginTransaction();//Load the employeeEmployeeEntity employee1 = (EmployeeEntity) sessionOne.load(EmployeeEntity.class, 1);//Just to ensure that employee is loasded from DBSystem.out.println(employee1.getFirstName());//Get the employee for natural id i.e. SSN; This does not execute another SQL SELECT as entity is already present in sessionEmployeeEntity employee2 = (EmployeeEntity) sessionOne.bySimpleNaturalId(EmployeeEntity.class).load(12345);//Verify that employee1 and employee2 refer to same objectassert(employee1 == employee2);sessionOne.flush();sessionOne.close();System.out.println("====================================");Session sessionTwo = HibernateUtil.getSessionFactory().openSession();sessionTwo.beginTransaction();//Get the employee for natural id i.e. SSN; entity is not present in this sessionEmployeeEntity employee = (EmployeeEntity) sessionTwo.bySimpleNaturalId(EmployeeEntity.class).load(12345);sessionTwo.flush();sessionTwo.close();HibernateUtil.shutdown();}private static void setupTestData(){Session session = HibernateUtil.getSessionFactory().openSession();session.beginTransaction();//Create EmployeeEmployeeEntity emp = new EmployeeEntity();emp.setEmployeeId(1);emp.setFirstName("Lokesh");emp.setLastName("Gupta");emp.setSSN(12345);session.save(emp);session.getTransaction().commit();session.close();}}Output:Hibernate: insert into Employee (SSN, FIRST_NAME, LAST_NAME, ID) values (?, ?, ?, ?)Hibernate: select employeeen0_.ID as ID1_1_0_, employeeen0_.SSN as SSN2_1_0_, employeeen0_.FIRST_NAME as FIRST_NA3_1_0_,employeeen0_.LAST_NAME as LAST_NAM4_1_0_ from Employee employeeen0_ where employeeen0_.ID=?Lokesh====================================Hibernate: select employeeen_.ID as ID1_1_ from Employee employeeen_ where employeeen_.SSN=?Hibernate: select employeeen0_.ID as ID1_1_0_, employeeen0_.SSN as SSN2_1_0_, employeeen0_.FIRST_NAME as FIRST_NA3_1_0_,employeeen0_.LAST_NAME as LAST_NAM4_1_0_ from Employee employeeen0_ where employeeen0_.ID=?
请密切注意如果会话中不存在实体,并且如果您使用实体的自然 ID 获取实体,则使用自然 ID 提取第一个主要 ID; 然后使用此主要 ID 提取实体。 如果会话中已经存在实体,则返回相同实体的引用,而无需在数据库中执行其他SELECT语句。
复合自然 ID 示例
@Entity@Table(name = "Employee")public class EmployeeEntity implements Serializable{private static final long serialVersionUID = -1798070786993154676L;@Id@Column(name = "ID", unique = true, nullable = false)private Integer employeeId;@Column(name = "FIRST_NAME", unique = false, nullable = false, length = 100)private String firstName;@Column(name = "LAST_NAME", unique = false, nullable = false, length = 100)private String lastName;//Natural id part 1@NaturalIdInteger seatNumber;//Natural id part 2@NaturalIdString departmentName;//Setters and Getters}
CompositeNaturalIdExample.java
public class CompositeNaturalIdExample{public static void main(String[] args){setupTestData();Session sessionOne = HibernateUtil.getSessionFactory().openSession();sessionOne.beginTransaction();//Get the employee for natural id i.e. SSN; entity is not present in this sessionEmployeeEntity employee = (EmployeeEntity) sessionOne.byNaturalId(EmployeeEntity.class).using("seatNumber", 12345).using("departmentName", "IT").load();System.out.println(employee.getFirstName());sessionOne.flush();sessionOne.close();HibernateUtil.shutdown();}private static void setupTestData(){Session session = HibernateUtil.getSessionFactory().openSession();session.beginTransaction();//Create EmployeeEmployeeEntity emp = new EmployeeEntity();emp.setEmployeeId(1);emp.setFirstName("Lokesh");emp.setLastName("Gupta");emp.setSeatNumber(12345);emp.setDepartmentName("IT");session.save(emp);session.getTransaction().commit();session.close();}}Output:Hibernate: insert into Employee (departmentName, FIRST_NAME, LAST_NAME, seatNumber, ID) values (?, ?, ?, ?, ?)Hibernate: select employeeen_.ID as ID1_1_ from Employee employeeen_ where employeeen_.departmentName=? and employeeen_.seatNumber=?Hibernate: select employeeen0_.ID as ID1_1_0_, employeeen0_.departmentName as departme2_1_0_, employeeen0_.FIRST_NAME as FIRST_NA3_1_0_,employeeen0_.LAST_NAME as LAST_NAM4_1_0_, employeeen0_.seatNumber as seatNumb5_1_0_ from Employee employeeen0_ where employeeen0_.ID=?Lokesh
复合自然 ID 的实体获取逻辑与简单自然 ID 相同。 除了使用多个自然键而不是一个以外,没有区别。
这些就是所有这些众所周知的注解。 继续在评论中发表您的想法。
祝您学习愉快!
