亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Spring?data?jpa緩存機(jī)制使用總結(jié)

 更新時(shí)間:2023年12月01日 08:44:30   作者:qq_34485381  
這篇文章主要介紹了Spring?data?jpa緩存機(jī)制使用總結(jié),具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教

Spring data jpa緩存機(jī)制

Spring data jpa  的使用讓我們操作數(shù)據(jù)庫(kù)變得非常簡(jiǎn)單,開(kāi)發(fā)人員只需要編寫(xiě)repository接口,Spring將自動(dòng)提供實(shí)現(xiàn),尤其是基礎(chǔ)的的CURD 操作,為我們封裝好的同時(shí)也做了一些性能上的優(yōu)化。

但也正因?yàn)槿绱耍@些基礎(chǔ)的操作的背后并不是那么簡(jiǎn)單,稍有不慎就會(huì)得到我們意料之外的結(jié)果,接下來(lái)列舉一些工作中遇到的問(wèn)題。

一、案例

項(xiàng)目中遇到過(guò)這樣一個(gè)問(wèn)題,repository繼承了CrudRepository接口,直接使用save(S entity) 方法進(jìn)行數(shù)據(jù)保存,但是因?yàn)槟硞€(gè)字段的唯一約束沖突了,導(dǎo)致保存失敗并拋出了異常,但是save方法后的代碼邏輯卻執(zhí)行了,將數(shù)據(jù)保存到redis,這導(dǎo)致了數(shù)據(jù)庫(kù)和redis數(shù)據(jù)不一致。

代碼代碼大概是這樣子:

    @Override
    @Transactional
    public void save(SomeThingVo vo){
        SomeThingEntity entity = new SomeThingEntity();
        BeanUtils.copyProperties(vo,entity);
        //保存至數(shù)據(jù)庫(kù)
        someThingRepository.save(entity);
        //緩存
        cacheSomeThing(entity);
         //做一些其他事
        doSomeThingElse();
    }

然后對(duì)這個(gè)操作進(jìn)行了debug,發(fā)現(xiàn)到save方法結(jié)束,是沒(méi)有拋出異常的,然后繼續(xù)進(jìn)行保存redis等操作,直到方法結(jié)束才拋出了異常。

這時(shí)注意到了@Transactional注解加在了這個(gè)方法之上,那就是事務(wù)提交時(shí)才會(huì)報(bào)出 唯一約束沖突的異常,再聯(lián)想到Spring data Jpa的是用Hibernate實(shí)現(xiàn)的 , Hibernate是有緩存機(jī)制的,猜想不使用jpa自帶的save方法,就可以在保存時(shí)直接拋異常,而不執(zhí)行之后的代碼,然后進(jìn)行嘗試,的確如此;還有一種解決方式是使用saveAndFlush方法,立馬將緩存中的實(shí)體bean刷入數(shù)據(jù)庫(kù)。

二、分析

Hibernate緩存包括兩大類(lèi):一級(jí)緩存和二級(jí)緩存。

一級(jí)緩存又稱為“Session的緩存”,它是內(nèi)置的,不能被卸載(不能被卸載的意思就是這種緩存不具有可選性,必須有的功能,不可以取消session緩存)。由于Session對(duì)象的生命周期通常對(duì)應(yīng)一個(gè)數(shù)據(jù)庫(kù)事務(wù)或者一個(gè)應(yīng)用事務(wù),因此它的緩存是事務(wù)范圍的緩存在第一級(jí)緩存中,持久化類(lèi)的每個(gè)實(shí)例都具有唯一的OID。我們使用@Transactional 注解時(shí),JpaTransactionManager會(huì)在開(kāi)啟事務(wù)前打開(kāi)一個(gè)session,將事務(wù)綁定在這個(gè)session上,事務(wù)結(jié)束session關(guān)閉,所以后續(xù)內(nèi)容將以粗略以事務(wù)作為一級(jí)緩存的生存時(shí)段。

二級(jí)緩存又稱為“SessionFactory的緩存”,由于SessionFactory對(duì)象的生命周期和應(yīng)用程序的整個(gè)過(guò)程對(duì)應(yīng),因此二級(jí)緩存是進(jìn)程范圍或者集群范圍的緩存,有可能出現(xiàn)并發(fā)問(wèn)題,因此需要采用適當(dāng)?shù)牟l(fā)訪問(wèn)策略。第二級(jí)緩存是可選的,是一個(gè)可配置的插件,在默認(rèn)情況下,SessionFactory不會(huì)啟用這個(gè)插件,二級(jí)緩存應(yīng)用場(chǎng)景局限性比較大,適用于數(shù)據(jù)要求的實(shí)時(shí)性和準(zhǔn)確性不高、變動(dòng)很少的情況,此次我們僅針對(duì)一級(jí)緩存進(jìn)行詳細(xì)說(shuō)明。

我們使用CrudRepository.save() 方法保存或更新對(duì)象的流程如下

  

從上圖可以看出每次save方法執(zhí)行時(shí)都會(huì)用主鍵向數(shù)據(jù)庫(kù)發(fā)起一次查詢,來(lái)判斷是更新還是插入,此時(shí)spring data jpa 不會(huì)立馬向數(shù)據(jù)庫(kù)發(fā)送命令,而是將這條數(shù)據(jù)保存在一級(jí)緩存之中,然后返回緩存中實(shí)體對(duì)象,接下來(lái)繼續(xù)執(zhí)行后續(xù)的代碼。

如果想更新這條數(shù)據(jù)的值,可以直接修改這個(gè)實(shí)體對(duì)象,jpa會(huì)在事前提交之前的某個(gè)點(diǎn)(具體后面會(huì)說(shuō)明)自動(dòng)將這些變更的數(shù)據(jù)保存至數(shù)據(jù)庫(kù),并且在事務(wù)期間查詢這條數(shù)據(jù)都是優(yōu)先從緩存中獲取數(shù)據(jù)。

一級(jí)緩存的作用還是很明顯的,在整個(gè)事務(wù)中,在對(duì)同一條數(shù)據(jù)進(jìn)行了保存更新查詢操作都會(huì)以盡量少地請(qǐng)求數(shù)據(jù)庫(kù)的方式進(jìn)行優(yōu)化,降低了網(wǎng)絡(luò)io開(kāi)銷(xiāo)。     

三、聯(lián)想

有利就有弊,就像第一部分描述的,因?yàn)檠舆t提交 ,數(shù)據(jù)的正確性驗(yàn)證(數(shù)據(jù)庫(kù)限制方面,比如約束)并沒(méi)有立馬執(zhí)行,有時(shí)候完全是我們不能承受的,我們想要的效果并不是這樣。

接下來(lái)設(shè)想一下其他場(chǎng)景:

1、何時(shí)會(huì)將數(shù)據(jù)提交至數(shù)據(jù)庫(kù)?

實(shí)際上這中情況是不存在的。

測(cè)試代碼和結(jié)果如下:

 @Transactional(rollbackFor = {Exception.class})
   public SomeThingEntity save(SomeThingVo vo) {
       SomeThingEntity entity = new SomeThingEntity();
       BeanUtils.copyProperties(vo,entity);
       SomeThingEntity someThingEntity = someThingRepository.save(entity);
       log.info("保存方法結(jié)束");
       String code = "GOODS_" + someThingEntity.getCode() ;
       someThingEntity.setCode(code);
       log.info("開(kāi)始查找");
      SomeThingEntity searchThing = someThingRepository.searchByCode(code);
      log.info("查找結(jié)果:{}" , searchThing);
      SomeThingEntity getThing = someThingRepository.getOne(someThingEntity.getId());
      log.info("執(zhí)行了一次JPA查詢\n\r" +
              "someThingEntity == getThing : {}\n\r" +
              "searchThing == getThing :{}" , someThingEntity == getThing , searchThing == getThing );
      return someThingEntity;
   }


打印日志:

1 Hibernate: select somethinge0_.id as id1_3_0_, somethinge0_.code as code2_3_0_, somethinge0_.description as descript3_3_0_, somethinge0_.price as price4_3_0_ from tb_something somethinge0_ where somethinge0_.id=?

2 保存方法結(jié)束

3 開(kāi)始查找

4 Hibernate: insert into tb_something (code, description, price, id) values (?, ?, ?, ?)

5 Hibernate: update tb_something set code=?, description=?, price=? where id=?

6 Hibernate: select somethinge0_.id as id1_3_, somethinge0_.code as code2_3_, somethinge0_.description as descript3_3_, somethinge0_.price as price4_3_ from tb_something somethinge0_ where somethinge0_.code=?

7 查找結(jié)果:SomeThingEntity(id=5, code=GOODS_005, price=100, description=書(shū)包)

8 執(zhí)行了一次JPA查詢

9 someThingEntity == getThing : true

10 searchThing == getThing :true

11 Hibernate: update tb_something set code=?, description=?, price=? where id=?

從日志可見(jiàn):

  • save()方法執(zhí)行時(shí)只打印了一個(gè)查詢sql       
  • someThingRepository.searchByCode()方法執(zhí)行前各打印了一條插入sql和更新sql
  • someThingRepository.searchByCode() 進(jìn)行了查詢  
  • getOne()并沒(méi)有打印sql,直接獲取緩存中的對(duì)象

最后比對(duì)這些實(shí)體都是同一個(gè)對(duì)象,即緩存中的對(duì)象。

將代碼中someThingRepository.searchByCode方法改為其他讀寫(xiě)語(yǔ)句,嘗試多次,得出以下結(jié)論:

(1)未提交至數(shù)據(jù)庫(kù)的操作會(huì)在下次請(qǐng)求到數(shù)據(jù)庫(kù)時(shí)一起提交至數(shù)據(jù)庫(kù)執(zhí)行

(2)在事務(wù)提交前存在未提交的數(shù)據(jù),會(huì)提交至數(shù)據(jù)庫(kù)執(zhí)行

2、實(shí)體對(duì)象加入緩存后

我們寫(xiě)sql更新數(shù)據(jù),再用自己的sql獲取這條數(shù)據(jù),得到的是緩存中的數(shù)據(jù)還是更新后的數(shù)據(jù)

這次測(cè)試代碼和結(jié)果如下:

@Transactional(rollbackFor = {Exception.class})
 public SomeThingEntity save(SomeThingVo vo) {
       SomeThingEntity entity = new SomeThingEntity();
       BeanUtils.copyProperties(vo,entity);
       SomeThingEntity someThingEntity = someThingRepository.save(entity);
       log.info("開(kāi)始更新");
       Integer fenPrice = entity.getPrice() * 100;
       someThingRepository.updatePriceByCode(someThingEntity.getCode(),fenPrice);
      //Session session = (Session) entityManger.getDelegate();
      //session.clear();
      SomeThingEntity searchThing = someThingRepository.searchByCode(someThingEntity.getCode());
     log.info("searchThing = {}",searchThing);
     log.info("searchThing == someThingEntity {}",searchThing == someThingEntity);
     //someThingEntity.setDescription("");
     return someThingEntity;
 }

傳入?yún)?shù):{id=20,code='GOODS_020",price=100,description="書(shū)包"}

打印日志:

1Hibernate: select somethinge0_.id as id1_3_0_, somethinge0_.code as code2_3_0_, somethinge0_.description as descript3_3_0_, somethinge0_.price as price4_3_0_ from tb_something somethinge0_ where somethinge0_.id=?

2 開(kāi)始更新

3 Hibernate: insert into tb_something (code, description, price, id) values (?, ?, ?, ?)

4 Hibernate: update tb_something set price=? where code=?

5 Hibernate: select * from tb_something  where code = ? 

6 searchThing = SomeThingEntity(id=20, code=GOODS_020, price=100, description=書(shū)包)

7 searchThing == someThingEntity true

數(shù)據(jù)庫(kù)結(jié)果:{id=20,code='GOODS_020",price=10000,description="書(shū)包"}

從日志中可見(jiàn):

someThingRepository.updatePriceByCode(someThingEntity.getCode(),fenPrice) 執(zhí)行打印了相關(guān)更新sql(第4行日志),目的  將price由100 改為10000

我們的查詢方法向數(shù)據(jù)庫(kù)發(fā)起了查詢;

打印的結(jié)果不是我們更新后的結(jié)果,price仍然為100;

查詢的結(jié)果對(duì)象和緩存中的對(duì)象比較,是同一個(gè)對(duì)象;  

測(cè)試說(shuō)明:

執(zhí)行我們的查詢方法后,jpa返回給我們的仍然是緩存中的值,這樣子的話我們?cè)谶@個(gè)事務(wù)中怎么查詢都拿不到我們變更后的值! jpa不會(huì)根據(jù)我們的update方法自動(dòng)刷新緩存,后邊查詢出來(lái)的數(shù)據(jù)也不會(huì)覆蓋緩存中的數(shù)據(jù)。

那么一些同學(xué)可能會(huì)把一個(gè)事務(wù)涵蓋內(nèi)容的比較多,在頂層的service就加了@Transactional ,就可能在一些操作上進(jìn)入了這樣的場(chǎng)景,在緩存存在的情況,手動(dòng)update,后續(xù)有去查詢使用,最終使用了錯(cuò)誤的數(shù)據(jù)。

如果非要在當(dāng)前事務(wù)中查詢到正確數(shù)據(jù)的話,那就手動(dòng)清除session中的緩存吧(上述代碼中 10、11行)。

另外,放開(kāi)上述代碼中的15行,最終保存在數(shù)據(jù)庫(kù)的結(jié)果為 {id=20,code='GOODS_020",price=100,description=""} ,price的值會(huì)被緩存中的覆蓋。

總結(jié)

Spring data jpa 的這些操作都是簡(jiǎn)單常用而又容易忽視的,我們?cè)谑褂脮r(shí)要考慮一下是否得當(dāng)。

對(duì)于這樣的緩存機(jī)制我們要做的是 將事務(wù)控制在合適的范圍,將不需要在事務(wù)中執(zhí)行的內(nèi)容就移出去;在需要sql明確執(zhí)行好的情況,就主要避開(kāi)使用會(huì)延遲提交的方法。

規(guī)范的代碼和設(shè)計(jì)是質(zhì)量的一個(gè)重要保證之一。

以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。

相關(guān)文章

  • java線程之死鎖

    java線程之死鎖

    這篇文章主要介紹了Java線程之死鎖,死鎖是這樣一種情形-多個(gè)線程同時(shí)被阻塞,它們中的一個(gè)或者全部都在等待某個(gè)資源被釋放。由于線程被無(wú)限期地阻塞,因此程序不可能正常終止
    2022-05-05
  • 淺談java中為什么實(shí)體類(lèi)需要實(shí)現(xiàn)序列化

    淺談java中為什么實(shí)體類(lèi)需要實(shí)現(xiàn)序列化

    下面小編就為大家?guī)?lái)一篇淺談java中為什么實(shí)體類(lèi)需要實(shí)現(xiàn)序列化。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2017-05-05
  • 在Java的Hibernate框架中使用SQL語(yǔ)句的簡(jiǎn)單介紹

    在Java的Hibernate框架中使用SQL語(yǔ)句的簡(jiǎn)單介紹

    這篇文章主要介紹了在Java的Hibernate框架中使用SQL語(yǔ)句的方法,Hibernate是Java的SSH三大web開(kāi)發(fā)框架之一,需要的朋友可以參考下
    2016-01-01
  • 詳解spring mvc 請(qǐng)求轉(zhuǎn)發(fā)和重定向

    詳解spring mvc 請(qǐng)求轉(zhuǎn)發(fā)和重定向

    這篇文章主要介紹了詳解spring mvc 請(qǐng)求轉(zhuǎn)發(fā)和重定向,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2017-02-02
  • Java8 lambda表達(dá)式2種常用方法代碼解析

    Java8 lambda表達(dá)式2種常用方法代碼解析

    這篇文章主要介紹了Java8 lambda表達(dá)式2種常用方法代碼解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-08-08
  • spring?bean標(biāo)簽中的init-method和destroy-method詳解

    spring?bean標(biāo)簽中的init-method和destroy-method詳解

    這篇文章主要介紹了spring?bean標(biāo)簽中的init-method和destroy-method,在很多項(xiàng)目中,經(jīng)常在xml配置文件中看到init-method 或者 destroy-method ,因此整理收集下,方便以后參考和學(xué)習(xí),需要的朋友可以參考下
    2023-04-04
  • 使用maven構(gòu)建java9 service實(shí)例詳解

    使用maven構(gòu)建java9 service實(shí)例詳解

    本篇文章主要介紹了使用maven構(gòu)建java9 service實(shí)例詳解,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2018-02-02
  • IDEA啟動(dòng)Tomcat時(shí)控制臺(tái)出現(xiàn)亂碼問(wèn)題及解決

    IDEA啟動(dòng)Tomcat時(shí)控制臺(tái)出現(xiàn)亂碼問(wèn)題及解決

    這篇文章主要介紹了IDEA啟動(dòng)Tomcat時(shí)控制臺(tái)出現(xiàn)亂碼問(wèn)題及解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-02-02
  • Java進(jìn)程cpu占用過(guò)高問(wèn)題解決

    Java進(jìn)程cpu占用過(guò)高問(wèn)題解決

    這篇文章主要介紹了Java進(jìn)程cpu占用過(guò)高問(wèn)題解決,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-04-04
  • SpringBoot2.3新特性優(yōu)雅停機(jī)詳解

    SpringBoot2.3新特性優(yōu)雅停機(jī)詳解

    這篇文章主要介紹了SpringBoot2.3新特性優(yōu)雅停機(jī)詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2020-05-05

最新評(píng)論