1. 程式人生 > >hibernate三級快取

hibernate三級快取

Hibernate中的快取一共有三種,一級快取、二級快取、查詢快取。快取除了使用Hibernate自帶的快取,還可以使用redis進行快取,或是MongoDB進行快取

所使用的Demo:

User.java檔案

 User.java

hibernate.cfg.xml

 hibernate.cfg.xml

其中:

        <property name="show_sql">true</property>
        <property name="format_sql">true</property>

表示開啟列印底層執行的SQL日誌。

下面這是圖片反應了hibernate快取的大致流程:

1,一級快取

每個 Session 物件創建出來,就會分配一塊快取空間,可以儲存 session 物件訪問的物件資訊。 session 關閉後會自動清除快取,手動清除可以用session.clear() , session.evict(obj) 。 Session 一級快取是獨享。
load/get/save/update/saveorupdate 方法處理的物件都會放入快取中

Configuration conf = new Configuration();
conf.configure("hibernate.cfg.xml");//讀取連線引數和對映描述資訊
SessionFactory factory = conf.buildSessionFactory();
Session session = factory.openSession();
            
User user1 = (User)session.load(User.class,1);
System.out.println(user1.getName());//honny,如果不呼叫用getName()方法,那麼資料不會顯示,因為load()預設使用的是一種延遲載入的機制,只有使用到資料的時候才會到資料庫中查詢

//先從session快取中查詢,如果沒找到再去資料庫獲取
User user2 = (User)session.load(User.class,1);
System.out.println(user2.getName());//honny
            
System.out.println(user1==user2);//true,因為user1和user2使用的是同一個session

然後再來看一看控制檯:

 

從控制檯中,我們也可以看出上只執行了一次SQL查詢。

 

一級查詢的優缺點:

優點:可以減少查詢資料庫的次數,加快查詢速度。
缺點:在批量操作中容易導致記憶體溢位問題。

2,二級快取

二級快取是SessionFactory 物件快取,可以被創建出的多個 Session 物件共享。

下面是一張圖片體現一級快取和二級快取的關係:

 

從這個我們就看出了二級快取包含了一級快取。

二級快取預設是關閉的,如果要使用需要手動開啟,下面是開啟過程:

1.匯入ehcache 工具包和 ehcache.xml 配置檔案(配置檔案放到src路徑下)

echache工具包包括:ehcache-core-2.4.3.jar,hibernate-ehcache-4.2.21.Final.jar,slf4j-api-1.6.1.jar

ehcache.xml 檔案

 ehcache.xml

2.在 hibernate.cfg.xml 中配置引數開啟二級快取,啟用 ehcache

<property name="hibernate.cache.use_sencond_level_cache">true</property>
<property name="hibernate.cache.region.factory_class">
org.hibernate.cache.ehcache.EhCacheRegionFactory
</property>

3.在要快取的物件型別中,指定 @Cache 註解標記

@Entity
@Table(name="user")//表示對應的表名
@Cache(usage=CacheConcurrencyStrategy.READ_WRITE)
public class User {
    //........
}

到這裡hibernate的二級快取配置就配好了,下面來測試一下:

Configuration conf = new Configuration();
conf.configure("hibernate.cfg.xml");//讀取連線引數和對映描述資訊
SessionFactory factory = conf.buildSessionFactory();
Session session1 = factory.openSession();
User user1=(User)session1.load(User.class, 1);
System.out.println(user1.getName());//honny

Session session2 = factory.openSession();
//先從快取中查詢,如果沒有查到再去資料庫中取
User user2=(User)session2.load(User.class, 1);
System.out.println(user2.getName());//honny

下面是控制檯列印的列印:

我們可以看出,用同一個SessionFactory的兩個不同session物件查詢相同的資料,只從資料庫中取了一次。

3,查詢快取

一級和二級快取,只能快取單個物件,如果需要快取一個結果集,必須使用查詢快取。

查詢快取預設也是關閉的,如需使用需要手動開啟,下面是開啟過程:

1.針對的物件必需已經開啟了二級快取

2.在 hibernate.cfg.xml 中新增開啟查詢快取的配置

<property name="hibernate.cache.use_query_cache">true</property>

3.在查詢執行前,呼叫 query.setCacheable(true); 

下面看一看測試:

        String hql="from User";
        Configuration conf=new Configuration();
        conf.configure("hibernate.cfg.xml");
        SessionFactory factory=conf.buildSessionFactory();
        Session session1 = factory.openSession();
        Query query1 = session1.createQuery(hql);
        query1.setCacheable(true);//設定開啟快取
        List list1 = query1.list();
        for(Object user:list1){
            System.out.println(((User)user).getName());
        }
        System.out.println("------------------------");
        Session session2 = factory.openSession();
        Query query2 = session2.createQuery(hql);
        query2.setCacheable(true);
        List list2 =query2.list();
        for(Object user:list2){
            System.out.println(((User)user).getName());
        }

然後來看一看控制檯:

 

從控制檯中,我們可以看出,底層查詢的資料庫的過程也只執行了一次。

 

上面就是hibernate的三種快取。最後總結一下,並不是所有的方法都會產生快取效果,只有“load/get/save/update/saveorupdate”才會產生快取效果。三種快取中一級快取是預設開啟的,二級快取和三級快取預設是關閉的。