日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問(wèn) 生活随笔!

生活随笔

當(dāng)前位置: 首頁(yè) > 编程资源 > 编程问答 >内容正文

编程问答

浅谈Mybatis的一级缓存和二级缓存

發(fā)布時(shí)間:2024/4/15 编程问答 40 豆豆
生活随笔 收集整理的這篇文章主要介紹了 浅谈Mybatis的一级缓存和二级缓存 小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.

MyBatis的緩存機(jī)制

緩存的引入

當(dāng)我們大量執(zhí)行重復(fù)的查詢SQL語(yǔ)句的時(shí)候,會(huì)頻繁的和數(shù)據(jù)庫(kù)進(jìn)行通信,會(huì)增加查詢時(shí)間等影響用戶體驗(yàn)的問(wèn)題,可以通過(guò)緩存,以降低網(wǎng)絡(luò)流量,使網(wǎng)站加載速度更快.

MyBatis的一級(jí)緩存

默認(rèn)情況下,MyBatis只啟用了本地的會(huì)話緩存,它僅僅對(duì)一個(gè)會(huì)話中的數(shù)據(jù)進(jìn)行緩存。這也就是大家常說(shuō)的MyBatis一級(jí)緩存,一級(jí)緩存的作用域是SqlSession。

第1種情況:同個(gè)session進(jìn)行兩次相同查詢

代碼驗(yàn)證

//測(cè)試一級(jí)緩存@Testpublic void testCache() throws IOException {//加載配置文件InputStream rs = Resources.getResourceAsStream("MyBatisConfig.xml");//獲取工廠建造類SqlSessionFactoryBuilder sqlSessionFactoryBuilder = new SqlSessionFactoryBuilder();//獲取工廠對(duì)象SqlSessionFactory build = sqlSessionFactoryBuilder.build(rs);//獲取SqlSession對(duì)象SqlSession sqlSession = build.openSession(true);//得到代理的對(duì)象StudentMapper mapper = sqlSession.getMapper(StudentMapper.class);//根據(jù)名字查詢Student stu1 = mapper.findStudentByName("小花花");Student stu2 = mapper.findStudentByName("小花花");/*這里相等,說(shuō)明默認(rèn)開(kāi)啟一級(jí)緩存,就是在同一個(gè)sqlSession域中有效,只要查詢相同的語(yǔ)句,就是從一級(jí)緩存中獲取*/System.out.println("兩次查詢結(jié)果的學(xué)生對(duì)象是否是同一個(gè):"+(stu1==stu2));//關(guān)閉會(huì)話sqlSession.close();rs.close();}

執(zhí)行結(jié)果


由上面的執(zhí)行結(jié)果得知
MyBatis是默認(rèn)開(kāi)啟一級(jí)緩存的,作用域是同一個(gè)SqlSession,在同一個(gè)sqlSession中查詢相同的sql語(yǔ)句,第一次查詢后會(huì)將結(jié)果對(duì)象存放在一級(jí)緩存中,在后面查詢相同的sql語(yǔ)句的時(shí)候會(huì)在一級(jí)緩存中去獲取,不用在與數(shù)據(jù)庫(kù)進(jìn)行通信,大大降低了查詢速度.

第2種情況:同個(gè)session進(jìn)行兩次不同的查詢。

** 代碼演示**

//測(cè)試一級(jí)緩存@Testpublic void testCache() throws IOException {//加載配置文件InputStream rs = Resources.getResourceAsStream("MyBatisConfig.xml");//獲取工廠建造類SqlSessionFactoryBuilder sqlSessionFactoryBuilder = new SqlSessionFactoryBuilder();//獲取工廠對(duì)象SqlSessionFactory build = sqlSessionFactoryBuilder.build(rs);//獲取SqlSession對(duì)象SqlSession sqlSession = build.openSession(true);//得到代理的對(duì)象StudentMapper mapper = sqlSession.getMapper(StudentMapper.class);//根據(jù)名字查詢Student stu1 = mapper.findStudentByName("小花花");Student stu2 = mapper.findStudentByName("小付");System.out.println("兩次查詢結(jié)果的學(xué)生對(duì)象是否是同一個(gè):"+(stu1==stu2));//關(guān)閉會(huì)話sqlSession.close();rs.close();}

控制臺(tái)輸出

第3種情況:不同session,進(jìn)行相同查詢。

** 代碼演示**

//測(cè)試一級(jí)緩存@Testpublic void testCache() throws IOException {//加載配置文件InputStream rs = Resources.getResourceAsStream("MyBatisConfig.xml");//獲取工廠建造類SqlSessionFactoryBuilder sqlSessionFactoryBuilder = new SqlSessionFactoryBuilder();//獲取工廠對(duì)象SqlSessionFactory build = sqlSessionFactoryBuilder.build(rs);//獲取SqlSession對(duì)象SqlSession sqlSession = build.openSession(true);StudentMapper mapper = sqlSession.getMapper(StudentMapper.class);//獲取SqlSession2對(duì)象SqlSession sqlSession2 = build.openSession(true);StudentMapper mapper2 = sqlSession2.getMapper(StudentMapper.class);//根據(jù)名字查詢Student stu1 = mapper.findStudentByName("小花花");Student stu2 = mapper2.findStudentByName("小花花");System.out.println("兩次查詢結(jié)果的學(xué)生對(duì)象是否是同一個(gè):"+(stu1==stu2));//關(guān)閉會(huì)話sqlSession.close();rs.close();}

控制臺(tái)輸出

第4種情況:同個(gè)session,查詢之后更新數(shù)據(jù),再次查詢相同的語(yǔ)句。

代碼演示

//測(cè)試一級(jí)緩存@Testpublic void testCache() throws IOException {//加載配置文件InputStream rs = Resources.getResourceAsStream("MyBatisConfig.xml");//獲取工廠建造類SqlSessionFactoryBuilder sqlSessionFactoryBuilder = new SqlSessionFactoryBuilder();//獲取工廠對(duì)象SqlSessionFactory build = sqlSessionFactoryBuilder.build(rs);//獲取SqlSession對(duì)象SqlSession sqlSession = build.openSession(true);//得到代理的對(duì)象StudentMapper mapper = sqlSession.getMapper(StudentMapper.class);//根據(jù)名字查詢Student stu1 = mapper.findStudentByName("小花花");//修改數(shù)據(jù)stu1.setAge(18);//執(zhí)行更新語(yǔ)句mapper.updateStudent(stu1);//再次查詢Student stu2 = mapper.findStudentByName("小花花");System.out.println("兩次查詢結(jié)果的學(xué)生對(duì)象是否是同一個(gè):"+(stu1==stu2));//關(guān)閉會(huì)話sqlSession.close();rs.close();}

控制臺(tái)輸出

一級(jí)緩存結(jié)論

MyBatis一級(jí)緩存的運(yùn)行過(guò)程是這樣的:執(zhí)行SQL語(yǔ)句的過(guò)程中,首次執(zhí)行它時(shí)從數(shù)據(jù)庫(kù)獲取的所有數(shù)據(jù)會(huì)被存儲(chǔ)在一段高速緩存中,今后執(zhí)行這條語(yǔ)句時(shí)就會(huì)從高速緩存中讀取結(jié)果,而不是再次查詢數(shù)據(jù)庫(kù)。MyBatis提供了默認(rèn)下基于Java HashMap的緩存實(shí)現(xiàn)。

重點(diǎn)是要明白:MyBatis執(zhí)行SQL語(yǔ)句之后,這條語(yǔ)句的執(zhí)行結(jié)果被緩存,以后再執(zhí)行這條語(yǔ)句的時(shí)候,會(huì)直接從緩存中拿結(jié)果,而不是再次執(zhí)行SQL。但是一旦執(zhí)行新增或更新或刪除操作,緩存就會(huì)被清除。下面將分情況來(lái)驗(yàn)證一下。

很明顯,以上各種情況驗(yàn)證了一級(jí)緩存的概念,在同個(gè)SqlSession中,查詢語(yǔ)句相同的sql會(huì)被緩存,但是一旦執(zhí)行新增或更新或刪除操作,緩存就會(huì)被清除。

MyBatis的二級(jí)緩存(全局緩存)

引入

MyBatis 一級(jí)緩存最大的共享范圍就是一個(gè)SqlSession內(nèi)部,那么如果多個(gè) SqlSession 需要共享緩存,則需要開(kāi)啟二級(jí)緩存.

當(dāng)二級(jí)緩存開(kāi)啟后,同一個(gè)命名空間(namespace) 所有的操作語(yǔ)句,都影響著一個(gè)共同的 cache,也就是二級(jí)緩存被多個(gè) SqlSession 共享,是一個(gè)全局的變量。當(dāng)開(kāi)啟緩存后,數(shù)據(jù)的查詢執(zhí)行的流程就是 二級(jí)緩存 -> 一級(jí)緩存 -> 數(shù)據(jù)庫(kù)。

二級(jí)緩存開(kāi)啟條件
二級(jí)緩存默認(rèn)是不開(kāi)啟的,需要手動(dòng)開(kāi)啟二級(jí)緩存,實(shí)現(xiàn)二級(jí)緩存的時(shí)候,MyBatis要求返回的**POJO必須是可序列化的**。開(kāi)啟二級(jí)緩存的條件也是比較簡(jiǎn)單.

在MyBatis核心配置文件中配置開(kāi)啟二級(jí)緩存

<!-- 通知 MyBatis 框架開(kāi)啟二級(jí)緩存 --> <settings><setting name="cacheEnabled" value="true"/> </settings>

在mapper配置文件中開(kāi)啟當(dāng)前命名空間的查詢結(jié)果存放在二級(jí)緩存中

<!-- 表示DEPT表查詢結(jié)果保存到二級(jí)緩存(共享緩存)cache 標(biāo)簽有多個(gè)屬性,一起來(lái)看一些這些屬性分別代表什么意義eviction: 緩存回收策略,有這幾種回收策略LRU - 最近最少回收,移除最長(zhǎng)時(shí)間不被使用的對(duì)象FIFO - 先進(jìn)先出,按照緩存進(jìn)入的順序來(lái)移除它們SOFT - 軟引用,移除基于垃圾回收器狀態(tài)和軟引用規(guī)則的對(duì)象WEAK - 弱引用,更積極的移除基于垃圾收集器和弱引用規(guī)則的對(duì)象默認(rèn)是 LRU 最近最少回收策略flushinterval 緩存刷新間隔,緩存多長(zhǎng)時(shí)間刷新一次,默認(rèn)不清空,設(shè)置一個(gè)毫秒值readOnly: 是否只讀;true 只讀,MyBatis 認(rèn)為所有從緩存中獲取數(shù)據(jù)的操作都是只讀操作,不會(huì)修改數(shù)據(jù)。MyBatis 為了加快獲取數(shù)據(jù),直接就會(huì)將數(shù)據(jù)在緩存中的引用交給用戶。不安全,速度快。讀寫(默認(rèn)):MyBatis 覺(jué)得數(shù)據(jù)可能會(huì)被修改size : 緩存存放多少個(gè)元素type: 指定自定義緩存的全類名(實(shí)現(xiàn)Cache 接口即可)blocking: 若緩存中找不到對(duì)應(yīng)的key,是否會(huì)一直blocking,直到有對(duì)應(yīng)的數(shù)據(jù)進(jìn)入緩存。我這里就使用默認(rèn)的,不去設(shè)置參數(shù)--><cacheeviction="FIFO"flushInterval="60000"size="512"readOnly="true"/>

實(shí)體類對(duì)象要Serializable序列化

@Data //這個(gè)注解是lombok包下的,可以自動(dòng)幫我們添加getset方法等 public class Student implements Serializable {private Integer sid;private String name;private Integer age; }

代碼驗(yàn)證

//測(cè)試一級(jí)緩存@Testpublic void testCache() throws IOException {//加載配置文件InputStream rs = Resources.getResourceAsStream("MyBatisConfig.xml");//獲取工廠建造類SqlSessionFactoryBuilder sqlSessionFactoryBuilder = new SqlSessionFactoryBuilder();//獲取工廠對(duì)象SqlSessionFactory build = sqlSessionFactoryBuilder.build(rs);//獲取SqlSession對(duì)象SqlSession sqlSession = build.openSession(true);StudentMapper mapper = sqlSession.getMapper(StudentMapper.class);//根據(jù)名字查詢Student stu1 = mapper.findStudentByName("小花花");System.out.println("第一次查出的對(duì)象"+stu1);//關(guān)閉會(huì)話后,會(huì)將一級(jí)緩存的數(shù)據(jù)保存在二級(jí)緩存中sqlSession.close();//獲取SqlSession2對(duì)象SqlSession sqlSession2 = build.openSession(true);StudentMapper mapper2 = sqlSession2.getMapper(StudentMapper.class);//新的會(huì)話查詢就會(huì)從二級(jí)緩存中獲取數(shù)據(jù)Student stu2 = mapper2.findStudentByName("小花花");System.out.println("第二次查出的對(duì)象"+stu1);System.out.println("兩次查詢結(jié)果的學(xué)生對(duì)象是否是同一個(gè):"+(stu1==stu2));//關(guān)閉會(huì)話sqlSession2.close();rs.close();}

控制臺(tái)輸出

圖解查詢語(yǔ)句執(zhí)行流程

總結(jié)

以上是生活随笔為你收集整理的浅谈Mybatis的一级缓存和二级缓存的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。

如果覺(jué)得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。