【Java EE 学习 45】【Hibernate学习第二天】【对象的三种状态】【一对多关系的操作】
一、对象的三种状态。
1.对象有三种状态:持久化状态、临时状态、脱管状态(游离状态)
2.Session的特定方法能使得一个对象从一个状态转换到另外一个状态。
3.三种状态的说明
(1)临时状态:临时状态的最明显标识就是它没有被保存到session中。
一下情况下,java对象进入临时状态:当通过new语句刚刚创建了一个Java对象,它处于临时状态,此时不和数据库中的任何记录对应。
(2)持久化状态:持久化状态的明显标识就是被保存到了session的缓存中;持久化状态的对象和数据库中的相关记录对应;Session在清理缓存的时候,会根据持久化对象的属性变化,来同步更新数据库;在同一个Session的缓存中,数据库表中的每条记录只对应唯一的持久化对象。
Session的很多方法都能够触发java对象进入持久化状态。
* Session的save方法能够把临时对象转变成持久化对象
* Session的load方法或者get方法返回的对象总是处于持久化状态
* Session的update、saveOrUpdate和lock方法能够使得游离的对象转变成持久化对象。
* 当一个持久化对象关联一个临时对象,在允许级联保存的情况下,Session在清理缓存的时候会把这个临时对象也转变成为持久化对象。
(3)脱管状态(游离状态):处于该种状态的对象不再存在于Session缓存中;一般情况下,游离状态的对象是由持久化状态的对象转变过来的,因此数据库中还可能存在与之对应的记录。
Session调用一下的方法能够使得对象从持久化状态转变成为脱管状态。
* 当调用Session的close方法的时候,Session的缓存会被清空,缓存中所有的持久化状态对象都会转变成为脱管状态的对象。如果在应用程序中没有引用变量引用这些游离对象,这个对象就会结束生命周期。
* Session的evict方法能够从缓存中删除一个持久化状态对象,使得它变成游离状态。当Session的缓存中保存了大量的持久化状态对象的时候,会消耗掉很多内存,为了提高性能,可以考虑调用evict方法,从缓存中删除掉一些持久化对象。
4.状态转换图
二、一对多关系操作
这里以班级和学生之间的一对多的关系为例。数据库表创建和测试实例:
1 DROP TABLE IF EXISTS student; 2 DROP TABLE IF EXISTS classes; 3 /* 4 创建班级表 5 */ 6 CREATE TABLE classes( 7 cid BIGINT, 8 cname VARCHAR(32), 9 CONSTRAINT classes_pk PRIMARY KEY(cid) 10 ); 11 INSERT INTO classes(cid,cname) VALUES(1,'一年级二班'); 12 INSERT INTO classes(cid,cname) VALUES(2,'二年级一班'); 13 INSERT INTO classes(cid,cname) VALUES(3,'三年级四班'); 14 /* 15 创建学生表 16 */ 17 CREATE TABLE student( 18 sid BIGINT, 19 sname VARCHAR(32), 20 cid BIGINT, 21 CONSTRAINT student_pk PRIMARY KEY(sid), 22 CONSTRAINT student_fk FOREIGN KEY (cid) REFERENCES classes(cid) 23 ); 24 INSERT INTO student(sid,sname,cid) VALUES(1,'张三',1); 25 INSERT INTO student(sid,sname,cid) VALUES(2,'李四',1); 26 INSERT INTO student(sid,sname,cid) VALUES(3,'王五',2); 27 INSERT INTO student(sid,sname,cid) VALUES(4,'赵六',2); 28 INSERT INTO student(sid,sname,cid) VALUES(5,'陈七',3); 29 INSERT INTO student(sid,sname,cid) VALUES(6,'小强',3);
1.单向关联:
分为单向1-n和单向n-1关联两种。
单向1-n关联:只能从1的一端访问到n的一端,反过来则不可以;举例,只能从班级对象获取所有关联的学生对象,而不能根据现有的学生对象获取其所在的班级信息。
单向n-1关联:只能从n的一端访问到1的一端,反过来则不可以;举例,只能从学生对象获取所有关联的班级对象,而不能根据现有的班级对象那个获取其所有学生。
2.持久化类和映射文件应当进行适当的配置。
持久化类:在一的一端应当添加Set集合成员变量。如在Classes类中应当添加Set<Student>变量,并设置setter和getter方法。
在多的一端应当添加一的的一端对应的Class成员变量。如在Student一端应当添加Classes classes,并设置setter和getter方法;
映射文件,在一的一端应当配置<set>标签以标志其关联的对象,在多的一端应当配置<many-to-one>标签以标志其关联的对象。
3.映射文件中添加set标签(class标签中)
<set name="students" cascade="save-update"> <key column="cid"></key> <one-to-many class="com.mysql.hibernate.domain.Student"/> </set>
set标签中的几个属性:
name:对应的持久化类中的set集合的属性名
cascade:表示级联操作的类型,save-update/all
table:表示对应的数据库表名。
inverse:表示是否要维护关系,注意和cascade之间的区别,如果是false表示维护关系,如果是true表示不维护关系。
key标签:指明关联表的外键名。这里关联表Student的外键是cid。应当注意填写的应当是Student中的外键名,而不是Classes的主键名。
one-to-manay标签:class属性指明了关联对象的类型,这里表示Calsses对象关联的对象类型是Student
4.配置文件中应当进行适当的配置
在hibernate.cfg.xml配置文件中应当进行相应的配置以便对映射文件进行注册。
使用标签mapping(session-factory标签下):
<mapping resource="com/kdyzm/moretoone/config/Classes.hbm.xml" /> <mapping resource="com/kdyzm/moretoone/config/Student.hbm.xml" />
三、多对一关系操作
1.多对一关系中,应当在持久化类中添加一的一端的类型属性,并设置set和get方法
2.在映射文件中进行相应的配置。
<!-- 多对一标签没有inverse属性,但是还有cascade属性,理解为什么没有inverse属性 --> <many-to-one name="classes" class="com.kdyzm.moretoone.domain.Classes"> <!-- 描述外键的,表示通过该字段建立起和Classes之间的关系 --> <column name="cid"></column> </many-to-one>
3.标签中的相应属性的说明
many-to-one标签的name属性:标识当前类中成员变量的名称,class属性标识成员变量所属的类别。
column标签:name属性标识外键的名称。
4.注意事项
many-to-one标签没有inverse属性,因为通过对当前持久化类对应的持久化表进行的CRUD操作就已经在维护关系了,所以不需要也必须不能有该属性。
many-to-one标签的cascade属性:当设置为save-updat的时候就会自动持久化关联的对象,默认不设置,如果不设置的话,就会在持久化当前对象的时候抛出异常:TransientObjectException
四、inverse属性
1.在hibernate中inverse属性的取值决定了是由双向关联的哪一方维护表和表之间的关系。inverse=false为主动方,inverse=true为被动方,由主动方负责维护关联关系。
2.在没有设置inverse属性的时候,两边默认都是false,即默认都维护关系。
3.设置原则
在1-n关系中,将n放设置为主动方将有助于改善性能。举例,如果让国家元首记住全国人民的名字是非常困难的,但是要让全国人民都记住国家元首的名字就要简单得多。
实际上效率提高的真正原因是使用这种原则会少发出update语句。
4.在建立两个对象的关联的时候,应当同时修改关联两端的响应属性
classes.getStudents.add(student);
student.setClasses(classes);
5.在删除双向关联关系的时候,也应该修改关联两端的对象属性
classes.getStudents.remove(student);
student.setClasses(null);
五、cascade属性的取值说明