对于1-N关联,Hibernate推荐使用双向关联,而且不要让1的一端控制关联关系,而使用N的一端控制关联关系。双向的1-N关联与N-1关联是完全相同的两种情形,两端都需要增加对关联属性的访问,N的一端增加引用到关联实体的属性,1的一端增加集合属性,集合元素为关联关联实体。
无连接表的双向1-N关联,N的一端需要增加@ManyToOne注解来修饰代表关联实体的属性,而1的一端则需要使用@OneToMany注解来修饰代表关联实体的属性,且不能控制关联关系。而且使用@OneToMany注解时指定mappedBy属性——一旦为@OneToMany、@ManyToMany指定了该属性,则表明当前实体不能控制关联关系,这时Hibernate也不允许使用@JoinColumn或@JoinTable修饰代表关联实体的属性了。如下是Person类。
@Entity @Table(name="person_inf") public class Person { // 标识属性 @Id @Column(name="person_id") @GeneratedValue(strategy=GenerationType.IDENTITY) private Integer id; private String name; private int age; // 定义该Person实体所有关联的Address实体 // 指定mappedBy属性表明该Person实体不控制关联关系 @OneToMany(targetEntity=Address.class , mappedBy="person") private Set<Address> addresses = new HashSet<>(); // id的setter和getter方法 public void setId(Integer id) { this.id = id; } public Integer getId() { return this.id; } // name的setter和getter方法 public void setName(String name) { this.name = name; } public String getName() { return this.name; } // age的setter和getter方法 public void setAge(int age) { this.age = age; } public int getAge() { return this.age; } // addresses的setter和getter方法 public void setAddresses(Set<Address> addresses) { this.addresses = addresses; } public Set<Address> getAddresses() { return this.addresses; } }
下面是Address的类。
@Entity @Table(name="address_inf") public class Address { // 标识属性 @Id @Column(name="address_id") @GeneratedValue(strategy=GenerationType.IDENTITY) private int addressId; // 定义地址详细信息的成员变量 private String addressDetail; // 定义该Address实体关联的Person实体 @ManyToOne(targetEntity=Person.class) // 定义名为person_id外键列,该外键列引用person_inf表的person_id列。 @JoinColumn(name="person_id" , referencedColumnName="person_id" , nullable=false) private Person person; // 无参数的构造器 public Address() { } // 初始化全部成员变量的构造器 public Address(String addressDetail) { this.addressDetail = addressDetail; } // addressId的setter和getter方法 public void setAddressId(int addressId) { this.addressId = addressId; } public int getAddressId() { return this.addressId; } // addressDetail的setter和getter方法 public void setAddressDetail(String addressDetail) { this.addressDetail = addressDetail; } public String getAddressDetail() { return this.addressDetail; } // person的setter和getter方法 public void setPerson(Person person) { this.person = person; } public Person getPerson() { return this.person; } }
我们用如下的类,来保存Person与Address的类。
public class PersonManager { public static void main(String[] args) { PersonManager mgr = new PersonManager(); mgr.testPerson(); HibernateUtil.sessionFactory.close(); } private void testPerson() { Session session = HibernateUtil.currentSession(); Transaction tx = session.beginTransaction(); // 创建一个瞬态的Person对象 Person p = new Person(); // 设置Person的name为owen.org字符串 p.setName("owen.org"); p.setAge(29); // 持久化Person对象(对应于插入主表记录) session.save(p); // 创建一个瞬态的Address对象 Address a = new Address("广州天河"); // 先设置Person和Address之间的关联关系 a.setPerson(p); // 再持久化Address对象(对应于插入从表记录) session.persist(a); // 创建一个瞬态的Address对象 Address a2 = new Address("上海虹口"); // 先设置Person和Address之间的关联关系 a2.setPerson(p); // 再持久化Address对象(对应于插入从表记录) session.persist(a2); tx.commit(); HibernateUtil.closeSession(); } }
后台的SQL语句:
Hibernate: insert into person_inf (age, name) values (?, ?) Hibernate: insert into address_inf (addressDetail, person_id) values (?, ?) Hibernate: insert into address_inf (addressDetail, person_id) values (?, ?)
执行的结果如下:
Address类则需要使用@JoinTable映射连接表。
@Entity @Table(name="address_inf") public class Address { // 标识属性 @Id @Column(name="address_id") @GeneratedValue(strategy=GenerationType.IDENTITY) private int addressId; // 定义地址详细信息的成员变量 private String addressDetail; // 定义该Address实体关联的Person实体 @ManyToOne(targetEntity=Person.class) // 映射连接表,指定连接表为person_address @JoinTable(name="person_address", // 指定连接表中address_id列参照当前实体对应数据表的address_id主键列 joinColumns=@JoinColumn(name="address_id" , referencedColumnName="address_id", unique=true), // 指定连接表中person_id列参照当前实体的关联实体对应数据表的person_id主键列 inverseJoinColumns=@JoinColumn(name="person_id" , referencedColumnName="person_id") ) private Person person; // 无参数的构造器 public Address() { } // 初始化全部成员变量的构造器 public Address(String addressDetail) { this.addressDetail = addressDetail; } // addressId的setter和getter方法 public void setAddressId(int addressId) { this.addressId = addressId; } public int getAddressId() { return this.addressId; } // addressDetail的setter和getter方法 public void setAddressDetail(String addressDetail) { this.addressDetail = addressDetail; } public String getAddressDetail() { return this.addressDetail; } // person的setter和getter方法 public void setPerson(Person person) { this.person = person; } public Person getPerson() { return this.person; } }
对于Person的实体来说,也是可控制关联关系的,所以要去掉@OneToMany的注解mappedBy属性,并同时添加@JoinTable注解。
@Entity @Table(name="person_inf") public class Person { // 标识属性 @Id @Column(name="person_id") @GeneratedValue(strategy=GenerationType.IDENTITY) private Integer id; private String name; private int age; // 定义该Person实体所有关联的Address实体 // @OneToMany(targetEntity=Address.class ,mappedBy="person") @OneToMany(targetEntity=Address.class) // 映射连接表,指定连接表为person_address @JoinTable(name="person_address", // 指定连接表中person_id列参照当前实体对应数据表的person_id主键列 joinColumns=@JoinColumn(name="person_id" , referencedColumnName="person_id"), // 指定连接表中address_id列参照当前实体的关联实体对应数据表的address_id主键列 inverseJoinColumns=@JoinColumn(name="address_id" , referencedColumnName="address_id", unique=true) ) private Set<Address> addresses = new HashSet<>(); // id的setter和getter方法 public void setId(Integer id) { this.id = id; } public Integer getId() { return this.id; } // name的setter和getter方法 public void setName(String name) { this.name = name; } public String getName() { return this.name; } // age的setter和getter方法 public void setAge(int age) { this.age = age; } public int getAge() { return this.age; } // addresses的setter和getter方法 public void setAddresses(Set<Address> addresses) { this.addresses = addresses; } public Set<Address> getAddresses() { return this.addresses; } }
保存两个实体的类如下。
public class PersonManager { public static void main(String[] args) { PersonManager mgr = new PersonManager(); mgr.testPerson(); HibernateUtil.sessionFactory.close(); } private void testPerson() { Session session = HibernateUtil.currentSession(); Transaction tx = session.beginTransaction(); // 创建一个瞬态的Person对象 Person p = new Person(); // 设置Person的name为owen字符串 p.setName("owen"); p.setAge(21); // 创建一个瞬态的Address对象 Address a = new Address("广州天河"); // 设置Person和Address之间的关联关系 a.setPerson(p); //创建一个瞬态的Address对象 Address a2 = new Address("上海虹口"); // 设置Person和Address之间的关联关系 p.getAddresses().add(a2); // 由于采用了连接表来维护1-N关联关系,因此不存在主从表关系, // 程序可以随意控制先持久化哪个实体。 // 持久化Address对象 session.persist(a2); // 持久化Person对象 session.save(p); // 持久化Address对象 session.persist(a); tx.commit(); HibernateUtil.closeSession(); } }
执行的SQL语言如下。
Hibernate: insert into address_inf (addressDetail) values (?) Hibernate: insert into person_inf (age, name) values (?, ?) Hibernate: insert into address_inf (addressDetail) values (?) Hibernate: insert into person_address (person_id, address_id) values (?, ?) Hibernate: insert into person_address (person_id, address_id) values (?, ?)
执行结果如下: