6.1.1 无连接表的单向N--->1关联

单向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,查看数据库:


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值