亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 開發 > Java > 正文

深入了解MyBatis二級緩存

2024-07-14 08:43:19
字體:
來源:轉載
供稿:網友

一、創建Cache的完整過程

我們從SqlSessionFactoryBuilder解析mybatis-config.xml配置文件開始:

Reader reader = Resources.getResourceAsReader("mybatis-config.xml");SqlSessionFactory sqlSessionFactory = new SqlSessionFactoryBuilder().build(reader);

然后是:

XMLConfigBuilder parser = new XMLConfigBuilder(inputStream, environment, properties);return build(parser.parse());

看parser.parse()方法:

parseConfiguration(parser.evalNode("/configuration"));

看處理Mapper.xml文件的位置:

mapperElement(root.evalNode("mappers"));

看處理Mapper.xml的XMLMapperBuilder:

XMLMapperBuilder mapperParser = new XMLMapperBuilder(inputStream, configuration,    resource, configuration.getSqlFragments());mapperParser.parse();

繼續看parse方法:

configurationElement(parser.evalNode("/mapper"));

到這里:

String namespace = context.getStringAttribute("namespace");if (namespace.equals("")) { throw new BuilderException("Mapper's namespace cannot be empty");}builderAssistant.setCurrentNamespace(namespace);cacheRefElement(context.evalNode("cache-ref"));cacheElement(context.evalNode("cache"));

從這里看到namespace就是xml中<mapper>元素的屬性。然后下面是先后處理的cache-ref和cache,后面的cache會覆蓋前面的cache-ref,但是如果一開始cache-ref沒有找到引用的cache,他就不會被覆蓋,會一直到最后處理完成為止,最后如果存在cache,反而會被cache-ref覆蓋。這里是不是看著有點暈、有點亂?所以千萬別同時配置這兩個,實際上也很少有人會這么做。

看看MyBatis如何處理<cache/>:

private void cacheElement(XNode context) throws Exception {  if (context != null) {    String type = context.getStringAttribute("type", "PERPETUAL");    Class<? extends Cache> typeClass = typeAliasRegistry.resolveAlias(type);    String eviction = context.getStringAttribute("eviction", "LRU");    Class<? extends Cache> evictionClass = typeAliasRegistry.resolveAlias(eviction);    Long flushInterval = context.getLongAttribute("flushInterval");    Integer size = context.getIntAttribute("size");    boolean readWrite = !context.getBooleanAttribute("readOnly", false);    boolean blocking = context.getBooleanAttribute("blocking", false);    Properties props = context.getChildrenAsProperties();    builderAssistant.useNewCache(typeClass, evictionClass,       flushInterval, size, readWrite, blocking, props);  }}

從源碼可以看到MyBatis讀取了那些屬性,而且很容易可以到這些屬性的默認值。

創建Java的cache對象方法為builderAssistant.useNewCache,我們看看這段代碼:

public Cache useNewCache(Class<? extends Cache> typeClass,             Class<? extends Cache> evictionClass,             Long flushInterval,             Integer size,             boolean readWrite,             boolean blocking,             Properties props) {  typeClass = valueOrDefault(typeClass, PerpetualCache.class);  evictionClass = valueOrDefault(evictionClass, LruCache.class);  Cache cache = new CacheBuilder(currentNamespace)      .implementation(typeClass)      .addDecorator(evictionClass)      .clearInterval(flushInterval)      .size(size)      .readWrite(readWrite)      .blocking(blocking)      .properties(props)      .build();  configuration.addCache(cache);  currentCache = cache;  return cache;}

從調用該方法的地方,我們可以看到并沒有使用返回值cache,在后面的過程中創建MappedStatement的時候使用了currentCache。

二、使用Cache過程

在系統中,使用Cache的地方在CachingExecutor中:

@Overridepublic <E> List<E> query(    MappedStatement ms, Object parameterObject,     RowBounds rowBounds, ResultHandler resultHandler,     CacheKey key, BoundSql boundSql) throws SQLException { Cache cache = ms.getCache();

獲取cache后,先判斷是否有二級緩存。

只有通過<cache/>,<cache-ref/>或@CacheNamespace,@CacheNamespaceRef標記使用緩存的Mapper.xml或Mapper接口(同一個namespace,不能同時使用)才會有二級緩存。

 if (cache != null) {

如果cache存在,那么會根據sql配置(<insert>,<select>,<update>,<delete>的flushCache屬性來確定是否清空緩存。

  flushCacheIfRequired(ms);

然后根據xml配置的屬性useCache來判斷是否使用緩存(resultHandler一般使用的默認值,很少會null)。

  if (ms.isUseCache() && resultHandler == null) {

確保方法沒有Out類型的參數,mybatis不支持存儲過程的緩存,所以如果是存儲過程,這里就會報錯。

   ensureNoOutParams(ms, parameterObject, boundSql);

沒有問題后,就會從cache中根據key來取值:

   @SuppressWarnings("unchecked")   List<E> list = (List<E>) tcm.getObject(cache, key);

如果沒有緩存,就會執行查詢,并且將查詢結果放到緩存中。

   if (list == null) {    list = delegate.<E>query(ms, parameterObject,       rowBounds, resultHandler, key, boundSql);    tcm.putObject(cache, key, list); // issue #578 and #116   }

返回結果

   return list;  } }

沒有緩存時,直接執行查詢

 return delegate.<E>query(ms, parameterObject, rowBounds, resultHandler, key, boundSql);}

在上面的代碼中tcm.putObject(cache, key, list);這句代碼是緩存了結果。但是實際上直到sqlsession關閉,MyBatis才以序列化的形式保存到了一個Map(默認的緩存配置)中。

三、Cache使用時的注意事項

1. 只能在【只有單表操作】的表上使用緩存

不只是要保證這個表在整個系統中只有單表操作,而且和該表有關的全部操作必須全部在一個namespace下。

2. 在可以保證查詢遠遠大于insert,update,delete操作的情況下使用緩存

這一點不需要多說,所有人都應該清楚。記住,這一點需要保證在1的前提下才可以!

四、避免使用二級緩存

可能會有很多人不理解這里,二級緩存帶來的好處遠遠比不上他所隱藏的危害。

  • 緩存是以namespace為單位的,不同namespace下的操作互不影響。
  • insert,update,delete操作會清空所在namespace下的全部緩存。
  • 通常使用MyBatis Generator生成的代碼中,都是各個表獨立的,每個表都有自己的namespace。

為什么避免使用二級緩存

在符合【Cache使用時的注意事項】的要求時,并沒有什么危害。

其他情況就會有很多危害了。

針對一個表的某些操作不在他獨立的namespace下進行。

例如在UserMapper.xml中有大多數針對user表的操作。但是在一個XXXMapper.xml中,還有針對user單表的操作。

這會導致user在兩個命名空間下的數據不一致。如果在UserMapper.xml中做了刷新緩存的操作,在XXXMapper.xml中緩存仍然有效,如果有針對user的單表查詢,使用緩存的結果可能會不正確。

更危險的情況是在XXXMapper.xml做了insert,update,delete操作時,會導致UserMapper.xml中的各種操作充滿未知和風險。

有關這樣單表的操作可能不常見。但是你也許想到了一種常見的情況。

多表操作一定不能使用緩存

為什么不能?

首先不管多表操作寫到那個namespace下,都會存在某個表不在這個namespace下的情況。

例如兩個表:role和user_role,如果我想查詢出某個用戶的全部角色role,就一定會涉及到多表的操作。

<select id="selectUserRoles" resultType="UserRoleVO"> select * from user_role a,role b where a.roleid = b.roleid and a.userid = #{userid}</select>

像上面這個查詢,你會寫到那個xml中呢??

不管是寫到RoleMapper.xml還是UserRoleMapper.xml,或者是一個獨立的XxxMapper.xml中。如果使用了二級緩存,都會導致上面這個查詢結果可能不正確。

如果你正好修改了這個用戶的角色,上面這個查詢使用緩存的時候結果就是錯的。

這點應該很容易理解。

在我看來,就以MyBatis目前的緩存方式來看是無解的。多表操作根本不能緩存。

如果你讓他們都使用同一個namespace(通過<cache-ref>)來避免臟數據,那就失去了緩存的意義。

看到這里,實際上就是說,二級緩存不能用。整篇文章介紹這么多也沒什么用了。

五、挽救二級緩存?

想更高效率的使用二級緩存是解決不了了。

但是解決多表操作避免臟數據還是有法解決的。解決思路就是通過攔截器判斷執行的sql涉及到那些表(可以用jsqlparser解析),然后把相關表的緩存自動清空。但是這種方式對緩存的使用效率是很低的。

設計這樣一個插件是相當復雜的,既然我沒想著去實現,就不廢話了。

最后還是建議,放棄二級緩存,在業務層使用可控制的緩存代替更好。

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,謝謝大家對VeVb武林網的支持。


注:相關教程知識閱讀請移步到JAVA教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
久久久国产一区二区| 国产精品444| 日韩在线不卡视频| 国产精品久久久亚洲| 尤物tv国产一区| 欧美一区三区三区高中清蜜桃| 国产午夜精品理论片a级探花| 日韩欧美中文字幕在线播放| 久久久女女女女999久久| 97精品在线观看| 91免费看片网站| 欧美电影免费观看高清完整| 亚洲精品国产综合久久| 人人澡人人澡人人看欧美| 国产精品成人一区二区| 国产丝袜精品视频| 综合激情国产一区| 欧美精品少妇videofree| 在线播放日韩专区| 亚洲人a成www在线影院| 91av在线免费观看视频| 国产午夜精品一区二区三区| 欧美激情a∨在线视频播放| 成人女保姆的销魂服务| 久久久视频精品| 国产噜噜噜噜久久久久久久久| 日韩欧美大尺度| 亚洲精品自产拍| 国产在线拍偷自揄拍精品| 国内精品久久久久久影视8| 久久成人这里只有精品| 国产精品中文字幕在线| 国产精品视频26uuu| 成人深夜直播免费观看| 91精品久久久久久久久久久| 综合136福利视频在线| 日韩欧美国产成人| 尤物九九久久国产精品的特点| 久久久久久久久久久成人| 这里只有精品在线播放| 亚洲精品一区中文| 国产欧美中文字幕| 国产一区二区三区视频| 国产亚洲精品久久久久久| 日韩在线欧美在线| 色偷偷偷亚洲综合网另类| 日韩中文理论片| 久久久久久久久久久免费| 超碰日本道色综合久久综合| 综合激情国产一区| 中文字幕在线成人| 欧美性色视频在线| 岛国视频午夜一区免费在线观看| 国产精品pans私拍| 国产精品欧美一区二区| 欧美理论电影在线观看| www.国产精品一二区| 成人午夜激情免费视频| 欧美在线观看网址综合| 高清视频欧美一级| 日韩电影免费观看在线观看| 欧美激情一区二区久久久| 日韩黄色在线免费观看| 日韩中文字幕网址| 亚洲天堂第二页| 久久影院免费观看| 成人免费网视频| 欧美激情视频在线观看| 91精品国产高清久久久久久久久| 国产精品电影网站| 国产成人精品久久久| 久久精品亚洲94久久精品| 欧美性xxxxxxx| 成人午夜在线观看| 91精品成人久久| 色哟哟网站入口亚洲精品| 久久男人资源视频| 亚洲va欧美va国产综合剧情| 亚洲国产精品yw在线观看| 欧美日韩国产页| 91精品国产乱码久久久久久蜜臀| 久久视频在线直播| 91精品国产色综合久久不卡98口| 日本精品一区二区三区在线播放视频| 永久免费精品影视网站| 成人在线中文字幕| 视频一区视频二区国产精品| 91po在线观看91精品国产性色| 国产精品久久久久久久美男| 国产精品成人一区| 欧美疯狂xxxx大交乱88av| 国产精品色悠悠| 欧美成人精品在线| 亚洲精品动漫100p| 日韩av网站在线| 精品视频在线播放色网色视频| 国产精品日韩电影| 国产精品美女主播在线观看纯欲| 国产午夜一区二区| 91久久精品国产91久久| 亚洲天堂av在线免费| 国产成人精品一区二区三区| 国产欧美精品一区二区三区-老狼| 一本大道久久加勒比香蕉| 亚洲第一视频在线观看| 国产精品video| 亚洲激情免费观看| 最新国产精品拍自在线播放| 亚洲区中文字幕| 日韩精品中文字幕在线播放| 国产精品网红直播| 欧美精品成人91久久久久久久| 日韩禁在线播放| 久久人人97超碰精品888| 亚洲最新在线视频| 亚洲专区国产精品| 亚洲一区中文字幕| 日本高清久久天堂| 成人福利视频网| 成人国产精品免费视频| 精品av在线播放| 2019国产精品自在线拍国产不卡| 久久精品国产一区二区三区| 国产精品综合网站| 日韩成人在线免费观看| 国产一区二区黑人欧美xxxx| 国产不卡在线观看| 国产精品视频xxxx| 日韩有码视频在线| 热re99久久精品国产66热| 精品香蕉在线观看视频一| 成人福利在线观看| 日韩av在线影院| 久久青草精品视频免费观看| 日韩精品中文字幕在线观看| 欧美俄罗斯乱妇| 91高潮在线观看| 国产一区视频在线| 欧美成人免费小视频| 国产精品老女人精品视频| 亚洲毛片在线看| 国产精品永久免费| 亚洲精品中文字幕女同| 久久久精品一区| 欧美极品少妇xxxxⅹ免费视频| www.日本久久久久com.| 久久99久久久久久久噜噜| 中文字幕精品久久| 午夜精品三级视频福利| xvideos亚洲人网站| 91av在线网站| 久久久久久91香蕉国产| 欧美精品videos| 日韩精品欧美激情| 91精品国产高清久久久久久91| 国产ts人妖一区二区三区| 久久免费视频观看| 国产精品扒开腿爽爽爽视频| 欧美黑人巨大精品一区二区| 欧美性xxxx在线播放| 欧美电影在线播放| 久久久999精品免费| 精品美女国产在线| 91夜夜未满十八勿入爽爽影院|