java框架篇---hibernate之session狀態


Session接口是Hibernate向程序提供操縱數據庫的最主要接口,是單線程對象,它提供了基本的保存、更新、刪除和查詢方法。它有一個緩存,保存了持久化對象,當清理緩存時,按照這些持久化對象同步更新數據庫。

注意:session的某些方法(persist,load)不會立即把改動寫入數據庫,而是緩存到session的一級緩存中,除非顯示調用flush,或者關閉session時才會更新到數據庫

  1. 臨時狀態(Transient):沒與session關聯
  2. 持久化狀態(Persistent):與session關聯,沒close
  3. 游離狀態(Detached):當session.close后

Session的方法詳解

1.保存 

          save:立即插入數據庫,並且返回主鍵

          persist:不立即(延遲)插入數據庫,無返回值

2.獲取

             load:加載對象后,對對象的改動不會立即刷新到db,必須flush到db

                       ex: User user=session.load(User.class,2);

                                user.setName('gt');

                                user.flush();   (延遲加載)

             get:加載對象后,對對象的改動立即刷新到db

3.更新

           update:持久化對象,更新

     saveOrUpdate:包含save()和update()功能,如果傳入的參數是臨時對象(沒有保存過)就調用save()方法;如果傳入的參數是游離對象,就調用update()方法

           merge:不會持久化對象,只會把托管對象的修改更新到db

4.刪除

          delete:從數據庫中刪除與JAVA對象對應的記錄

 

5.清理

           flush:把緩存同步到db

           clear:清除session的緩存大小(更新批量時,應考慮)

 

三種狀態的轉換關系

下面通過實例講解:

實體:

package cn.itcast.h_session_method;

public class User {
    private Integer id;
    private String name;
    private byte[] data = new byte[1024 * 1024 * 5];

    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;
    }

}

映射文件:

<?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="cn.itcast.h_session_method">
    
    <!-- 
        lazy屬性:默認為true,默認可以懶加載。
     -->
    <class name="User" table="user" lazy="true">
        <id name="id">
            <generator class="native"></generator>
        </id>
        <property name="name"/>
    </class>
    
</hibernate-mapping>

配置文件:hibernate.cfg.xml

<?xml version='1.0' encoding='utf-8'?>
<!DOCTYPE hibernate-configuration PUBLIC
        "-//Hibernate/Hibernate Configuration DTD 3.0//EN"
        "http://www.hibernate.org/dtd/hibernate-configuration-3.0.dtd">

<hibernate-configuration>

<session-factory>

    <!--數據庫連接設置 -->
    <property name="connection.driver_class">
        com.mysql.jdbc.Driver
    </property>
    <property name="connection.url">
        jdbc:mysql://localhost:3306/mytest
    </property>
    <property name="connection.username">root</property>
    <property name="connection.password">root</property>


    <!-- 方言 -->
    <property name="dialect">
        org.hibernate.dialect.MySQL5Dialect
    </property>

    <!-- 控制台顯示SQL -->
    <property name="show_sql">true</property>

    <!-- 自動更新表結構 -->
    <property name="hbm2ddl.auto">update</property>
    <mapping resource="cn/itcast/h_session_method/User.hbm.xml" />

</session-factory>

</hibernate-configuration>

測試文件

package cn.itcast.h_session_method;

import org.hibernate.Session;
import org.hibernate.SessionFactory;
import org.hibernate.cfg.Configuration;
import org.junit.Test;

import com.java1234.util.HibernateSessionFactory;

public class App {

    private static SessionFactory sessionFactory = HibernateSessionFactory.getSessionFactory();

    // save():把臨時狀態變為持久化狀態(交給Sessioin管理)
    // 會生成:insert into ...
    @Test
    public void testSave() throws Exception {
        Session session = sessionFactory.openSession();
        session.beginTransaction();
        // --------------------------------------------

        User user = new User(); // 臨時狀態
        user.setName("test");
        session.save(user); // 變為了持久化狀態

        // --------------------------------------------
        session.getTransaction().commit();
        session.close();

        user.setName("李四"); // 游離狀態
        System.out.println(user.getName()); // 游離狀態
    }

    // update():把游離狀態變為持久化狀態
    // 會生成:update ...
    // 在更新時,對象不存在就報錯
    @Test
    public void testUpdate() throws Exception {
        Session session = sessionFactory.openSession();
        session.beginTransaction();
        // --------------------------------------------

        User user = (User) session.get(User.class, 1);
        System.out.println(user.getName()); // 持久化狀態

        // session.clear(); // 清除Session中所有的對象
        session.evict(user); // 清除Session中一個指定的對象

        user.setName("newname3");
        session.update(user);
        System.out.println("----");
        // session.flush(); // 刷出到數據庫

        // --------------------------------------------
        session.getTransaction().commit(); // 
        session.close();
    }

    // saveOrUpdate():把臨時或游離狀態轉為持久化狀態
    // 會生成:insert into 或 update ...
    // 在更新時,對象不存在就報錯
    // 本方法是根據id判斷對象是什么狀態的:如果id為原始值(對象的是null,原始類型數字是0)就是臨時狀態,如果不是原始值就是游離狀態。
    @Test
    public void testSaveOrUpdate() throws Exception {
        Session session = sessionFactory.openSession();
        session.beginTransaction();
        // --------------------------------------------

        User user = new User();
        user.setId(3); // 自己生成一個游離狀態對象
        user.setName("newName");

        session.saveOrUpdate(user);

        // --------------------------------------------
        session.getTransaction().commit();
        session.close();
    }

    // delete():把持久化或游離轉為刪除狀態
    // 會生成:delete ...
    // 如果刪除的對象不存在,就會拋異常
    @Test
    public void testDelete() throws Exception {
        Session session = sessionFactory.openSession();
        session.beginTransaction();
        // --------------------------------------------

        // User user = (User) session.get(User.class, 2); // 持久化

        User user = new User();
        user.setId(300);

        session.delete(user);
        session.flush();

        System.out.println("---");

        // --------------------------------------------
        session.getTransaction().commit();
        session.close();
    }

    // get():獲取數據,是持久化狀態
    // 會生成:select ... where id=?
    // 會馬上執行sql語句
    // 如果數據不存在,就返回null
    @Test
    public void testGet() throws Exception {
        Session session = sessionFactory.openSession();
        session.beginTransaction();
        // --------------------------------------------

        User user = (User) session.get(User.class, 5); // 持久化
        System.out.println(user.getClass());
        // System.out.println("---");
        // System.out.println(user.getName());

        // --------------------------------------------
        session.getTransaction().commit();
        session.close();
    }

    // load():獲取數據,是持久化狀態
    // 會生成:select ... where id=?
    // load()后返回的是一個代理對象,要求類不能是final的,否則不能生成子類代理,就不能使用懶加載功能了。
    // 讓懶加載失效的方式:一、把實體寫成final的;二、在hbm.xml中寫<class ... lazy="false">
    // 不會馬上執行sql語句,而是在第1次使用非id或class屬性時執行sql。
    // 如果數據不存在,就拋異常:ObjectNotFoundException
    @Test
    public void testLoad() throws Exception {
        Session session = sessionFactory.openSession();
        session.beginTransaction();
        // --------------------------------------------

        User user = (User) session.load(User.class, 5);
        System.out.println(user.getClass());
        System.out.println("---");
        System.out.println(user.getId());
        System.out.println(user.getName());
        // System.out.println(user.getName());

        // --------------------------------------------
        session.getTransaction().commit();
        session.close();
    }

    // 操作大量數據,要防止Session中對象過多而內存溢出
    @Test
    public void testBatchSave() throws Exception {
        Session session = sessionFactory.openSession();
        session.beginTransaction();
        // --------------------------------------------

        for (int i = 0; i < 30; i++) {
            User user = new User();
            user.setName("測試");
            session.save(user);

            if (i % 10 == 0) {
                session.flush(); // 先刷出
                session.clear(); // 再清空
            }
        }

        // --------------------------------------------
        session.getTransaction().commit();
        session.close();
    }

    @Test
    public void test2() throws Exception {
        Session session = sessionFactory.openSession();
        session.beginTransaction();
        // --------------------------------------------

        User user = (User) session.get(User.class, 5); // 持久化
        System.out.println(user.getName());

        // session.clear();
        // user = (User) session.get(User.class, 5); // 持久化

        session.refresh(user); // 刷新Session緩存中對象的狀態,即重新select一下
        System.out.println(user.getName());

        // --------------------------------------------
        session.getTransaction().commit();
        session.close();
    }
}

對於剛創建的一個對象,如果session中和數據庫中都不存在該對象,那么該對象就是臨時對象(Transient)

臨時對象調用save方法,或者游離對象調用update方法可以使該對象變成持久化對象,如果對象是持久化對象時,那么對該對象的任何修改,都會在提交事務時才會與之進行比較,如果不同,則發送一條update語句,否則就不會發送語句

游離對象就是,數據庫存在該對象,但是該對象又沒有被session所托管


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM