hibernate一对一双向外键关联
一、双向关联
在单向关联的基础上,另一个Object也持有关联对象
二、场景
一夫一妻
三、R
CREATE TABLE `wife` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`name` varchar(50) NOT NULL,
PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=15 DEFAULT CHARSET=utf8
CREATE TABLE `husband` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`name` varchar(50) NOT NULL,
`wifeid` int(11) DEFAULT NULL,
PRIMARY KEY (`id`),
KEY `fk_wife` (`wifeid`),
CONSTRAINT `fk_wife` FOREIGN KEY (`wifeid`) REFERENCES `wife` (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=20 DEFAULT CHARSET=utf8
四、O
wife
package com.linys.model; /** * Wife entity. @author MyEclipse Persistence Tools */ public class Wife implements java.io.Serializable { // Fields /** * */ private static final long serialVersionUID = 1L; private Integer id; private String name; private Husband husband; // Constructors /** default constructor */ public Wife() { } /** minimal constructor */ public Wife(String name) { this.name = name; } // Property accessors public Integer getId() { return this.id; } public void setId(Integer id) { this.id = id; } public String getName() { return this.name; } public void setName(String name) { this.name = name; } public Husband getHusband() { return husband; } public void setHusband(Husband husband) { this.husband = husband; } }
husband
package com.linys.model; /** * Husband entity. @author MyEclipse Persistence Tools */ public class Husband implements java.io.Serializable { // Fields /** * */ private static final long serialVersionUID = 1L; private Integer id; private Wife wife; private String name; // Constructors /** default constructor */ public Husband() { } /** minimal constructor */ public Husband(String name) { this.name = name; } /** full constructor */ public Husband(Wife wife, String name) { this.wife = wife; this.name = name; } // Property accessors public Integer getId() { return this.id; } public void setId(Integer id) { this.id = id; } public Wife getWife() { return this.wife; } public void setWife(Wife wife) { this.wife = wife; } public String getName() { return this.name; } public void setName(String name) { this.name = name; } }
五、映射文件
Wife.hbm.xml
<?xml version="1.0" encoding="utf-8"?> <!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN" "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd"> <!-- Mapping file autogenerated by MyEclipse Persistence Tools --> <hibernate-mapping> <class name="com.linys.model.Wife" table="wife"> <id name="id" type="java.lang.Integer"> <column name="id" /> <generator class="native" /> </id> <property name="name" type="java.lang.String"> <column name="name" length="50" not-null="true" /> </property> <one-to-one name="husband" class="com.linys.model.Husband" property-ref="wife" cascade="all"/> </class> </hibernate-mapping>
其中one-to-one是和单向外键关联不同地方
one-to-one:指定在Wife这个Object类中用于双向关联的属性husband
property-ref: 在关联对象中用于与此Object对象关联的属性
cascade: 级联
级联:
1 只有“关系标记”才有cascade属性:many-to-one,one-to-one ,any,
set(map, bag, idbag, list, array) + one-to-many(many-to-many)
2 级联指的是当主控方执行操作时,关联对象(被动方)是否同步执行同一操作。
pojo和它的关系属性的关系就是“主控方 -- 被动方”的关系,如果关系属性是一个set,那么被动方就是set中的一个一个元素,。
比如:学校(School)有三个属性:地区(Address),校长(TheMaster)和学生(Set, 元素为Student)
执行session.delete(school)时,级联决定是否执行session.delete(Address),session.delete(theMaster),
是否对每个aStudent执行session.delete(aStudent)。
3 一个操作因级联cascade可能触发多个关联操作。前一个操作叫“主控操作”,后一个操作叫“关联操作”。
cascade属性的可选值:
all : 所有情况下均进行关联操作。
none:所有情况下均不进行关联操作。这是默认值。
save-update:在执行save/update/saveOrUpdate时进行关联操作。
delete:在执行delete时进行关联操作。
具体执行什么“关联操作”是根据“主控操作”来的:
“主控操作” “关联操作”
session.saveOrUpdate --> session.saveOrUpdate (执行saveOrUpdate实际上会执行save或者update)
session.save ----> session.saveOrUpdate
session.udpate --> session.saveOrUpdate
session.delete --> session.delete
4 主控操作和关联操作的先后顺序是“先保存one,再保存many;先删除many,再删除one;先update主控方,再update被动方”
对于one-to-one,当其属性constrained="false"(默认值)时,它可看作one-to-many关系;
当其属性constrained="true"时,它可看作many-to-one关系;
对many-to-many,它可看作one-to-many。
比如:学校(School)有三个属性:地区(Address),校长(TheMaster,其constrained="false")和学生(Set, 元素为Student)
当执行session.save(school)时,
实际的执行顺序为:session.save(Address);
session.save(school);
session.save(theMaster);
for( 对每一个student ){
session.save(aStudent);
}
当执行session.delete(school)时,
实际的执行顺序为:session.delete(theMaster);
for( 对每一个student ){
session.delete(aStudent);
}
session.delete(school);
session.delete(Address);
当执行session.update(school)时,
实际的执行顺序为:session.update(school);
session.saveOrUpdate(Address);
session.saveOrUpdate(theMaster);
for( 对每一个student ){
session.saveOrUpdate(aStudent);
}
注意:update操作因级联引发的关联操作为saveOrUpdate操作,而不是update操作。
saveOrUpdate与update的区别是:前者根据操作对象是保存了还是没有保存,而决定执行update还是save
extends: 实际中,删除学校不会删除地区,即地区的cascade一般设为false
总结:级联(cascade)就是操作一个对象时,对它的属性(其cascade=true)也进行这个操作。一般将级联加在one一方,因为比如one方的记录改了,则在many中的关联记录也要跟着变化
Husband.hbm.xml
<?xml version="1.0" encoding="utf-8"?> <!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN" "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd"> <!-- Mapping file autogenerated by MyEclipse Persistence Tools --> <hibernate-mapping> <class name="com.linys.model.Husband" table="husband"> <id name="id" type="java.lang.Integer"> <column name="id" /> <generator class="native" /> </id> <property name="name" type="java.lang.String"> <column name="name" length="50" not-null="true" /> </property> <many-to-one name="wife" class="com.linys.model.Wife" unique="true"> <column name="wifeid" /> </many-to-one> </class> </hibernate-mapping>
六、测试类
package com.linys.model; import org.hibernate.Session; import org.hibernate.SessionFactory; import org.hibernate.Transaction; import org.hibernate.cfg.Configuration; import org.junit.AfterClass; import org.junit.BeforeClass; import org.junit.Test; public class TestOne2OneUniFK { static SessionFactory sf; @BeforeClass public static void setUpBeforeClass() throws Exception { sf = new Configuration().configure().buildSessionFactory(); } @Test public void testOne2OneUniFK(){ Wife wife=new Wife(); wife.setName("wife"); Husband husband=new Husband(); husband.setName("husband"); husband.setWife(wife); wife.setHusband(husband); Session session=sf.openSession(); Transaction ts=session.beginTransaction(); //插入可以 // session.save(wife); // session.save(husband); //插入可以 // session.save(husband); // session.save(wife); //插入不行 // session.save(husband); //插入可以,存wife的同时,存husband session.save(wife); ts.commit(); } @AfterClass public static void tearDownAfterClass() throws Exception { sf.close(); } }