开篇前言
上篇博文【SSH进阶之路】hibernate映射——一对一单向关联映射,小编介绍了一对一的单向关联映射,单向是指只能从人(Person)这端加载身份证端(IdCard),但是反过来,不能从身份证端加载人得信息。为什么呢,因为对象模型具有方向性,在前面的博文Hibernate基本映射中,小编介绍了单向和双向,所谓的单向就是一段只能加载另一个端,不能反过来;双向就是两端都可以加载另一端,可以这样来理解,单向就是一厢情愿,双向就是两情相悦。ok,那么问题来了,如果我们想从IdCard这端加载Person,我们要怎么办呢?今天的博文,小编就来简单介绍一下一对一双向关联映射,同样的,一对一双向关联映射也存在两种策略,主键关联和唯一外键关联,小编依然会配合相应的demo进行讲解,不足之处,还请小伙伴们多多指教。
映射原理
双向关联映射与单向关联映射的原理是一样的,双向关联映射并不影响存储,只影响加载。所以,双向关联映射和单向关联映射的关系模型是一样的即数据库的表结构是一样的,只是IdCard的实体类和配置文件(IdCard.hbm.xml)发生了一点点变化。无论是单向关联映射还是双向关联映射,他们都属于一对一关联映射,只是他们主键的生成策略不同,分为主键关联映射和唯一外键关联映射。小编依然从uml图入手,介绍对象模型以及关系模型,首先对象如下如下所示:
通过uml图以及联系实际生活,我们可以很容易知道,一个人只有一张身份证,并且有唯一的一个身份证号,Person和IdCard之间的关系是一对一的关系,两个对象之间的关系维护还是由person端来决定,因为关系只能由一端维护主键,否则不是乱套了`(*∩_∩*)′。ok,接着我们来编写相应的代码部分。
第一步、建立两个实体类Person 和 IdCard并且生成相应的get和set方法,代码如下所示:
package com.bjpowernode.hibernate; public class Person { private int id; private String name; private IdCard idCard; public int getId() { return id; } public IdCard getIdCard() { return idCard; } public void setIdCard(IdCard idCard) { this.idCard = idCard; } public void setId(int id) { this.id = id; } public String getName() { return name; } public void setName(String name) { this.name = name; } }
IdCard类代码如下所示:
package com.bjpowernode.hibernate; public class IdCard { private int id; private String cardNo; private Person person; public Person getPerson() { return person; } public void setPerson(Person person) { this.person = person; } public int getId() { return id; } public void setId(int id) { this.id = id; } public String getCardNo() { return cardNo; } public void setCardNo(String cardNo) { this.cardNo = cardNo; } }
从上面的代码我们可以看出来,Person端没有变化,但是我们在IdCard端加上了Person的引用。
第二步、编写Person.hbm.xml和IdCard.hbm.xml文件里面的代码,首先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> <class name="com.bjpowernode.hibernate.Person" table="t_person"> <id name="id"> <!-- 采用foreign生成策略,forgeign会取得关联对象的标识 --> <generator class="foreign"> <!-- property只关联对象 --> <param name="property">idCard</param> </generator> </id> <property name="name"/> <one-to-one name="idCard" constrained="true"/> </class> </hibernate-mapping>
IdCard.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> <class name="com.bjpowernode.hibernate.IdCard" table="t_idCard"> <id name="id"> <generator class="native"/> </id> <property name="cardNo"/> <one-to-one name="person"/> </class> </hibernate-mapping>
第三步、编写hibernate.cfg.xml的代码,建立数据库并且添加相应的映射,代码如下所示:
<!DOCTYPE hibernate-configuration PUBLIC "-//Hibernate/Hibernate Configuration DTD 3.0//EN" "http://hibernate.sourceforge.net/hibernate-configuration-3.0.dtd"> <hibernate-configuration> <session-factory > <!-- MySql数据库驱动 --> <property name="hibernate.connection.driver_class">com.mysql.jdbc.Driver</property> <!-- 数据库名称 --> <property name="hibernate.connection.url"> jdbc:mysql:///hibernate_one2one_pk_2</property> <!-- 数据库的用户名 --> <property name="hibernate.connection.username">root</property> <!-- 数据库的密码 --> <property name="hibernate.connection.password">123456</property> <property name="hibernate.dialect">org.hibernate.dialect.MySQLDialect</property> <!-- 显示语句 --> <property name="hibernate.show_sql">true</property> <!-- 格式排版 --> <!-- <property name="hibernate.format_sql">true</property> --> <mapping resource="com/bjpowernode/hibernate/Person.hbm.xml"/> <mapping resource="com/bjpowernode/hibernate/IdCard.hbm.xml"/> </session-factory> </hibernate-configuration>
接着创建数据库,并且运行ExportDB,相应的表结构如下所示:
从上面的截图可以看出,同时生成了personId作为主键和外键,同时限制约束了两张表的主键相同唯一,ok,数据库已经建立好了,相应的表也已经生成,现在小编来编写相应的方法,向数据库中插入数据,运行效果如下所示:
上面介绍的是主键关联,小编开始介绍第二种唯一外键关联,双向关联的唯一外键关联和单向唯一外键关联映射类似,在其一对一的指向端也就是Person端存在一个唯一外键,该唯一外键与被指向端(IdCard)相关联,关联主键的值相同,ok,接着介绍第二种唯一外键关联,同一对一单向关联映射类似,主键关联即利用主键进行关联,关联主键的值相同。下面我们看一下映射文件,首先我们来看IdCard.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> <class name="com.bjpowernode.hibernate.IdCard" table="t_idCard"> <id name ="id"> <generator class="native"/> </id> <property name="cardNo"/> <one-to-one name="person" property-ref="idCard"/> </class> </hibernate-mapping>
接着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> <class name="com.bjpowernode.hibernate.Person" table="t_person"> <id name="id"> <!-- 采用foreign生成策略,forgeign会取得关联对象的标识 --> <generator class="foreign"> <!-- property只关联对象 --> <param name="property">idCard</param> </generator> </id> <property name="name"/> <!-- one-to-one指示hibernate如何加载其关联对象,默认根据主键加载 也就是拿到关系字段值,根据对端的主键来加载关联对象 constrained="true表示,当前主键(person的主键)还是一个外键 参照了对端的主键(IdCard的主键),也就是会生成外键约束语句 --> <one-to-one name="idCard" constrained="true"/> </class> </hibernate-mapping>
接着创建数据库,并且运行ExportDB,相应的表结构如下所示:
小编寄语:该博文小编主要简单的介绍了一对一双向关联映射, 一对一单向和双向关联映射的区别正是对象模型和关系模型的区别之一。对象模型,有方向性。她到底是单向还是双向是由对象模型决定的即配置文件决定。关系模型,没有方向性或者说是双向的。从任何一端都可以加载另一端。在下篇博文中,小编将继续介绍hibernate映射的相关知识,敬请期待……