持久化对象的 四种状态

持久化对象的状态




持久化对象的 四种状态_第1张图片









~~~~~~~~~~~~~~~~~~~~~~~~~~~分割线~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

1. 项目结构


2. hibernate.cfg.xml

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE hibernate-configuration PUBLIC
		"-//Hibernate/Hibernate Configuration DTD 3.0//EN"
		"http://hibernate.sourceforge.net/hibernate-configuration-3.0.dtd">
<hibernate-configuration>

    <session-factory>
    	<property name="hibernate.connection.username">root</property>
    	<property name="hibernate.connection.password">chuck</property>
    	<property name="hibernate.connection.driver_class">com.mysql.jdbc.Driver</property>
    	<property name="hibernate.connection.url">jdbc:mysql://localhost:3306/hibernate</property>
    	
    	<property name="hibernate.dialect">org.hibernate.dialect.MySQLInnoDBDialect</property>
    	<property name="hibernate.show_sql">true</property>
    	<property name="hibernate.format_sql">true</property>
    	<property name="hibernate.hbm2ddl.auto">update</property>
    	<property name="hibernate.connection.isolation">2</property>
    	
    	
    	<mapping resource="com/baidu/hibernate/app/Person.hbm.xml"  />
    	
    	
    </session-factory>
    
</hibernate-configuration>
3.持久化类 Person.java
package com.baidu.hibernate.app;

import java.util.Date;

public class Person {
	
	private Integer id;
	private String name;
	private String interest;
	private Date birth;
	public Integer getId() {
		return id;
	}
	public void setId(Integer id) {
		this.id = id;
	}
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	public String getInterest() {
		return interest;
	}
	public void setInterest(String interest) {
		this.interest = interest;
	}
	public Date getBirth() {
		return birth;
	}
	public void setBirth(Date birth) {
		this.birth = birth;
	}
	public Person() {
		super();
	}
	public Person(String name, String interest, Date birth) {
		super();
		this.name = name;
		this.interest = interest;
		this.birth = birth;
	}
	@Override
	public String toString() {
		return "Person [id=" + id + ", name=" + name + ", interest=" + interest
				+ ", birth=" + birth + "]";
	}

}
4. 对象-关系映射 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="com.baidu.hibernate.app">

    <class name="Person" table="PERSON" >
  
        <id name="id" type="java.lang.Integer" >
            <column name="ID" />
            <generator class="hilo" />
        </id>
        
        <property name="name" type="java.lang.String">
            <column name="NAME" />
        </property>
        
        <property name="interest" type="java.lang.String">
            <column name="INTEREST" />
        </property>
        
        <property name="birth" type="java.util.Date">
            <column name="BIRTH" />
        </property>
    </class>
    
</hibernate-mapping>
~~~~~~~~~~~~~~~~~分割线~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~下面是方法测试类~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
5. 方法测试类 TestHibernatePerson.java

①  掌握 persist() 和save() 的区别

② 掌握 get() 和load()  的区别

③ 如何才能避免update 方法不盲目的触发 UPDATE 的SQL语句呢? 如何配置

④ saveOrUpdate() 方法 在什么情况下触发 save() 方法,什么情况下触发update() 方法 。什么样的对象也会被视为游离对象?

⑤ 能否 在删除后即把OID 置为空呢? 如何配置


package com.baidu.hibernate.app;

import static org.junit.Assert.*;

import java.sql.Connection;
import java.sql.SQLException;
import java.util.Date;

import org.hibernate.Session;
import org.hibernate.SessionFactory;
import org.hibernate.Transaction;
import org.hibernate.cfg.Configuration;
import org.hibernate.jdbc.Work;
import org.hibernate.service.ServiceRegistry;
import org.hibernate.service.ServiceRegistryBuilder;
import org.junit.After;
import org.junit.Before;
import org.junit.Test;

public class TestHibernatePerson {
	
	private SessionFactory sessionFactory;
	private Session session;
	private Transaction transaction;
	
	
	@Before
	public void init(){
		Configuration configuration = new Configuration().configure();
		ServiceRegistry serviceRegistry = new ServiceRegistryBuilder()
					.applySettings(configuration.getProperties())
					.buildServiceRegistry();
		
		sessionFactory = configuration.buildSessionFactory(serviceRegistry);
		
		session = sessionFactory.openSession();
		transaction = session.beginTransaction();
		System.out.println("init");
	}
	
	@After
	public void destroy(){
		transaction.commit();
		System.out.println("WO  在 commit 后");
		session.close();
		sessionFactory.close();
		System.out.println("distroy");
	}
	
	
	
	/**
	 * 1. save() 方法
	 * 	①. 使一个临时对象变为持久化对象
	 * 	②. 为对象分配ID
	 *  ③. 在flush 缓存时,会发送一条INSERT 语句
	 *  ④. 在save() 方法之前设置的 ID 是无效的
	 *  ⑤. 持久化对象的ID 是不能被修改的
	 * 
	 * save()方式两条sql 语句:
	 * 			 ①.发送一条select 语句
	 * 			 ②. 在发送一条update 语句
	 * 
	 * 执行完save()方法后,执行commit 是 会在发送一条insert 语句
	 */
	@Test
	public void testSave() {
		Person person = new Person();
		person.setName("ZhaoYun");
		person.setInterest("music2");
		person.setBirth(new Date());
		person.setId(20);
		
		System.out.println(person);
		System.out.println("wo zai save 前");
		session.save(person);
		System.out.println("wo zai save 后");
		
		System.out.println(person);
		System.out.println("WO  在 commit 前");
	}
	
	
	/**
	 * persist(): 也会执行INSERT 操作
	 * 
	 *  persist() 和save() 的区别:
	 * 			当对一个OID 不为null 的对象执行save()方法时,会把该对象以一个新的oid保存到数据库中,
	 * 			但是,当执行persist()方法时,则会抛出一个异常
	 */
	@Test
	public void testPersist(){
		Person person = new Person();
		person.setName("Zhao");
		person.setInterest("music");
		person.setBirth(new Date());
//		person.setId(30);//如果在最新persist 方法前的这里 这种的 id 属性,这会抛出异常
						//org.hibernate.PersistentObjectException: detached entity 
						//passed to persist: com.baidu.hibernate.app.Person	
		session.persist(person);
		
		System.out.println("WO  在 commit 前");
	}
	
	/**
	 * get() 和load() 方法
	 * 1. 执行get 方法: 会立即查询加载对象,
	 * 2. 执行load 方法, 若不使用该对象,则不会立即执行查询操作,而返回一个代理对象
	 * 
	 * 		get方法 是立即检索,load 方法是延迟检索。
	 * 
	 * 3. load 方法可能会抛出 org.hibernate.LazyInitializationException: 
	 * 			could not initialize proxy - no Session 异常
	 * 		原因:因为load 是延迟检索,在需要初始化代理对象之前,已经关闭了Session 导致
	 * 
	 * 4. 若数据库中没有对应的记录,session 也没有被关闭:
	 * 		get 返回 null
	 * 		load 若不使用该对象的任何属性,则不会抛异常
	 * 			 若需要初始化时,则会抛出异常。
	 */
	@Test
	public void testLoad(){
		Person person = (Person) session.load(Person.class, 1);
		
		session.close();
		
		System.out.println(person);
	}
	
	@Test
	public void testGet(){
		Person person = (Person) session.get(Person.class, 1);
		//System.out.println(person);
	}
	
	/**
	 * update:
	 * 
	 * 1. 若更新一个持久化对象,不需要显示的调用update() 方法,因为Transaction
	 * 	  的commit() 方法时,会先执行session 的flush 方法。
	 * 
	 * 2. 更新一个游离对象,需要显式的调用session的update方法,只有显示的调用update() 方法,才会把一个游离对象变成了一个持久化对象
	 * 
	 * 3. 如果调用获取对象后,重新设置了对象的属性,但是属性设置前后没有变化,
	 * 		即:数据库中记录id=1 的name属性是 LiuBei, 修改后的name属性 还是LiuBei ,
	 * 		不论是否显示的调用update() 方法,都发送update 的SQL语句 。
	 * 
	 * 注意: 前提: 当显示的调用了update() 方法时:
	 * 	①. 无论要更新的游离对象和数据表中的是否一致,都会发送UPDATE 的 SQL语句。
	 * 		
	 * 		如何才能避免update 方法不在盲目的触发 UPDATE 的SQL语句呢?
	 * 			为了解决这个问题,需要在.hbm.xml 文件的class 中添加 
	 * 			select-before-update=true(默认为false),但通常不需要设置该属性
	 * 			配置详解见下 A
	 * 
	 *  ②. 对于游离对象,可以修改其ID, 但若在数据表中没有和修改后的ID对应的记录,且又调用了update() 方法,则会抛出异常
	 *  						若在数据表中有和修改后的ID对应的记录,且又调用了update() 方法,则会更新数据表中
	 *  						和修改后的ID对应的记录
	 * 		
	 *  ③. 当update() 方法 关联一个游离对象时,如果在Session 的缓存中已经存在相同的OID 的持久化对象,则会抛出异常
	 *  	因为在session 缓存中不能有两个OID 相同的对象同时存在 : 即缓存中不能同时存在两个Id 相同的对象,
	 *  	对应同一条数据库的记录
	 */
	
	@Test
	public void testUpdate(){
		Person person = (Person) session.get(Person.class, 1);
		
		transaction.commit();;
		session.close();
		
		session = sessionFactory.openSession();
		transaction = session.beginTransaction();
		
		person.setName("Liu");
		session.update(person);
		System.out.println(person);
	
	}
	
	
	/**
	 * Session 的 saveOrUpdate() 方法同时包含了 save() 与 update() 方法的功能
	 * 
	 *  判定对象是否为临时对象的标准
	 *  		① Java 对象的 OID 为 null
	 *  		② 映射文件中为 <id> 设置了 unsaved-value  属性, 
	 *  			并且 Java 对象的 OID 取值与这个 unsaved-value 属性值匹配
	 * 			配置详解见下 B
	 * 注意:
	 * 	① 若 OID 不为null, 但是对应该ID的记录,在数据表中又不存在,则会抛出异常:org.hibernate.StaleStateException
	 *  ② 了解:OID 值等于id 的 unsaved-value 属性值的对象,也被认为是一个游离对象。	
	 *  	详解:Person.hbm.xml 配置中 的 id name="id" type="java.lang.Integer" unsaved-value="200"
	 *  		则id=200 的对象会被认为是游离对象
	 *  	
	 */
	@Test
	public void testSaveOrUpdate(){
		Person person = new Person("Tom", "eating fish", new Date());
		person.setId(200);
		session.saveOrUpdate(person);
	}
	
	
	/**
	 *delete: 执行删除操作,只要OID 和数据表中一条记录对应,就会准备执行delete 操作 
	 * 		  若OID 在数据表中没有对应的记录,则会抛出异常
	 * 
	 * 能否 在删除后即把OID 置为空呢? 可以
	 * 		Hibernate 的 cfg.xml 配置文件中有一个 hibernate.use_identifier_rollback 属性, 其默认值为 false, 
	 * 		若把它设为 true, 将改变 delete() 方法的运行行为:delete() 方法会把持久化对象或游离对象的 OID 
	 * 		设置为 null, 使它们变为临时对象 
	 * 
	 * 		配置详解见下 C
	 */
	
	@Test
	public void testDelete(){
		Person person = new Person();
		//person.setId(851968);//这时person 对象是 当new 的,还未被保存到数据库的数据表中,因此是游离对象
		//session.delete(person);
		
		
		person = (Person) session.get(Person.class, 884736);// 这时person 是从数据库总获取的 是一个持久化对象
		session.delete(person);
	}
	
	
	/**
	 * evict:从session 缓存中把指定的持久化对象移除
	 * 
	 */
	@Test
	public void testEvict(){
		Person person = (Person) session.get(Person.class, 819200);
		Person person2 = (Person) session.get(Person.class, 917504);
		
		person.setName("AA");
		person2.setName("BB");
		
		session.evict(person);
	}
	
	
	/**
	 * Hibernate 如何调用存储过程
	 * 
	 * 可以通过调用 Work 接口 来完成存储过程: 直接通过 JDBC API 来访问数据库的操作
	 * 
	 * Session 的 doWork(Work) 方法用于执行 Work 对象指定的操作, 
	 * 				即调用 Work 对象的 execute() 方法. 
	 * Session 会把当前使用的数据库连接传递给 execute() 方法.
	 * 
	 */
	@Test
	public void testDoWork(){
		
		session.doWork(new Work(){
			
			@Override
			public void execute(Connection connection) throws SQLException {
				System.out.println(connection);
				
			}
		});
	}		
	
}

③ 如何才能避免update 方法不盲目的触发 UPDATE 的SQL语句呢? 如何配置 

详解A:需要在.hbm.xml 文件的class 中添加 select-before-update=true

<?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="com.baidu.hibernate.app">

	<!-- 
		如何才能避免update 方法不在盲目的触发 UPDATE 的SQL语句呢?
			为了解决这个问题,需要在.hbm.xml 文件的class 中添加 
			select-before-update=true(默认为false),但通常不需要设置该属性
	 -->

    <class name="Person" table="PERSON" select-before-update=true>

        <id name="id" type="java.lang.Integer"  unsaved-value="200">
            <column name="ID" />
            <generator class="hilo" />
        </id>
        
        <property name="name" type="java.lang.String">
            <column name="NAME" />
        </property>
        
        <property name="interest" type="java.lang.String">
            <column name="INTEREST" />
        </property>
        
        <property name="birth" type="java.util.Date">
            <column name="BIRTH" />
        </property>
    </class>
    
</hibernate-mapping>

④ 调用saveOrUpdate() 方法 什么样的对象也会被视为游离对象?如何配置

详解B:需要在.hbm.xml 文件的id 中配置 

              因为unsaved-value 的值配置为多少 ,这会认为 id = 为多少  的对象也是游离对象,

              如下:unsaved-value=1 ,这id=1  的对象就会被认为也是游离对象

<?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="com.baidu.hibernate.app">

    <class name="Person" table="PERSON" select-before-update=true>
    	
    	<!-- 
    		OID 值等于id 的 unsaved-value 属性值的对象,也被认为是一个游离对象。
    				
    			例如:<id name="id" type="java.lang.Integer"  unsaved-value="1">
				 因为unsaved-value 的值是1 ,这会认为 id = 1  的对象 也是游离对象    				
    	 -->
    	
        <id name="id" type="java.lang.Integer"  unsaved-value="200">
            <column name="ID" />
            <generator class="hilo" />
        </id>
        
        <property name="name" type="java.lang.String">
            <column name="NAME" />
        </property>
        
        <property name="interest" type="java.lang.String">
            <column name="INTEREST" />
        </property>
        
        <property name="birth" type="java.util.Date">
            <column name="BIRTH" />
        </property>
    </class>
    
</hibernate-mapping>

⑤ 能否 在删除后即把OID 置为空呢? 如何配置

详解C:需要在hibernate.cfg.xml 文件中配置 

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE hibernate-configuration PUBLIC
		"-//Hibernate/Hibernate Configuration DTD 3.0//EN"
		"http://hibernate.sourceforge.net/hibernate-configuration-3.0.dtd">
<hibernate-configuration>

    <session-factory>
    	<property name="hibernate.connection.username">root</property>
    	<property name="hibernate.connection.password">chuck</property>
    	<property name="hibernate.connection.driver_class">com.mysql.jdbc.Driver</property>
    	<property name="hibernate.connection.url">jdbc:mysql://localhost:3306/hibernate</property>
    	
    	<property name="hibernate.dialect">org.hibernate.dialect.MySQLInnoDBDialect</property>
    	<property name="hibernate.show_sql">true</property>
    	<property name="hibernate.format_sql">true</property>
    	<property name="hibernate.hbm2ddl.auto">update</property>
    	<property name="hibernate.connection.isolation">2</property>
    	
    	<!-- 能否 在删除后即把OID 置为空呢? 可以
    		Hibernate 的 cfg.xml 配置文件中有一个 hibernate.use_identifier_rollback 属性, 其默认值为 false, 
    		若把它设为 true, 将改变 delete() 方法的运行行为:delete() 方法会把持久化对象或游离对象的 OID 
    		设置为 null, 使它们变为临时对象 
    	 -->
    	<property name="hibernate.use_identifier_rollback">true</property>
    	
    	<mapping resource="com/baidu/hibernate/app/Person.hbm.xml"  />
    	
    	
    </session-factory>
    
</hibernate-configuration>




你可能感兴趣的:(持久化对象的 四种状态)