20、hibernate查詢緩存(hibernate筆記)

一、簡(jiǎn)介

  • 查詢緩存(工程hibernate_query_cache)是針對(duì)普通屬性結(jié)果集的緩存,對(duì)實(shí)體對(duì)象的結(jié)果集只緩存id。
  • 查詢緩存的生命周期:當(dāng)前關(guān)聯(lián)的表發(fā)生修改,查詢緩存立即失效。

二、查詢緩存的配置和使用

  • 在hibernate.cfg.xml文件中啟用查詢緩存,如:
    <property name="hibernate.cache.use_query_cache">true</property>,因?yàn)閔ibernate中查詢緩存默認(rèn)是關(guān)閉的,這和二級(jí)緩存不一樣。
  • 在程序中必須手動(dòng)啟用查詢緩存,如:
    query.setCacheable(true);

測(cè)試:
使用的相關(guān)實(shí)體類和映射和二級(jí)緩存中使用的是一樣的。只需要在hibernate的配置文件中開啟查詢緩存即可。

/**
     * 開啟查詢緩存,關(guān)閉二級(jí)緩存
     * 
     * 開啟一個(gè)session,分別調(diào)用query.list
     */
    public void testCache1() {
        Session session = null;
        try {
            session = HibernateUtils.getSession();
            session.beginTransaction();
            
            Query query = session.createQuery("select s.name from Student s");
            //啟用查詢查詢緩存
            query.setCacheable(true);
            
            List names = query.list(); 
            for (Iterator iter=names.iterator();iter.hasNext(); ) {
                String name = (String)iter.next();
                System.out.println(name);
            }
            
            System.out.println("-------------------------------------");
            query = session.createQuery("select s.name from Student s");
            //啟用查詢查詢緩存
            query.setCacheable(true);
            
            //沒有發(fā)出查詢sql,因?yàn)閱⒂昧瞬樵兙彺?            names = query.list(); 
            for (Iterator iter=names.iterator();iter.hasNext(); ) {
                String name = (String)iter.next();
                System.out.println(name);
            }

            session.getTransaction().commit();
        }catch(Exception e) {
            e.printStackTrace();
            session.getTransaction().rollback();
        }finally {
            HibernateUtils.closeSession(session);
        }
    }

說(shuō)明:這個(gè)例子中我們可以看到如果沒有開啟二級(jí)緩存和查詢緩存,那么會(huì)發(fā)出兩次sql,但是如果我們將查詢緩存打開,同時(shí)在程序中開啟,那么在第二次使用list查詢普通屬性的時(shí)候是不會(huì)再次發(fā)出sql的。

/**
     * 開啟查詢緩存,關(guān)閉二級(jí)緩存
     * 
     * 開啟兩個(gè)session,分別調(diào)用query.list
     */
    public void testCache2() {
        Session session = null;
        try {
            session = HibernateUtils.getSession();
            session.beginTransaction();
            
            Query query = session.createQuery("select s.name from Student s");
            //啟用查詢查詢緩存
            query.setCacheable(true);
            
            List names = query.list(); 
            for (Iterator iter=names.iterator();iter.hasNext(); ) {
                String name = (String)iter.next();
                System.out.println(name);
            }
            session.getTransaction().commit();
        }catch(Exception e) {
            e.printStackTrace();
            session.getTransaction().rollback();
        }finally {
            HibernateUtils.closeSession(session);
        }
        
        System.out.println("-------------------------------------");
        
        try {
            session = HibernateUtils.getSession();
            session.beginTransaction();
            
            Query query = session.createQuery("select s.name from Student s");
            //啟用查詢查詢緩存
            query.setCacheable(true);
            
            //不會(huì)發(fā)出查詢sql,因?yàn)椴樵兙彺娴纳芷诤蛃ession無(wú)關(guān)
            List names = query.list(); 
            for (Iterator iter=names.iterator();iter.hasNext(); ) {
                String name = (String)iter.next();
                System.out.println(name);
            }
            session.getTransaction().commit();
        }catch(Exception e) {
            e.printStackTrace();
            session.getTransaction().rollback();
        }finally {
            HibernateUtils.closeSession(session);
        }
    }   

說(shuō)明:這個(gè)例子主要是為了說(shuō)明查詢緩存的聲明周期和session是無(wú)關(guān)的,而這里又是查詢的普通屬性,所以在第二次使用list方法的時(shí)候不會(huì)發(fā)出sql語(yǔ)句。

/**
     * 開啟查詢緩存,關(guān)閉二級(jí)緩存
     * 
     * 開啟兩個(gè)session,分別調(diào)用query.iterate
     */
    public void testCache3() {
        Session session = null;
        try {
            session = HibernateUtils.getSession();
            session.beginTransaction();
            
            Query query = session.createQuery("select s.name from Student s");
            //啟用查詢查詢緩存
            query.setCacheable(true);
             
            for (Iterator iter=query.iterate();iter.hasNext(); ) {
                String name = (String)iter.next();
                System.out.println(name);
            }
            session.getTransaction().commit();
        }catch(Exception e) {
            e.printStackTrace();
            session.getTransaction().rollback();
        }finally {
            HibernateUtils.closeSession(session);
        }
        
        System.out.println("-------------------------------------");
        
        try {
            session = HibernateUtils.getSession();
            session.beginTransaction();
            
            Query query = session.createQuery("select s.name from Student s");
            //啟用查詢查詢緩存
            query.setCacheable(true);
             
            //查詢緩存只對(duì)query.list()起作用,query.iterate不起作用,也就是query.iterate不使用
            //查詢緩存
            for (Iterator iter=query.iterate();iter.hasNext(); ) {
                String name = (String)iter.next();
                System.out.println(name);
            }
            session.getTransaction().commit();
        }catch(Exception e) {
            e.printStackTrace();
            session.getTransaction().rollback();
        }finally {
            HibernateUtils.closeSession(session);
        }
    }

說(shuō)明:這個(gè)例子主要為了說(shuō)明查詢緩存只對(duì)list方法起作用,而對(duì)iterate方法是不起作用的。

/**
     * 關(guān)閉查詢緩存,關(guān)閉二級(jí)緩存
     * 
     * 開啟兩個(gè)session,分別調(diào)用query.list查詢實(shí)體對(duì)象
     */
    public void testCache4() {
        Session session = null;
        try {
            session = HibernateUtils.getSession();
            session.beginTransaction();
            
            Query query = session.createQuery("select s from Student s");
            //啟用查詢查詢緩存
            //query.setCacheable(true);
            
            List students = query.list(); 
            for (Iterator iter=students.iterator();iter.hasNext(); ) {
                Student student = (Student)iter.next();
                System.out.println(student.getName());
            }
            session.getTransaction().commit();
        }catch(Exception e) {
            e.printStackTrace();
            session.getTransaction().rollback();
        }finally {
            HibernateUtils.closeSession(session);
        }
        
        System.out.println("-------------------------------------");
        
        try {
            session = HibernateUtils.getSession();
            session.beginTransaction();
            
            Query query = session.createQuery("select s from Student s");
            //啟用查詢查詢緩存
            //query.setCacheable(true);
            
            //會(huì)發(fā)出查詢sql,因?yàn)閘ist默認(rèn)每次都會(huì)發(fā)出查詢sql
            List students = query.list(); 
            for (Iterator iter=students.iterator();iter.hasNext(); ) {
                Student student = (Student)iter.next();
                System.out.println(student.getName());
            }
            session.getTransaction().commit();
        }catch(Exception e) {
            e.printStackTrace();
            session.getTransaction().rollback();
        }finally {
            HibernateUtils.closeSession(session);
        }
    }

說(shuō)明:之前我們都是查詢的普通屬性,這里我們是查詢實(shí)體類。在使用list查詢實(shí)體對(duì)象的時(shí)候每次都會(huì)發(fā)出sql,即使在同一個(gè)session中也是一樣,即默認(rèn)是每次都發(fā)出sql語(yǔ)句。

/**
     * 開啟查詢緩存,關(guān)閉二級(jí)緩存
     * 
     * 開啟兩個(gè)session,分別調(diào)用query.list查詢實(shí)體對(duì)象
     */
    public void testCache5() {
        Session session = null;
        try {
            session = HibernateUtils.getSession();
            session.beginTransaction();
            
            Query query = session.createQuery("select s from Student s");
            //啟用查詢查詢緩存
            query.setCacheable(true);
            
            List students = query.list(); 
            for (Iterator iter=students.iterator();iter.hasNext(); ) {
                Student student = (Student)iter.next();
                System.out.println(student.getName());
            }
            session.getTransaction().commit();
        }catch(Exception e) {
            e.printStackTrace();
            session.getTransaction().rollback();
        }finally {
            HibernateUtils.closeSession(session);
        }
        
        System.out.println("-------------------------------------");
        
        try {
            session = HibernateUtils.getSession();
            session.beginTransaction();
            
            Query query = session.createQuery("select s from Student s");
            //啟用查詢查詢緩存
            query.setCacheable(true);
            
            //會(huì)發(fā)出n條查詢語(yǔ)句,因?yàn)殚_啟了查詢緩存,關(guān)閉了二級(jí)緩存,那么查詢緩存會(huì)緩存實(shí)體對(duì)象的id
            //所以hibernate會(huì)根據(jù)實(shí)體對(duì)象的id去查詢相應(yīng)的實(shí)體,如果緩存中不存在相應(yīng)的
            //實(shí)體那么將發(fā)出根據(jù)實(shí)體id查詢的sql語(yǔ)句,否則不會(huì)發(fā)出sql使用緩存中的數(shù)據(jù)
            List students = query.list(); 
            for (Iterator iter=students.iterator();iter.hasNext(); ) {
                Student student = (Student)iter.next();
                System.out.println(student.getName());
            }
            session.getTransaction().commit();
        }catch(Exception e) {
            e.printStackTrace();
            session.getTransaction().rollback();
        }finally {
            HibernateUtils.closeSession(session);
        }
    }

說(shuō)明:這里我們開啟了查詢緩存,那么在第二次使用list方法的時(shí)候就會(huì)出現(xiàn)N+1問(wèn)題,因?yàn)椴樵兙彺鏁?huì)保存普通屬性id,所以出現(xiàn)N+1問(wèn)題。

/**
     * 開啟查詢緩存,開啟二級(jí)緩存
     * 
     * 開啟兩個(gè)session,分別調(diào)用query.list查詢實(shí)體對(duì)象
     */
    public void testCache6() {
        Session session = null;
        try {
            session = HibernateUtils.getSession();
            session.beginTransaction();
            
            Query query = session.createQuery("select s from Student s");
            //啟用查詢查詢緩存
            query.setCacheable(true);
            
            List students = query.list(); 
            for (Iterator iter=students.iterator();iter.hasNext(); ) {
                Student student = (Student)iter.next();
                System.out.println(student.getName());
            }
            session.getTransaction().commit();
        }catch(Exception e) {
            e.printStackTrace();
            session.getTransaction().rollback();
        }finally {
            HibernateUtils.closeSession(session);
        }
        
        System.out.println("-------------------------------------");
        
        try {
            session = HibernateUtils.getSession();
            session.beginTransaction();
            
            Query query = session.createQuery("select s from Student s");
            //啟用查詢查詢緩存
            query.setCacheable(true);
            
            //不會(huì)發(fā)出查詢sql,因?yàn)殚_啟了二級(jí)緩存和查詢緩存,查詢緩存緩存了實(shí)體對(duì)象的id列表
            //hibernate會(huì)根據(jù)實(shí)體對(duì)象的id列表到二級(jí)緩存中取得相應(yīng)的數(shù)據(jù)
            List students = query.list(); 
            for (Iterator iter=students.iterator();iter.hasNext(); ) {
                Student student = (Student)iter.next();
                System.out.println(student.getName());
            }
            session.getTransaction().commit();
        }catch(Exception e) {
            e.printStackTrace();
            session.getTransaction().rollback();
        }finally {
            HibernateUtils.closeSession(session);
        }
    }

說(shuō)明:這里我們將查詢緩存和二級(jí)緩存之后使用list方法去查詢實(shí)體對(duì)象,那么在第二次使用list方法去查詢的時(shí)候就不會(huì)再次發(fā)出sql語(yǔ)句了。

最后:

  • 1.查詢緩存的生命周期和session的生命周期沒有任何關(guān)系;

  • 2.和一、二級(jí)緩存一樣,查詢緩存對(duì)迭代query.iterate()不起作用,只對(duì)list起作用。

  • 3.開啟查詢緩存,關(guān)閉二級(jí)緩存,開啟兩個(gè)session,分別調(diào)用query.list查詢實(shí)體對(duì)象時(shí),在第二次查詢時(shí)會(huì)發(fā)出N條查詢語(yǔ)句,這是因?yàn)殚_啟了查詢緩存,關(guān)閉了二級(jí)緩存,那么查詢緩存會(huì)緩存實(shí)體對(duì)象的id,之后再次查詢時(shí),Hibernate會(huì)根據(jù)實(shí)體對(duì)象的id去查詢相應(yīng)的實(shí)體,如果緩存中不存在相應(yīng)的實(shí)體,那么將根據(jù)實(shí)體id發(fā)出N條查詢語(yǔ)句,否則直接從緩存中取得相應(yīng)的數(shù)據(jù),不會(huì)發(fā)出sql。而如果開啟了二級(jí)緩存則不會(huì)出現(xiàn)上面的N+1問(wèn)題。

  • 4.由于當(dāng)前關(guān)聯(lián)的表發(fā)生修改的時(shí)候查詢緩存立即失效,所以其利用率不高,比如第一次查詢高一班到高三班的所有學(xué)生,而在此過(guò)程中高二班學(xué)生有變化,那么查詢緩存就會(huì)失效,這樣導(dǎo)致利用率比較低。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,739評(píng)論 6 534
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,634評(píng)論 3 419
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人,你說(shuō)我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,653評(píng)論 0 377
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,063評(píng)論 1 314
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 71,835評(píng)論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,235評(píng)論 1 324
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,315評(píng)論 3 442
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,459評(píng)論 0 289
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 49,000評(píng)論 1 335
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 40,819評(píng)論 3 355
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 43,004評(píng)論 1 370
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,560評(píng)論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,257評(píng)論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,676評(píng)論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,937評(píng)論 1 288
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 51,717評(píng)論 3 393
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 48,003評(píng)論 2 374

推薦閱讀更多精彩內(nèi)容

  • 這部分主要是開源Java EE框架方面的內(nèi)容,包括Hibernate、MyBatis、Spring、Spring ...
    雜貨鋪老板閱讀 1,414評(píng)論 0 2
  • Hibernate是一個(gè)開放源代碼的對(duì)象關(guān)系映射框架,它對(duì)JDBC進(jìn)行了非常輕量級(jí)的對(duì)象封裝,它將POJO與數(shù)據(jù)庫(kù)...
    蘭緣小妖閱讀 1,218評(píng)論 1 18
  • 1. Java基礎(chǔ)部分 基礎(chǔ)部分的順序:基本語(yǔ)法,類相關(guān)的語(yǔ)法,內(nèi)部類的語(yǔ)法,繼承相關(guān)的語(yǔ)法,異常的語(yǔ)法,線程的語(yǔ)...
    子非魚_t_閱讀 31,722評(píng)論 18 399
  • web service 相關(guān) 什么是Web Service? 答:從表面上看,Web Service就是一個(gè)應(yīng)用程...
    niuben閱讀 933評(píng)論 0 3
  • 這幾天因?yàn)殚_學(xué)在即、兒子的暑假作業(yè)未寫、兒子每天都會(huì)說(shuō)幾遍補(bǔ)作業(yè)卻遲遲沒有行動(dòng)等一系列的情況,讓我的情緒又有點(diǎn)低落...
    瑾言珅行閱讀 182評(píng)論 1 3