1:为什么要使用组件映射
答:建立关系数据模型的一个重要原则是在不会导致数据冗余的前提下,尽可能减少数据库表的数目及表之间的外键参照关系。以员工信息为例,员工信息中有员工的家庭地址信息,如果把地址信息单独放在一张表中,然后建立员工信息表和地址信息表之间的外键参照关系,当每次查询员工信息时,都需建立者两个表的连接。建立表的连接是很耗时的操作,为了提高数据库运行性能,可以把这两张表的信息整合在一张员工信息表EMPINFO中。
2:什么是组件映射:组件是一个被包含的对象,它和它的所有者同存于一张表中,也就是说它仅仅是个值类型,而不是一个实体。值类型和实体的区别在于值类型没有标识符,当然了持久化一个值类型也就不需要标识符属性
详见代码:
步骤一:
创建JavaBean
//步骤一:创建EmpHomeAddress和EmpInfo public class EmpHomeAddress { private String ehomestreet; private String ehomecity; private String ehomeprovince; private String ehomezipcode; private EmpInfo empinfo; } EmpInfo创建如下: public class EmpInfo { private Integer eid; private String ename; private EmpHomeAddress ehome; }
步骤二:创建配置文件EmpInfo.hbm.xml
<?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN" "http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd"> <hibernate-mapping package="cn.happy.component"> <class name="EmpInfo" table="EMPINFO"> <id name="eid" column="EID"> <generator class="native"></generator> </id> <property name="ename" column="ENAME" type="string"></property> <component name="ehome" class="EmpHomeAddress"> <parent name="empinfo"/> <property name="ehomestreet" column="EHOMESTREET" type="string"></property> <property name="ehomecity" column="EHOMECITY" type="string"></property> <property name="ehomeprovince" column="EHOMEPROVINCE" type="string"></property> <property name="ehomezipcode" column="EHOMEZIPCODE" type="string"></property> </component> </class> </hibernate-mapping>
在上面的EmpInfo.hbm.xml文件中,Address作为EmpInfo的组件被定义,共使用了两次,分别对应于EmpInfo的两个属性。根据配置文件知道,EmpInfo可以获得它的组件,儿Address却不能得到它的所有者,如果也想让Address得到它的所有者,可以在<component>中添加一个节点<parent name="EmpInfo",同时在Address的类中增加EmpInfo属性,并提供getter和setter方法即可使用如address.getEmpInfo()来操作EmpInfo类。
步骤三:测试代码
package Test; import org.hibernate.Session; import org.hibernate.Transaction; import org.junit.Test; import util.HibernateUtil; import entity.EmpHomeAddress; import entity.EmpInfo; public class MyTest { @Test public void Test() { Session session = HibernateUtil.currentSession(); Transaction tx = session.beginTransaction(); EmpInfo emp=new EmpInfo(); emp.setEname("张靓颖"); //创建一个员工地址对象 EmpHomeAddress address=new EmpHomeAddress(); address.setEhomecity("北京"); address.setEhomeprovince("北京"); address.setEhomestreet("五道口"); address.setEhomezipcode("100000"); address.setEmpinfo(emp); emp.setEhome(address); session.save(emp); tx.commit(); System.out.println("ok==="); } }
生成的sql语句以及表结构: