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”才會產生快取效果。三種快取中一級快取是預設開啟的,二級快取和三級快取預設是關閉的。