Hibernate一級緩存和二級緩存詳解
一、一級緩存二級緩存的概念解釋
(1)一級緩存就是Session級別的緩存,一個Session做了一個查詢操作,它會把這個操作的結(jié)果放在一級緩存中,如果短時間內(nèi)這個session(一定要同一個session)又做了同一個操作,那么hibernate直接從一級緩存中拿,而不會再去連數(shù)據(jù)庫,取數(shù)據(jù);
(2)二級緩存就是SessionFactory級別的緩存,顧名思義,就是查詢的時候會把查詢結(jié)果緩存到二級緩存中,如果同一個sessionFactory創(chuàng)建的某個session執(zhí)行了相同的操作,hibernate就會從二級緩存中拿結(jié)果,而不會再去連接數(shù)據(jù)庫;
(3)Hibernate中提供了兩級Cache,第一級別的緩存是Session級別的緩存,它是屬于事務(wù)范圍的緩存。這一級別的緩存由hibernate管理的,一般情況下無需進行干預(yù);第二級別的緩存是SessionFactory級別的緩存,它是屬于進程范圍或群集范圍的緩存。這一級別的緩存可以進行配置和更改,并且可以動態(tài)加載和卸載。 Hibernate還為查詢結(jié)果提供了一個查詢緩存,它依賴于第二級緩存;
二、一級緩存和二級緩存的比較
(1)第一級緩存 第二級緩存 存放數(shù)據(jù)的形式相互關(guān)聯(lián)的持久化對象 對象的散裝數(shù)據(jù) 緩存的范圍事務(wù)范圍,每個事務(wù)都有單獨的第一級緩存進程范圍或集群范圍,緩存被同一個進程或集群范圍內(nèi)的所有事務(wù)共享并發(fā)訪問策略由于每個事務(wù)都擁有單獨的第一級緩存,不會出現(xiàn)并發(fā)問題,無需提供并發(fā)訪問策略由于多個事務(wù)會同時訪問第二級緩存中相同數(shù)據(jù),因此必須提供適當(dāng)?shù)牟l(fā)訪問策略,來保證特定的事務(wù)隔離級別數(shù)據(jù)過期策略沒有提供數(shù)據(jù)過期策略。
(2)處于一級緩存中的對象永遠不會過期,除非應(yīng)用程序顯式清空緩存或者清除特定的對象必須提供數(shù)據(jù)過期策略,如基于內(nèi)存的緩存中的對象的最大數(shù)目,允許對象處于緩存中的最長時間,以及允許對象處于緩存中的最長空閑時間物理存儲介質(zhì)內(nèi)存內(nèi)存和硬盤。
(3)對象的散裝數(shù)據(jù)首先存放在基于內(nèi)存的緩存中,當(dāng)內(nèi)存中對象的數(shù)目達到數(shù)據(jù)過期策略中指定上限時,就會把其余的對象寫入基于硬盤的緩存中。
(4)緩存的軟件實現(xiàn)在Hibernate的Session的實現(xiàn)中包含了緩存的實現(xiàn)由第三方提供,Hibernate僅提供了緩存適配器(CacheProvider)。用于把特定的緩存插件集成到Hibernate中。
(5)啟用緩存的方式只要應(yīng)用程序通過Session接口來執(zhí)行保存、更新、刪除、加載和查詢數(shù)據(jù)庫數(shù)據(jù)的操作,Hibernate就會啟用第一級緩存,把數(shù)據(jù)庫中的數(shù)據(jù)以對象的形式拷貝到緩存中,對于批量更新和批量刪除操作,如果不希望啟用第一級緩存,可以繞過Hibernate API,直接通過JDBC API來執(zhí)行指操作。
(6)用戶可以在單個類或類的單個集合的粒度上配置第二級緩存。如果類的實例被經(jīng)常讀但很少被修改,就可以考慮使用第二級緩存。
(7)只有為某個類或集合配置了第二級緩存,Hibernate在運行時才會把它的實例加入到第二級緩存中。用戶管理緩存的方式第一級緩存的物理介質(zhì)為內(nèi)存,由于內(nèi)存容量有限,必須通過恰當(dāng)?shù)臋z索策略和檢索方式來限制加載對象的數(shù)目。Session的 evit()方法可以顯式清空緩存中特定對象,但這種方法不值得推薦。第二級緩存的物理介質(zhì)可以是內(nèi)存和硬盤,因此第二級緩存可以存放大量的數(shù)據(jù),數(shù)據(jù)過期策略的maxElementsInMemory屬性值可以控制內(nèi)存中的對象數(shù)目。
(8)管理第二級緩存主要包括兩個方面:選擇需要使用第二級緩存的持久類,設(shè)置合適的并發(fā)訪問策略:選擇緩存適配器,設(shè)置合適的數(shù)據(jù)過期策略。
三、 一級緩存的管理
(1)當(dāng)應(yīng)用程序調(diào)用Session的save()、update()、savaeOrUpdate()、get()或load(),以及調(diào)用查詢接口的 list()、iterate()或filter()方法時,如果在Session緩存中還不存在相應(yīng)的對象,Hibernate就會把該對象加入到第一級緩存中。當(dāng)清理緩存時,Hibernate會根據(jù)緩存中對象的狀態(tài)變化來同步更新數(shù)據(jù)庫。 Session為應(yīng)用程序提供了兩個管理緩存的方法: evict(Object obj):從緩存中清除參數(shù)指定的持久化對象。 clear():清空緩存中所有持久化對象。
(2)save、update、saveOrupdate、load、list、iterate、lock會向一級緩存存放數(shù)據(jù);
save 案例:
//添加一個學(xué)生 Student student=new Student(); student.setName("小東"); s.save(student);//放入一級緩存 //我馬上查詢 Student stu2=(Student) s.get(Student.class, student.getId()); //select System.out.println("你剛剛加入的學(xué)生名字是"+stu2.getName());
(3)什么操作會從一級緩存取數(shù)據(jù):get、load、list
get / load 會首先從一級緩存中取,如沒有.再有不同的操作[get 會立即向數(shù)據(jù)庫發(fā)請求,而load 會返回一個代理對象,直到用戶真的去使用數(shù)據(jù),才會向數(shù)據(jù)庫發(fā)請求;
//查詢45號學(xué)生 Student stu=(Student) s.get(Student.class, 45); System.out.println("|||||||||||||||||||"); String hql="from Student where id=45"; Student stu2=(Student) s.createQuery(hql).uniqueResult(); System.out.println(stu2.getName());
從上面的案例,我們看出 query.list() query.uniueResut() 不會從一級緩取數(shù)據(jù)! 但是query.list 或者query.uniqueRestu() 會向一級緩存放數(shù)據(jù)的.
注意:
① 一級緩存不需要配置,就可以使用,它本身沒有保護機制,所以我們程序員要考慮這個問題,我們可以同 evict 或者 clear來清除session緩存中對象. evict 是清除一個對象,clear是清除所有的sesion緩存對象
② session級緩存中對象的生命周期, 當(dāng)session關(guān)閉后,就自動銷毀.
③ 我們自己用HashMap來模擬一個Session緩存,加深對緩存的深入.
四、Hibernate二級緩存的管理
1. Hibernate二級緩存策略的一般過程如下:
1) 條件查詢的時候,總是發(fā)出一條select * from table_name where …. (選擇所有字段)這樣的SQL語句查詢數(shù)據(jù)庫,一次獲得所有的數(shù)據(jù)對象。
2) 把獲得的所有數(shù)據(jù)對象根據(jù)ID放入到第二級緩存中。
3) 當(dāng)Hibernate根據(jù)ID訪問數(shù)據(jù)對象的時候,首先從Session一級緩存中查;查不到,如果配置了二級緩存,那么從二級緩存中查;查不到,再查詢數(shù)據(jù)庫,把結(jié)果按照ID放入到緩存。
4) 刪除、更新、增加數(shù)據(jù)的時候,同時更新緩存。Hibernate二級緩存策略,是針對于ID查詢的緩存策略,對于條件查詢則毫無作用。為此,Hibernate提供了針對條件查詢的Query Cache。
5) 二級緩存的對象可能放在內(nèi)存,也可能放在磁盤.
2. 什么樣的數(shù)據(jù)適合存放到第二級緩存中?
1) 很少被修改的數(shù)據(jù)
2) 不是很重要的數(shù)據(jù),允許出現(xiàn)偶爾并發(fā)的數(shù)據(jù)
3) 不會被并發(fā)訪問的數(shù)據(jù)
4) 參考數(shù)據(jù),指的是供應(yīng)用參考的常量數(shù)據(jù),它的實例數(shù)目有限,它的實例會被許多其他類的實例引用,實例極少或者從來不會被修改。
3. 不適合存放到第二級緩存的數(shù)據(jù)?
1) 經(jīng)常被修改的數(shù)據(jù)
2) 財務(wù)數(shù)據(jù),絕對不允許出現(xiàn)并發(fā)
3) 與其他應(yīng)用共享的數(shù)據(jù)。
4. 常用的緩存插件 Hibernater二級緩存是一個插件,下面是幾種常用的緩存插件:
◆EhCache:可作為進程范圍的緩存,存放數(shù)據(jù)的物理介質(zhì)可以是內(nèi)存或硬盤,對Hibernate的查詢緩存提供了支持。
◆OSCache:可作為進程范圍的緩存,存放數(shù)據(jù)的物理介質(zhì)可以是內(nèi)存或硬盤,提供了豐富的緩存數(shù)據(jù)過期策略,對Hibernate的查詢緩存提供了支持。
◆SwarmCache:可作為群集范圍內(nèi)的緩存,但不支持Hibernate的查詢緩存。
◆JBossCache:可作為群集范圍內(nèi)的緩存,支持事務(wù)型并發(fā)訪問策略,對Hibernate的查詢緩存提供了支持。
5. 配置Hibernate二級緩存的主要步驟:
1) 選擇需要使用二級緩存的持久化類,設(shè)置它的命名緩存的并發(fā)訪問策略。這是最值得認(rèn)真考慮的步驟。
2) 選擇合適的緩存插件,然后編輯該插件的配置文件。
<property name="hbm2ddl.auto">update</property> <!-- 啟動二級緩存 --> <property name="cache.use_second_level_cache">true</property> <!-- 指定使用哪種二級緩存 --> <property name="cache.provider_class">org.hibernate.cache.OSCacheProvider</property> <mapping resource="com/hsp/domain/Department.hbm.xml" /> <mapping resource="com/hsp/domain/Student.hbm.xml" /> <!-- 指定哪個domain啟用二級緩存 特別說明二級緩存策略: 1. read-only 2. read-write 3. nonstrict-read-write 4. transcational --> <class-cache class="com.hsp.domain.Student" usage="read-write"/>
3)可以把oscache.properties文件放在 src目錄下,這樣你可以指定放入二級緩存的對象capacity 大小. 默認(rèn)1000
6.使用二級緩存:
// TODO Auto-generated method stub //通過獲取一個sesion,讓hibernate框架運行(config->加載hibernate.cfg.xml) Session s=null; Transaction tx=null; try { //我們使用基礎(chǔ)模板來講解. s=HibernateUtil.openSession(); tx=s.beginTransaction(); //查詢45號學(xué)生 Student stu1=(Student) s.get(Student.class, 45);//45->一級緩存 System.out.println(stu1.getName()); tx.commit(); } catch (Exception e) { e.printStackTrace(); if(tx!=null){ tx.rollback(); } }finally{ if(s!=null && s.isOpen()){ s.close(); } } System.out.println("*********************************"); try { //我們使用基礎(chǔ)模板來講解. s=HibernateUtil.openSession(); tx=s.beginTransaction(); //查詢45號學(xué)生 Student stu1=(Student) s.get(Student.class, 45); System.out.println(stu1.getName()); Student stu3=(Student) s.get(Student.class, 46); System.out.println(stu3.getName()); tx.commit(); } catch (Exception e) { e.printStackTrace(); if(tx!=null){ tx.rollback(); } }finally{ if(s!=null && s.isOpen()){ s.close(); } } //完成一個統(tǒng)計,統(tǒng)計的信息在Sessfactory //SessionFactory對象. Statistics statistics= HibernateUtil.getSessionFactory().getStatistics(); System.out.println(statistics); System.out.println("放入"+statistics.getSecondLevelCachePutCount()); System.out.println("命中"+statistics.getSecondLevelCacheHitCount()); System.out.println("錯過"+statistics.getSecondLevelCacheMissCount());
在配置了二級緩存后,請大家要注意可以通過 Statistics,查看你的配置命中率高不高
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,謝謝大家對腳本之家的支持。如果你想了解更多相關(guān)內(nèi)容請查看下面相關(guān)鏈接
- Hibernate核心類和接口的詳細介紹
- Hibernate的各種保存方式的區(qū)別詳解
- Hibernate中Session.get()方法和load()方法的詳細比較
- Hibernate用ThreadLocal模式(線程局部變量模式)管理Session
- hibernate測試時遇到的幾個異常及解決方法匯總
- Hibernate中使用HQLQuery查詢?nèi)繑?shù)據(jù)和部分?jǐn)?shù)據(jù)的方法實例
- Hibernate實現(xiàn)many-to-many的映射關(guān)系
- SpringBoot 使用hibernate validator校驗
- Hibernate之CRUD操作實踐
- Hibernate連接三種數(shù)據(jù)庫的配置文件
相關(guān)文章
深入理解SpringMVC中央調(diào)度器DispatcherServlet
這篇文章主要介紹了SpringMVC核心之中央調(diào)度器DispatcherServlet的相關(guān)知識,包括SpringMVC請求處理過程及SrpingMVC容器和spring?IOC容器關(guān)系,需要的朋友可以參考下2022-05-05Spring中的之啟動過程obtainFreshBeanFactory詳解
這篇文章主要介紹了Spring中的之啟動過程obtainFreshBeanFactory詳解,在refresh時,prepareRefresh后,馬上就調(diào)用了obtainFreshBeanFactory創(chuàng)建beanFactory以及掃描bean信息(beanDefinition),并通過BeanDefinitionRegistry注冊到容器中,需要的朋友可以參考下2024-02-02JAVA對象JSON數(shù)據(jù)互相轉(zhuǎn)換的四種常見情況
這篇文章主要介紹了JAVA對象JSON數(shù)據(jù)互相轉(zhuǎn)換的四種常見情況,需要的朋友可以參考下2014-04-04Java從單體架構(gòu)升級到微服務(wù)要注意的一些問題
這篇文章主要介紹了Java從單體架構(gòu)升級到微服務(wù)要注意的一些問題,對架構(gòu)感興趣的同學(xué),可以參考下2021-04-04System.getProperty(“l(fā)ine.separator“)含義及意義詳解
這篇文章主要介紹了System.getProperty(“l(fā)ine.separator“)含義,本文給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2023-05-05