1. 无连接表的N-1关联
对于无连接的N-1关联而言,程序只要在N的一端增加一列外键,让外键记录对象指所属的实体即可,Hibernate可使用@JoinColumn来修饰代表关联实体的属性,@JoinColumn用于映射底层的外键列。下面定义一个Person的类,Person与Address的关系就是N-1。
@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实体
@ManyToOne(targetEntity=Address.class)
// 映射外键列,指定外键列的列名为address_id、不允许为空
@JoinColumn(name="address_id" , nullable=false)
@Cascade(CascadeType.ALL)
private Address address;
// 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;
}
// address的setter和getter方法
public void setAddress(Address address)
{
this.address = address;
}
public Address getAddress()
{
return this.address;
}
}
程序无须从Address访问Person,所以Address无须增加Person属性。代码如下。
@Entity
@Table(name="address_inf")
public class Address
{
// 标识属性
@Id @Column(name="address_id")
@GeneratedValue(strategy=GenerationType.IDENTITY)
private int addressId;
// 定义地址详细信息的成员变量
private String addressDetail;
// 无参数的构造器
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和Address实体。
private void testCascase()
{
Session session = HibernateUtil.currentSession();
Transaction tx = session.beginTransaction();
// 创建一个Person对象
Person p = new Person();
// 创建一个瞬态的Address对象
Address a = new Address("广州天河"); // ①
// 设置Person的name为owen.org字符串
p.setName("owen.org");
p.setAge(21);
// 设置Person和Address之间的关联关系
p.setAddress(a);
// 持久化Person对象
session.persist(p);
// 创建一个瞬态的Address对象
Address a2 = new Address("上海虹口"); // ②
// 修改持久化状态的Person对象
p.setAddress(a2); // ③
tx.commit();
HibernateUtil.closeSession();
}
上面的实体类中使用了@Cascade(CascadeType.ALL)修饰代表关联实体的属性,这意味着系统先自动级联插入主表记录,也就是先持久化Address对象,再持久化Person对象。也就是说,Hibernate先执行一条insert into address…,再执行一条insert into person….语句。而对于p.setAddress(a2),系统先执行insert into address…,后执行update person….。如果没有@Cascade(CascadeType.ALL),则会报TransientObjectException异常。如下是执行结果图。
2. 有连接表的N-1关联
在有连接的情况下会生成一张中间表,这个时候我们只要修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实体
@ManyToOne(targetEntity=Address.class)
// 显式使用@JoinTable映射连接表
@JoinTable(name="person_address", // 指定连接表的表名为person_address
// 指定连接表中person_id外键列,参照到当前实体对应表的主键列
joinColumns=@JoinColumn(name="person_id"
, referencedColumnName="person_id", unique=true),
// 指定连接表中address_id外键列,参照到当前实体的关联实体对应表的主键列
inverseJoinColumns=@JoinColumn(name="address_id"
, referencedColumnName="address_id")
)
private Address address;
// 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;
}
// address的setter和getter方法
public void setAddress(Address address)
{
this.address = address;
}
public Address getAddress()
{
return this.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字符串
p.setName("owen");
p.setAge(21);
// 创建一个瞬态的Address对象
Address a = new Address("广州天河");
// 设置Person和Address之间的关联关系
p.setAddress(a);
// 再持久化Address对象
session.persist(a);
// 创建一个瞬态的Address对象
Address a2 = new Address("上海虹口");
// 设置Person和Address之间的关联关系
p.setAddress(a2);
// 由于采用了连接表来维护N-1关联关系,因此不存在主从表关系,
// 程序可以随意控制先持久化哪个实体。
// 持久化Address对象
session.persist(a2);
// 持久化Person对象
session.save(p);
// 持久化Address对象
session.save(a);
tx.commit();
HibernateUtil.closeSession();
}
}
执行结果如下。