单向N--->1关联,比如在一个办公室里工作的多个人对应一个地址。为了让两个持久化类支持这种关联映射,程序应该在N的一端的持久化类中增加一个属性,该属性引用1的一端的关联实体。
我们先将test库里的表删除:
然后新建一个web工程,并编写代码:
Person.java :
public class Person {
private int id;
private String name;
private int age;
private Address address;
public int getId() {
return id;
}
public void setId(int id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
public Address getAddress() {
return address;
}
public void setAddress(Address address) {
this.address = address;
}
}
Address.java :
public class Address {
private int addressId;
private String addressDetail;
public int getAddressId() {
return addressId;
}
public void setAddressId(int addressId) {
this.addressId = addressId;
}
public String getAddressDetail() {
return addressDetail;
}
public void setAddressDetail(String addressDetail) {
this.addressDetail = addressDetail;
}
public Address(String addressDetail) {
super();
this.addressDetail = addressDetail;
}
}
Person.hbm.xml :
<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC
"-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<hibernate-mapping package="db.domain">
<class name="Person" table="persons">
<id name="id" type="integer">
<generator class="identity"></generator>
</id>
<property name="name" type="string">
<column name="name"></column>
</property>
<property name="age" type="integer">
<column name="age"></column>
</property>
<many-to-one name="address" class="Address" column="address_id" cascade="all"/>
</class>
</hibernate-mapping>
Person端增加了Address属性,该属性不是一个普通的组件属性,而是引用另一个持久化类的类,Hibernate使用
<many-to-one.../>元素映射N--->1的关联实体,直接采用<many-to-one.../>元素来映射关联实体将会在N的一端的数据表中增加一个外键列,用于参照主表记录。直接使用<many-to-one.../>元素来映射N--->1关联时,Hibernate将无需使用连接表,直接使用外键关联策略来处理这种关联映射。
Address.hbm.xml :
<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC
"-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<hibernate-mapping package="db.domain">
<class name="Address" table="address">
<id name="addressId" column="address_id">
<generator class="identity"></generator>
</id>
<property name="addressDetail" type="string">
<column name="addressDetail"></column>
</property>
</class>
</hibernate-mapping>
Test.java :
public class Test {
public static void main(String[] args) {
Session session=HibernateSessionFactory.getSession();
Transaction txt=session.beginTransaction();
Person p=new Person();
p.setName("tom");
p.setAge(20);
Address a1=new Address("广州天河");//①
p.setAddress(a1);
session.persist(p);//③
Address a2=new Address("上海虹口");//②
p.setAddress(a2);
txt.commit();
HibernateSessionFactory.closeSession();
}
}
上面程序创建了三个持久化实体,即一个Person对象,两个Address对象,程序只在③处保存了一次Person对象,从来不曾保存过Address对象。
程序在①处创建了一个瞬态的Address对象,当程序执行到③处时,系统准备保存Person对象,系统将要向persons表中插入一条记录------但该记录参照的主表记录还不曾保存(被参照的Address实体还处于瞬态),这时可能有如下两种情况发生:
① 系统抛出TransientObjectException异常:因为主表记录不曾插入,所以参照该记录的从表记录无法插入。
② 系统先自动级联插入主表记录,再插入从表记录。
因为上面的映射文件中指定了cascade="all",这意味着系统将先自动级联插入主表记录,也就是先持久化Address对象,再持久化Person对象。也就是说,Hibernate在③处将先执行一条insert into address...语句,再执行一条insert into persons...语句。换一个方向来说,如果上面的映射文件缺少cascade="all",则程序运行到③处将抛出TransientObjectException异常。
必须牢记:要么总是先持久化主表记录对应的实体,要么设置级联操作;否则当hibernate试图插入从表记录时,如果发现该从表记录参照的主表记录不存在,就会抛出异常。
运行Test.java,查看数据库: