日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

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

编程问答

通过源码分析MyBatis的缓存

發(fā)布時間:2025/3/21 编程问答 33 豆豆
生活随笔 收集整理的這篇文章主要介紹了 通过源码分析MyBatis的缓存 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

前方高能! 本文內(nèi)容有點多,通過實際測試?yán)?#43;源碼分析的方式解剖MyBatis緩存的概念,對這方面有興趣的小伙伴請繼續(xù)看下去~

MyBatis緩存介紹

首先看一段wiki上關(guān)于MyBatis緩存的介紹:

MyBatis支持聲明式數(shù)據(jù)緩存(declarative data caching)。當(dāng)一條SQL語句被標(biāo)記為“可緩存”后,首次執(zhí)行它時從數(shù)據(jù)庫獲取的所有數(shù)據(jù)會被存儲在一段高速緩存中,今后執(zhí)行這條語句時就會從高速緩存中讀取結(jié)果,而不是再次命中數(shù)據(jù)庫。MyBatis提供了默認(rèn)下基于Java HashMap的緩存實現(xiàn),以及用于與OSCache、Ehcache、Hazelcast和Memcached連接的默認(rèn)連接器。MyBatis還提供API供其他緩存實現(xiàn)使用。

重點的那句話就是:MyBatis執(zhí)行SQL語句之后,這條語句就是被緩存,以后再執(zhí)行這條語句的時候,會直接從緩存中拿結(jié)果,而不是再次執(zhí)行SQL

這也就是大家常說的MyBatis一級緩存,一級緩存的作用域scope是SqlSession。

MyBatis同時還提供了一種全局作用域global scope的緩存,這也叫做二級緩存,也稱作全局緩存。

一級緩存

測試

同個session進(jìn)行兩次相同查詢:

@Test public void test() {SqlSession sqlSession = sqlSessionFactory.openSession();try {User user = (User)sqlSession.selectOne("org.format.mybatis.cache.UserMapper.getById", 1);log.debug(user);User user2 = (User)sqlSession.selectOne("org.format.mybatis.cache.UserMapper.getById", 1);log.debug(user2);} finally {sqlSession.close();} }

MyBatis只進(jìn)行1次數(shù)據(jù)庫查詢:

==> Preparing: select * from USERS WHERE ID = ? ==> Parameters: 1(Integer) <== Total: 1 User{id=1, name='format', age=23, birthday=Sun Oct 12 23:20:13 CST 2014} User{id=1, name='format', age=23, birthday=Sun Oct 12 23:20:13 CST 2014}

同個session進(jìn)行兩次不同的查詢:

@Test public void test() {SqlSession sqlSession = sqlSessionFactory.openSession();try {User user = (User)sqlSession.selectOne("org.format.mybatis.cache.UserMapper.getById", 1);log.debug(user);User user2 = (User)sqlSession.selectOne("org.format.mybatis.cache.UserMapper.getById", 2);log.debug(user2);} finally {sqlSession.close();} }

MyBatis進(jìn)行兩次數(shù)據(jù)庫查詢:

==> Preparing: select * from USERS WHERE ID = ? ==> Parameters: 1(Integer) <== Total: 1 User{id=1, name='format', age=23, birthday=Sun Oct 12 23:20:13 CST 2014} ==> Preparing: select * from USERS WHERE ID = ? ==> Parameters: 2(Integer) <== Total: 1 User{id=2, name='FFF', age=50, birthday=Sat Dec 06 17:12:01 CST 2014}

不同session,進(jìn)行相同查詢:

@Test public void test() {SqlSession sqlSession = sqlSessionFactory.openSession();SqlSession sqlSession2 = sqlSessionFactory.openSession();try {User user = (User)sqlSession.selectOne("org.format.mybatis.cache.UserMapper.getById", 1);log.debug(user);User user2 = (User)sqlSession2.selectOne("org.format.mybatis.cache.UserMapper.getById", 1);log.debug(user2);} finally {sqlSession.close();sqlSession2.close();} }

MyBatis進(jìn)行了兩次數(shù)據(jù)庫查詢:

==> Preparing: select * from USERS WHERE ID = ? ==> Parameters: 1(Integer) <== Total: 1 User{id=1, name='format', age=23, birthday=Sun Oct 12 23:20:13 CST 2014} ==> Preparing: select * from USERS WHERE ID = ? ==> Parameters: 1(Integer) <== Total: 1 User{id=1, name='format', age=23, birthday=Sun Oct 12 23:20:13 CST 2014}

同個session,查詢之后更新數(shù)據(jù),再次查詢相同的語句:

@Test public void test() {SqlSession sqlSession = sqlSessionFactory.openSession();try {User user = (User)sqlSession.selectOne("org.format.mybatis.cache.UserMapper.getById", 1);log.debug(user);user.setAge(100);sqlSession.update("org.format.mybatis.cache.UserMapper.update", user);User user2 = (User)sqlSession.selectOne("org.format.mybatis.cache.UserMapper.getById", 1);log.debug(user2);sqlSession.commit();} finally {sqlSession.close();} }

更新操作之后緩存會被清除:

==> Preparing: select * from USERS WHERE ID = ? ==> Parameters: 1(Integer) <== Total: 1 User{id=1, name='format', age=23, birthday=Sun Oct 12 23:20:13 CST 2014} ==> Preparing: update USERS SET NAME = ? , AGE = ? , BIRTHDAY = ? where ID = ? ==> Parameters: format(String), 23(Integer), 2014-10-12 23:20:13.0(Timestamp), 1(Integer) <== Updates: 1 ==> Preparing: select * from USERS WHERE ID = ? ==> Parameters: 1(Integer) <== Total: 1 User{id=1, name='format', age=23, birthday=Sun Oct 12 23:20:13 CST 2014}

很明顯,結(jié)果驗證了一級緩存的概念,在同個SqlSession中,查詢語句相同的sql會被緩存,但是一旦執(zhí)行新增或更新或刪除操作,緩存就會被清除

源碼分析

在分析MyBatis的一級緩存之前,我們先簡單看下MyBatis中幾個重要的類和接口:

org.apache.ibatis.session.Configuration類:MyBatis全局配置信息類

org.apache.ibatis.session.SqlSessionFactory接口:操作SqlSession的工廠接口,具體的實現(xiàn)類是DefaultSqlSessionFactory

org.apache.ibatis.session.SqlSession接口:執(zhí)行sql,管理事務(wù)的接口,具體的實現(xiàn)類是DefaultSqlSession

org.apache.ibatis.executor.Executor接口:sql執(zhí)行器,SqlSession執(zhí)行sql最終是通過該接口實現(xiàn)的,常用的實現(xiàn)類有SimpleExecutor和CachingExecutor,這些實現(xiàn)類都使用了裝飾者設(shè)計模式

一級緩存的作用域是SqlSession,那么我們就先看一下SqlSession的select過程:

這是DefaultSqlSession(SqlSession接口實現(xiàn)類,MyBatis默認(rèn)使用這個類)的selectList源碼(我們例子上使用的是selectOne方法,調(diào)用selectOne方法最終會執(zhí)行selectList方法):

public <E> List<E> selectList(String statement, Object parameter, RowBounds rowBounds) {try {MappedStatement ms = configuration.getMappedStatement(statement);List<E> result = executor.query(ms, wrapCollection(parameter), rowBounds, Executor.NO_RESULT_HANDLER);return result;} catch (Exception e) {throw ExceptionFactory.wrapException("Error querying database. Cause: " + e, e);} finally {ErrorContext.instance().reset();} }

我們看到SqlSession最終會調(diào)用Executor接口的方法。

接下來我們看下DefaultSqlSession中的executor接口屬性具體是哪個實現(xiàn)類。

DefaultSqlSession的構(gòu)造過程(DefaultSqlSessionFactory內(nèi)部):

private SqlSession openSessionFromDataSource(ExecutorType execType, TransactionIsolationLevel level, boolean autoCommit) {Transaction tx = null;try {final Environment environment = configuration.getEnvironment();final TransactionFactory transactionFactory = getTransactionFactoryFromEnvironment(environment);tx = transactionFactory.newTransaction(environment.getDataSource(), level, autoCommit);final Executor executor = configuration.newExecutor(tx, execType, autoCommit);return new DefaultSqlSession(configuration, executor);} catch (Exception e) {closeTransaction(tx); // may have fetched a connection so lets call close()throw ExceptionFactory.wrapException("Error opening session. Cause: " + e, e);} finally {ErrorContext.instance().reset();} }

我們看到DefaultSqlSessionFactory構(gòu)造DefaultSqlSession的時候,Executor接口的實現(xiàn)類是由Configuration構(gòu)造的:

public Executor newExecutor(Transaction transaction, ExecutorType executorType, boolean autoCommit) {executorType = executorType == null ? defaultExecutorType : executorType;executorType = executorType == null ? ExecutorType.SIMPLE : executorType;Executor executor;if (ExecutorType.BATCH == executorType) {executor = new BatchExecutor(this, transaction);} else if (ExecutorType.REUSE == executorType) {executor = new ReuseExecutor(this, transaction);} else {executor = new SimpleExecutor(this, transaction);}if (cacheEnabled) {executor = new CachingExecutor(executor, autoCommit);}executor = (Executor) interceptorChain.pluginAll(executor);return executor; }

Executor根據(jù)ExecutorType的不同而創(chuàng)建,最常用的是SimpleExecutor,本文的例子也是創(chuàng)建這個實現(xiàn)類。 最后我們發(fā)現(xiàn)如果cacheEnabled這個屬性為true的話,那么executor會被包一層裝飾器,這個裝飾器是CachingExecutor。其中cacheEnabled這個屬性是mybatis總配置文件中settings節(jié)點中cacheEnabled子節(jié)點的值,默認(rèn)就是true,也就是說我們在mybatis總配置文件中不配cacheEnabled的話,它也是默認(rèn)為打開的。

現(xiàn)在,問題就剩下一個了,CachingExecutor執(zhí)行sql的時候到底做了什么?

帶著這個問題,我們繼續(xù)走下去(CachingExecutor的query方法):

public <E> List<E> query(MappedStatement ms, Object parameterObject, RowBounds rowBounds, ResultHandler resultHandler, CacheKey key, BoundSql boundSql) throws SQLException {Cache cache = ms.getCache();if (cache != null) {flushCacheIfRequired(ms);if (ms.isUseCache() && resultHandler == null) { ensureNoOutParams(ms, parameterObject, boundSql);if (!dirty) {cache.getReadWriteLock().readLock().lock();try {@SuppressWarnings("unchecked")List<E> cachedList = (List<E>) cache.getObject(key);if (cachedList != null) return cachedList;} finally {cache.getReadWriteLock().readLock().unlock();}}List<E> list = delegate.<E> query(ms, parameterObject, rowBounds, resultHandler, key, boundSql);tcm.putObject(cache, key, list); // issue #578. Query must be not synchronized to prevent deadlocksreturn list;}}return delegate.<E>query(ms, parameterObject, rowBounds, resultHandler, key, boundSql); }

其中Cache cache = ms.getCache();這句代碼中,這個cache實際上就是個二級緩存,由于我們沒有開啟二級緩存(二級緩存的內(nèi)容下面會分析),因此這里執(zhí)行了最后一句話。這里的delegate也就是SimpleExecutor,SimpleExecutor沒有Override父類的query方法,因此最終執(zhí)行了SimpleExecutor的父類BaseExecutor的query方法。

所以一級緩存最重要的代碼就是BaseExecutor的query方法!

BaseExecutor的屬性localCache是個PerpetualCache類型的實例,PerpetualCache類是實現(xiàn)了MyBatis的Cache緩存接口的實現(xiàn)類之一,內(nèi)部有個Map<Object, Object>類型的屬性用來存儲緩存數(shù)據(jù)。 這個localCache的類型在BaseExecutor內(nèi)部是寫死的。 這個localCache就是一級緩存!

接下來我們看下為何執(zhí)行新增或更新或刪除操作,一級緩存就會被清除這個問題。

首先MyBatis處理新增或刪除的時候,最終都是調(diào)用update方法,也就是說新增或者刪除操作在MyBatis眼里都是一個更新操作。

我們看下DefaultSqlSession的update方法:

public int update(String statement, Object parameter) {try {dirty = true;MappedStatement ms = configuration.getMappedStatement(statement);return executor.update(ms, wrapCollection(parameter));} catch (Exception e) {throw ExceptionFactory.wrapException("Error updating database. Cause: " + e, e);} finally {ErrorContext.instance().reset();} }

很明顯,這里調(diào)用了CachingExecutor的update方法:

public int update(MappedStatement ms, Object parameterObject) throws SQLException {flushCacheIfRequired(ms);return delegate.update(ms, parameterObject); }

這里的flushCacheIfRequired方法清除的是二級緩存,我們之后會分析。 CachingExecutor委托給了(之前已經(jīng)分析過)SimpleExecutor的update方法,SimpleExecutor沒有Override父類BaseExecutor的update方法,因此我們看BaseExecutor的update方法:

public int update(MappedStatement ms, Object parameter) throws SQLException {ErrorContext.instance().resource(ms.getResource()).activity("executing an update").object(ms.getId());if (closed) throw new ExecutorException("Executor was closed.");clearLocalCache();return doUpdate(ms, parameter); }

我們看到了關(guān)鍵的一句代碼: clearLocalCache(); 進(jìn)去看看:

public void clearLocalCache() {if (!closed) {localCache.clear();localOutputParameterCache.clear();} }

沒錯,就是這條,sqlsession沒有關(guān)閉的話,進(jìn)行新增、刪除、修改操作的話就是清除一級緩存,也就是SqlSession的緩存。

二級緩存

二級緩存的作用域是全局,換句話說,二級緩存已經(jīng)脫離SqlSession的控制了。

在測試二級緩存之前,我先把結(jié)論說一下:

二級緩存的作用域是全局的,二級緩存在SqlSession關(guān)閉或提交之后才會生效。

在分析MyBatis的二級緩存之前,我們先簡單看下MyBatis中一個關(guān)于二級緩存的類(其他相關(guān)的類和接口之前已經(jīng)分析過):

org.apache.ibatis.mapping.MappedStatement:

MappedStatement類在Mybatis框架中用于表示XML文件中一個sql語句節(jié)點,即一個<select />、<update />或者<insert />標(biāo)簽。Mybatis框架在初始化階段會對XML配置文件進(jìn)行讀取,將其中的sql語句節(jié)點對象化為一個個MappedStatement對象。

配置

二級緩存跟一級緩存不同,一級緩存不需要配置任何東西,且默認(rèn)打開。 二級緩存就需要配置一些東西。

本文就說下最簡單的配置,在mapper文件上加上這句配置即可:

<cache/>

其實二級緩存跟3個配置有關(guān):

  • mybatis全局配置文件中的setting中的cacheEnabled需要為true(默認(rèn)為true,不設(shè)置也行)
  • mapper配置文件中需要加入<cache>節(jié)點
  • mapper配置文件中的select節(jié)點需要加上屬性useCache需要為true(默認(rèn)為true,不設(shè)置也行)
  • 測試

    不同SqlSession,查詢相同語句,第一次查詢之后commit SqlSession:

    @Test public void testCache2() {SqlSession sqlSession = sqlSessionFactory.openSession();SqlSession sqlSession2 = sqlSessionFactory.openSession();try {String sql = "org.format.mybatis.cache.UserMapper.getById";User user = (User)sqlSession.selectOne(sql, 1);log.debug(user);// 注意,這里一定要提交。 不提交還是會查詢兩次數(shù)據(jù)庫sqlSession.commit();User user2 = (User)sqlSession2.selectOne(sql, 1);log.debug(user2);} finally {sqlSession.close();sqlSession2.close();} }

    MyBatis僅進(jìn)行了一次數(shù)據(jù)庫查詢:

    ==> Preparing: select * from USERS WHERE ID = ? ==> Parameters: 1(Integer) <== Total: 1 User{id=1, name='format', age=23, birthday=Sun Oct 12 23:20:13 CST 2014} User{id=1, name='format', age=23, birthday=Sun Oct 12 23:20:13 CST 2014}

    不同SqlSession,查詢相同語句,第一次查詢之后close SqlSession:

    @Test public void testCache2() {SqlSession sqlSession = sqlSessionFactory.openSession();SqlSession sqlSession2 = sqlSessionFactory.openSession();try {String sql = "org.format.mybatis.cache.UserMapper.getById";User user = (User)sqlSession.selectOne(sql, 1);log.debug(user);sqlSession.close();User user2 = (User)sqlSession2.selectOne(sql, 1);log.debug(user2);} finally {sqlSession2.close();} }

    MyBatis僅進(jìn)行了一次數(shù)據(jù)庫查詢:

    ==> Preparing: select * from USERS WHERE ID = ? ==> Parameters: 1(Integer) <== Total: 1 User{id=1, name='format', age=23, birthday=Sun Oct 12 23:20:13 CST 2014} User{id=1, name='format', age=23, birthday=Sun Oct 12 23:20:13 CST 2014}

    不同SqlSesson,查詢相同語句。 第一次查詢之后SqlSession不提交:

    @Test public void testCache2() {SqlSession sqlSession = sqlSessionFactory.openSession();SqlSession sqlSession2 = sqlSessionFactory.openSession();try {String sql = "org.format.mybatis.cache.UserMapper.getById";User user = (User)sqlSession.selectOne(sql, 1);log.debug(user);User user2 = (User)sqlSession2.selectOne(sql, 1);log.debug(user2);} finally {sqlSession.close();sqlSession2.close();} }

    MyBatis執(zhí)行了兩次數(shù)據(jù)庫查詢:

    ==> Preparing: select * from USERS WHERE ID = ? ==> Parameters: 1(Integer) <== Total: 1 User{id=1, name='format', age=23, birthday=Sun Oct 12 23:20:13 CST 2014} ==> Preparing: select * from USERS WHERE ID = ? ==> Parameters: 1(Integer) <== Total: 1 User{id=1, name='format', age=23, birthday=Sun Oct 12 23:20:13 CST 2014}

    源碼分析

    我們從在mapper文件中加入的<cache/>中開始分析源碼,關(guān)于MyBatis的SQL解析請參考另外一篇博客Mybatis解析動態(tài)sql原理分析。接下來我們看下這個cache的解析:

    XMLMappedBuilder(解析每個mapper配置文件的解析類,每一個mapper配置都會實例化一個XMLMapperBuilder類)的解析方法:

    private void configurationElement(XNode context) {try {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"));parameterMapElement(context.evalNodes("/mapper/parameterMap"));resultMapElements(context.evalNodes("/mapper/resultMap"));sqlElement(context.evalNodes("/mapper/sql"));buildStatementFromContext(context.evalNodes("select|insert|update|delete"));} catch (Exception e) {throw new BuilderException("Error parsing Mapper XML. Cause: " + e, e);} }

    我們看到了解析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);Properties props = context.getChildrenAsProperties();builderAssistant.useNewCache(typeClass, evictionClass, flushInterval, size, readWrite, props);} }

    解析完cache標(biāo)簽之后會使用builderAssistant的userNewCache方法,這里的builderAssistant是一個MapperBuilderAssistant類型的幫助類,每個XMLMappedBuilder構(gòu)造的時候都會實例化這個屬性,MapperBuilderAssistant類內(nèi)部有個Cache類型的currentCache屬性,這個屬性也就是mapper配置文件中cache節(jié)點所代表的值:

    public Cache useNewCache(Class<? extends Cache> typeClass,Class<? extends Cache> evictionClass,Long flushInterval,Integer size,boolean readWrite,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).properties(props).build();configuration.addCache(cache);currentCache = cache;return cache; }

    ok,現(xiàn)在mapper配置文件中的cache節(jié)點被解析到了XMLMapperBuilder實例中的builderAssistant屬性中的currentCache值里。

    接下來XMLMapperBuilder會解析select節(jié)點,解析select節(jié)點的時候使用XMLStatementBuilder進(jìn)行解析(也包括其他insert,update,delete節(jié)點):

    public void parseStatementNode() {String id = context.getStringAttribute("id");String databaseId = context.getStringAttribute("databaseId");if (!databaseIdMatchesCurrent(id, databaseId, this.requiredDatabaseId)) return;Integer fetchSize = context.getIntAttribute("fetchSize");Integer timeout = context.getIntAttribute("timeout");String parameterMap = context.getStringAttribute("parameterMap");String parameterType = context.getStringAttribute("parameterType");Class<?> parameterTypeClass = resolveClass(parameterType);String resultMap = context.getStringAttribute("resultMap");String resultType = context.getStringAttribute("resultType");String lang = context.getStringAttribute("lang");LanguageDriver langDriver = getLanguageDriver(lang);Class<?> resultTypeClass = resolveClass(resultType);String resultSetType = context.getStringAttribute("resultSetType");StatementType statementType = StatementType.valueOf(context.getStringAttribute("statementType", StatementType.PREPARED.toString()));ResultSetType resultSetTypeEnum = resolveResultSetType(resultSetType);String nodeName = context.getNode().getNodeName();SqlCommandType sqlCommandType = SqlCommandType.valueOf(nodeName.toUpperCase(Locale.ENGLISH));boolean isSelect = sqlCommandType == SqlCommandType.SELECT;boolean flushCache = context.getBooleanAttribute("flushCache", !isSelect);boolean useCache = context.getBooleanAttribute("useCache", isSelect);boolean resultOrdered = context.getBooleanAttribute("resultOrdered", false);// Include Fragments before parsingXMLIncludeTransformer includeParser = new XMLIncludeTransformer(configuration, builderAssistant);includeParser.applyIncludes(context.getNode());// Parse selectKey after includes and remove them.processSelectKeyNodes(id, parameterTypeClass, langDriver);// Parse the SQL (pre: <selectKey> and <include> were parsed and removed)SqlSource sqlSource = langDriver.createSqlSource(configuration, context, parameterTypeClass);String resultSets = context.getStringAttribute("resultSets");String keyProperty = context.getStringAttribute("keyProperty");String keyColumn = context.getStringAttribute("keyColumn");KeyGenerator keyGenerator;String keyStatementId = id + SelectKeyGenerator.SELECT_KEY_SUFFIX;keyStatementId = builderAssistant.applyCurrentNamespace(keyStatementId, true);if (configuration.hasKeyGenerator(keyStatementId)) {keyGenerator = configuration.getKeyGenerator(keyStatementId);} else {keyGenerator = context.getBooleanAttribute("useGeneratedKeys",configuration.isUseGeneratedKeys() && SqlCommandType.INSERT.equals(sqlCommandType))? new Jdbc3KeyGenerator() : new NoKeyGenerator();}builderAssistant.addMappedStatement(id, sqlSource, statementType, sqlCommandType,fetchSize, timeout, parameterMap, parameterTypeClass, resultMap, resultTypeClass,resultSetTypeEnum, flushCache, useCache, resultOrdered, keyGenerator, keyProperty, keyColumn, databaseId, langDriver, resultSets); }

    這段代碼前面都是解析一些標(biāo)簽的屬性,我們看到了最后一行使用builderAssistant添加MappedStatement,其中builderAssistant屬性是構(gòu)造XMLStatementBuilder的時候通過XMLMappedBuilder傳入的,我們繼續(xù)看builderAssistant的addMappedStatement方法:

    進(jìn)入setStatementCache:

    private void setStatementCache(boolean isSelect,boolean flushCache,boolean useCache,Cache cache,MappedStatement.Builder statementBuilder) {flushCache = valueOrDefault(flushCache, !isSelect);useCache = valueOrDefault(useCache, isSelect);statementBuilder.flushCacheRequired(flushCache);statementBuilder.useCache(useCache);statementBuilder.cache(cache); }

    最終mapper配置文件中的<cache/>被設(shè)置到了XMLMapperBuilder的builderAssistant屬性中,XMLMapperBuilder中使用XMLStatementBuilder遍歷CRUD節(jié)點,遍歷CRUD節(jié)點的時候?qū)⑦@個cache節(jié)點設(shè)置到這些CRUD節(jié)點中,這個cache就是所謂的二級緩存!

    接下來我們回過頭來看查詢的源碼,CachingExecutor的query方法:

    進(jìn)入TransactionalCacheManager的putObject方法:

    public void putObject(Cache cache, CacheKey key, Object value) {getTransactionalCache(cache).putObject(key, value); }private TransactionalCache getTransactionalCache(Cache cache) {TransactionalCache txCache = transactionalCaches.get(cache);if (txCache == null) {txCache = new TransactionalCache(cache);transactionalCaches.put(cache, txCache);}return txCache; }

    TransactionalCache的putObject方法:

    public void putObject(Object key, Object object) {entriesToRemoveOnCommit.remove(key);entriesToAddOnCommit.put(key, new AddEntry(delegate, key, object)); }

    我們看到,數(shù)據(jù)被加入到了entriesToAddOnCommit中,這個entriesToAddOnCommit是什么東西呢,它是TransactionalCache的一個Map屬性:

    private Map<Object, AddEntry> entriesToAddOnCommit;

    AddEntry是TransactionalCache內(nèi)部的一個類:

    private static class AddEntry {private Cache cache;private Object key;private Object value;public AddEntry(Cache cache, Object key, Object value) {this.cache = cache;this.key = key;this.value = value;}public void commit() {cache.putObject(key, value);} }

    好了,現(xiàn)在我們發(fā)現(xiàn)使用二級緩存之后:查詢數(shù)據(jù)的話,先從二級緩存中拿數(shù)據(jù),如果沒有的話,去一級緩存中拿,一級緩存也沒有的話再查詢數(shù)據(jù)庫。有了數(shù)據(jù)之后在丟到TransactionalCache這個對象的entriesToAddOnCommit屬性中。

    接下來我們來驗證為什么SqlSession commit或close之后,二級緩存才會生效這個問題。

    DefaultSqlSession的commit方法:

    public void commit(boolean force) {try {executor.commit(isCommitOrRollbackRequired(force));dirty = false;} catch (Exception e) {throw ExceptionFactory.wrapException("Error committing transaction. Cause: " + e, e);} finally {ErrorContext.instance().reset();} }

    CachingExecutor的commit方法:

    public void commit(boolean required) throws SQLException {delegate.commit(required);tcm.commit();dirty = false; }

    tcm.commit即 TransactionalCacheManager的commit方法:

    public void commit() {for (TransactionalCache txCache : transactionalCaches.values()) {txCache.commit();} }

    TransactionalCache的commit方法:

    public void commit() {delegate.getReadWriteLock().writeLock().lock();try {if (clearOnCommit) {delegate.clear();} else {for (RemoveEntry entry : entriesToRemoveOnCommit.values()) {entry.commit();}}for (AddEntry entry : entriesToAddOnCommit.values()) {entry.commit();}reset();} finally {delegate.getReadWriteLock().writeLock().unlock();} }

    發(fā)現(xiàn)調(diào)用了AddEntry的commit方法:

    public void commit() {cache.putObject(key, value); }

    發(fā)現(xiàn)了! AddEntry的commit方法會把數(shù)據(jù)丟到cache中,也就是丟到二級緩存中!

    關(guān)于為何調(diào)用close方法后,二級緩存才會生效,因為close方法內(nèi)部會調(diào)用commit方法。本文就不具體說了。 讀者有興趣的話看一看源碼就知道為什么了。

    其他

    Cache接口簡介

    org.apache.ibatis.cache.Cache是MyBatis的緩存接口,想要實現(xiàn)自定義的緩存需要實現(xiàn)這個接口。

    MyBatis中關(guān)于Cache接口的實現(xiàn)類也使用了裝飾者設(shè)計模式。

    我們看下它的一些實現(xiàn)類:

    簡單說明:

    LRU – 最近最少使用的:移除最長時間不被使用的對象。

    FIFO – 先進(jìn)先出:按對象進(jìn)入緩存的順序來移除它們。

    SOFT – 軟引用:移除基于垃圾回收器狀態(tài)和軟引用規(guī)則的對象。

    WEAK – 弱引用:更積極地移除基于垃圾收集器狀態(tài)和弱引用規(guī)則的對象。

    <cacheeviction="FIFO"flushInterval="60000"size="512"readOnly="true"/>

    可以通過cache節(jié)點的eviction屬性設(shè)置,也可以設(shè)置其他的屬性。

    cache-ref節(jié)點

    mapper配置文件中還可以加入cache-ref節(jié)點,它有個屬性namespace。

    如果每個mapper文件都是用cache-ref,且namespace都一樣,那么就代表著真正意義上的全局緩存。

    如果只用了cache節(jié)點,那僅代表這個這個mapper內(nèi)部的查詢被緩存了,其他mapper文件的不起作用,這并不是所謂的全局緩存。

    總結(jié)

    總體來說,MyBatis的源碼看起來還是比較輕松的,本文從實踐和源碼方面深入分析了MyBatis的緩存原理,希望對讀者有幫助。

    from:?https://www.cnblogs.com/fangjian0423/p/mybatis-cache.html

    總結(jié)

    以上是生活随笔為你收集整理的通过源码分析MyBatis的缓存的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

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

    中文字幕在线观看完整版 | 麻花豆传媒一二三产区 | 成年人免费在线 | 中文日韩在线 | 久久99久久99精品中文字幕 | 日韩欧美视频在线播放 | 久久草草影视免费网 | a色网站| 亚洲精品18p | 国产性天天综合网 | 久久免费国产视频 | 国产福利一区二区三区在线观看 | 超碰国产97 | 久久99国产综合精品免费 | 久久不卡电影 | 韩日视频在线 | 国产精品久久久久久久久大全 | 免费试看一区 | 亚洲欧美日韩精品久久久 | 日韩高清免费无专码区 | 日本特黄一级片 | 日韩专区视频 | 91精品免费| 国产亚洲精品中文字幕 | 久草9视频 | 亚洲视频精品在线 | 美女视频a美女大全免费下载蜜臀 | 国产精品久久久久久久久免费 | 国产午夜精品久久久久久久久久 | 欧美另类sm图片 | 亚洲精品a区 | 视频在线观看99 | 精品国产资源 | 18性欧美xxxⅹ性满足 | 人人射人人射 | 亚洲综合色视频在线观看 | 国产亚洲精品久久久久秋 | 亚洲国产日韩欧美在线 | 黄色软件在线观看 | 女人18片毛片90分钟 | 欧美一区二区三区在线看 | 99热精品在线观看 | 欧美精品久久久久久久久老牛影院 | 亚洲精品在线免费观看视频 | 叶爱av在线| 国产精品区二区三区日本 | 808电影| 天天操天天射天天操 | 久久精品毛片基地 | 久久久九色精品国产一区二区三区 | 美女视频是黄的免费观看 | 天天操夜夜操 | 97精品国产97久久久久久 | 日本性生活免费看 | 久久精品永久免费 | 色婷婷激情 | 欧美成人h版| 国产一区二区三精品久久久无广告 | 日韩四虎 | 中文字幕在线乱 | 丝袜av一区 | 国产成人精品一区一区一区 | 亚洲一区日韩在线 | 久久久久久99精品 | 免费色视频网址 | 免费十分钟| 亚洲免费视频观看 | 午夜久久影院 | 亚洲精品国产综合久久 | 人人干网站 | 久久av中文字幕片 | 国产中文字幕91 | 精品国精品自拍自在线 | 24小时日本在线www免费的 | 中文字幕免费一区二区 | 九九久久久 | 欧美a在线免费观看 | 免费视频 三区 | 国产成人在线综合 | 日日夜夜天天操 | 六月色婷婷 | 国产精品美女999 | 欧美一区二区伦理片 | 在线观看福利网站 | 亚洲爱爱视频 | 亚洲成人高清在线 | 欧美日韩中文在线观看 | 99国产成+人+综合+亚洲 欧美 | 国产在线视频导航 | 一级黄色在线视频 | 国产精品123 | 久久久久久久久久久高潮一区二区 | 久久免费看视频 | 天天操天天拍 | 日韩一级黄色片 | 日本3级在线观看 | 一级成人免费 | 久久久久久久久久久综合 | 91av久久| 91精品在线麻豆 | 久久久久北条麻妃免费看 | 日韩视频一区二区三区 | 日韩www在线 | 国产福利一区二区三区在线观看 | www久久九 | 国产打女人屁股调教97 | 日韩一二区在线 | 日精品在线观看 | 国产精品影音先锋 | 黄色一及电影 | 日韩在线观看免费 | 国产精品久久三 | 免费a级观看 | 免费日韩 精品中文字幕视频在线 | 在线观看中文字幕亚洲 | 午夜性福利| 国产剧情久久 | 免费精品在线 | 国产三级国产精品国产专区50 | 国产日本亚洲高清 | 激情欧美一区二区三区免费看 | 亚洲黄色在线播放 | 久久中文字幕导航 | 自拍超碰在线 | 日韩在线色视频 | 日韩欧美国产成人 | 成人在线观看网址 | 五月天久久久久 | 久久成人亚洲欧美电影 | 久久一区二区三区超碰国产精品 | 黄色av成人在线观看 | 久久久久亚洲最大xxxx | 国产精品情侣视频 | 天天天天天天天操 | 国产精品成人av电影 | 亚洲精品久久视频 | 91香蕉视频色版 | 亚欧日韩成人h片 | 国产一区二区精 | 天天干,夜夜爽 | 久久影视一区 | 91视频免费国产 | 手机在线黄色网址 | 麻豆免费视频 | 夜夜操天天摸 | 久久精品亚洲综合专区 | 狠狠色噜噜狠狠 | 精品99在线观看 | 91亚色视频 | 色噜噜日韩精品一区二区三区视频 | 国产九九精品视频 | 亚洲三级在线免费观看 | 国产精品视频地址 | 久草电影免费在线观看 | 久久综合久久综合久久 | 国产精品爽爽爽 | 久久久www成人免费精品张筱雨 | 麻豆视频在线免费观看 | 欧美日韩国产精品爽爽 | 在线观看午夜av | av看片网 | 黄色小说免费观看 | 在线观看成人小视频 | 中文av在线免费观看 | 欧美激情第八页 | 国产精品视频观看 | 最近中文字幕在线中文高清版 | 黄色大片日本 | 三级av在线免费观看 | 精品亚洲va在线va天堂资源站 | 最新国产视频 | 国产精品久久一区二区三区不卡 | 国产很黄很色的视频 | 免费看黄电影 | 亚洲精品黄网站 | 91麻豆精品国产91久久久无限制版 | 日韩欧美视频 | 亚洲精品永久免费视频 | 狠狠网亚洲精品 | 美女视频黄免费的久久 | 国产一区二区在线视频观看 | 五月香婷| av一级片在线观看 | 超碰人人超 | 日韩在线观看视频中文字幕 | 亚洲欧洲中文日韩久久av乱码 | 午夜美女wwww | 91桃色视频| av一级片在线观看 | 国产精品久一 | www亚洲国产 | 在线观看视频色 | 男女免费视频观看 | 久草在线视频在线观看 | 亚洲精品久久久蜜臀下载官网 | 亚洲永久精品在线 | 免费观看不卡av | 99久热精品 | 激情五月五月婷婷 | 欧美精品一区在线发布 | 五月天婷婷丁香花 | 精品一区二区av | 国产超碰在线 | 99精品在线观看视频 | 久久精品国产精品亚洲 | 国产精品成人一区二区三区吃奶 | 一级一片免费视频 | 成人av免费 | 99re8这里有精品热视频免费 | 97在线观看视频国产 | 久久人人看 | 91九色国产蝌蚪 | av网站播放 | 韩国在线一区 | 69夜色精品国产69乱 | 久久久免费国产 | 国产婷婷在线观看 | 欧美一区影院 | 一级黄色大片 | 91av视频在线免费观看 | 97超级碰碰碰碰久久久久 | 久99久精品视频免费观看 | 日韩国产精品一区 | 国产一区二区三精品久久久无广告 | 久久久久免费看 | 六月丁香激情综合色啪小说 | 欧美在线91 | 成人不用播放器 | 99久久日韩精品免费热麻豆美女 | 国产在线久草 | av在线免费网 | 国产一区在线看 | 一级一级一片免费 | 91成熟丰满女人少妇 | 91视频传媒 | 久久精品成人 | 亚洲欧洲国产视频 | 午夜久久久久久久久 | 久久久精品亚洲 | 2019天天干天天色 | 狠狠操在线 | 久久www免费视频 | 成人黄色电影在线 | 亚洲最新av网址 | 天天操天天综合网 | 手机在线免费av | 久久久久久综合网天天 | 免费视频黄 | 久久精品草 | 69亚洲精品| 黄色资源在线 | 久久久久国产精品午夜一区 | 黄av资源 | 一本一本久久a久久精品综合小说 | 亚洲精品美女久久久 | 久久精品在线 | 91pony九色丨交换 | 国产在线视频资源 | 99免费国产 | 九热在线 | 国语自产偷拍精品视频偷 | 国产69久久精品成人看 | 日韩av电影手机在线观看 | 亚洲国产午夜精品 | 91亚洲精品乱码久久久久久蜜桃 | 国产精品资源在线 | 玖玖视频| 国产黄大片 | 色午夜 | 夜夜澡人模人人添人人看 | 天天色婷婷 | 超碰97国产在线 | 久久久久久久久久毛片 | 久久www免费人成看片高清 | 韩国精品在线观看 | 涩涩网站在线观看 | 九九热国产 | av在线不卡观看 | 精品在线视频观看 | 国产免费久久久久 | 日韩三级av | 亚洲色图激情文学 | 亚洲影院天堂 | 黄色国产精品 | 国产黄大片在线观看 | 人人玩人人添人人 | 国产啊v在线 | 在线观看久久 | 国产在线国产 | 97电影网手机版 | 999久久久国产精品 高清av免费观看 | 久久激情视频 | 国产精品视频全国免费观看 | 午夜久久网 | 懂色av一区二区在线播放 | 国产麻豆视频免费观看 | www.夜夜骑.com | 毛片3 | 丁香五婷 | 视频在线日韩 | 久操操| 激情综合五月婷婷 | 丁香综合五月 | 国产成人精品不卡 | 最近中文字幕完整视频高清1 | 久久97久久| 在线之家官网 | 2019中文最近的2019中文在线 | 天天综合网入口 | 亚洲精品乱码久久久久久蜜桃动漫 | 日韩精品无 | 国产成人精品999在线观看 | 久久久久久久久亚洲精品 | 国内精品小视频 | 91九色成人蝌蚪首页 | 免费视频99 | 91看片看淫黄大片 | 久久免费精品国产 | 中文字幕视频播放 | 精品麻豆入口免费 | 美女免费视频观看网站 | 91理论电影 | 国产精品成人一区二区三区 | 999久久国产精品免费观看网站 | 久久国产午夜精品理论片最新版本 | 丁香综合av | 午夜资源站 | 国产亚洲精品久久久久久无几年桃 | 精品亚洲视频在线观看 | 国模精品一区二区三区 | 在线观看中文字幕av | 久久久久欧美精品 | 日韩久久一区 | 久久成人午夜视频 | 岛国精品一区二区 | 国产午夜精品久久久久久久久久 | 国产盗摄精品一区二区 | 国产高清视频在线观看 | 香蕉97视频观看在线观看 | 一级免费看| 国产小视频你懂的在线 | 中文字幕 91 | 日韩在线观看第一页 | 91精品在线免费观看视频 | 午夜丁香网 | 国产精品久久电影网 | 2020天天干夜夜爽 | 九九久久在线看 | 成人资源在线播放 | 天天操天天弄 | 91精品国产91热久久久做人人 | 色综合天天在线 | 国产爽视频 | 久久免费精彩视频 | 免费在线电影网址大全 | 精品一区91 | 嫩草91影院| 香蕉久久久久久av成人 | 国产成人精品av在线观 | 亚洲综合一区二区精品导航 | 国产成人精品一区二区三区网站观看 | 337p日本大胆噜噜噜噜 | 免费在线激情电影 | 国产又粗又猛又色又黄网站 | 日韩成人邪恶影片 | 国产理论片在线观看 | 玖玖视频免费在线 | 狠狠婷婷| 天天草天天干天天射 | 97国产精品 | 婷婷精品视频 | 久久er99热精品一区二区三区 | 日韩影视在线观看 | 免费观看一级一片 | 激情久久伊人 | 在线看黄色的网站 | 欧美日韩性视频在线 | 国产日韩欧美在线看 | 亚洲成人精品影院 | 五月天激情综合 | 中文字幕资源在线 | 日韩sese| 国产97av| 天天曰天天 | 激情综合一区 | 国产黄色片一级 | 婷婷新五月 | 国产亚洲情侣一区二区无 | 在线日韩av | 亚洲一区尤物 | h视频日本 | 国产精品一区二区电影 | 国产精品久久久久久婷婷天堂 | 亚洲最新av网址 | 九九热在线免费观看 | 日本精品久久久一区二区三区 | 天天爱天天操 | 日日躁夜夜躁xxxxaaaa | 欧美亚洲国产精品久久高清浪潮 | 97色在线视频 | 国产亚洲精品久久久久动 | 1024久久 | 亚洲91中文字幕无线码三区 | 亚洲视频专区在线 | 国产高清久久久 | 91精品视频导航 | 精品 一区 在线 | 国产成a人亚洲精v品在线观看 | 欧美精品天堂 | www.伊人网 | 亚洲第一av在线 | 欧美日韩免费观看一区二区三区 | 懂色av懂色av粉嫩av分享吧 | 黄色片免费看 | 日韩女同一区二区三区在线观看 | 91精品在线免费观看 | 国产小视频免费在线观看 | 日韩久久精品 | 久久精品国亚洲 | 久久爱综合 | 久久久久免费精品 | 日韩精品一区二区三区高清免费 | 精品国产乱码久久久久久1区二区 | 亚洲精品裸体 | 一区二区三区韩国免费中文网站 | 国产在线国偷精品产拍免费yy | 97超视频 | 天天干人人 | 日韩极品视频在线观看 | 99麻豆久久久国产精品免费 | 美女视频黄免费网站 | 中文字幕国内精品 | 亚洲成人999 | 五月婷婷在线综合 | 国产黑丝一区二区三区 | 日韩免费高清 | 国产免费久久精品 | 极品美女被弄高潮视频网站 | 免费午夜视频在线观看 | 91大神精品视频在线观看 | 亚洲黄色免费在线 | 久久私人影院 | 色视频在线看 | 国产伦精品一区二区三区免费 | 日本黄色免费播放 | 日韩av片无码一区二区不卡电影 | 日日夜夜天天久久 | 国产成人精品午夜在线播放 | 91在线入口 | 五月婷婷综合久久 | 国产精品自产拍在线观看网站 | 欧美日韩久久不卡 | 国产麻豆视频 | 97在线超碰 | 欧美一二在线 | 久草久草久草久草 | 91免费试看| 久久久久 免费视频 | 美女福利视频一区二区 | 亚洲天堂自拍视频 | 日韩久久一区 | 黄色软件在线观看免费 | 日韩高清不卡一区二区三区 | 成人小电影在线看 | 日韩av资源在线观看 | 精品一区电影 | 久草网在线观看 | 超碰日韩 | 欧美日韩在线免费观看视频 | 福利电影一区二区 | 久久精品人人做人人综合老师 | 中文视频在线播放 | 成人va天堂 | 一级性视频 | 亚洲美女免费精品视频在线观看 | 日韩影视精品 | 欧美a视频 | 九九九视频精品 | 久久精品视频在线免费观看 | 久久久久国产成人免费精品免费 | 国产亚洲视频中文字幕视频 | 黄免费网站 | 欧美激情精品久久久久久 | 精品视频免费观看 | 国产黄 | 久久免费视频在线 | 白丝av在线| 成人午夜精品福利免费 | 精品久久美女 | www久久99| 日韩精品久久久久久久电影竹菊 | 伊人五月天综合 | 欧美综合久久 | 狠狠久久婷婷 | 精品免费久久久久 | 欧美日韩高清一区二区 国产亚洲免费看 | 国内精品视频一区二区三区八戒 | 91探花在线| 色97在线| 日韩精品一区二区三区第95 | 日日爽天天爽 | 在线日韩视频 | 久久资源在线 | 日韩欧美一区二区三区免费观看 | 91在线免费公开视频 | 久久经典国产 | 天堂网中文在线 | 三级视频片 | 国产一区欧美一区 | 亚洲国产日韩欧美在线 | 天天干天天天 | 欧美91av| 99这里只有久久精品视频 | 免费色av| 亚洲欧美偷拍另类 | 免费在线一区二区三区 | 成人动漫精品一区二区 | 97电影在线看视频 | 五月天中文字幕mv在线 | 久久成人麻豆午夜电影 | 国产高清永久免费 | 亚洲男男gaygayxxxgv | 日韩视频一区二区三区在线播放免费观看 | 五月天久久狠狠 | 特级西西444www大胆高清无视频 | 狠狠干天天 | 97精品欧美91久久久久久 | 亚洲乱码久久 | 日韩一区二区三区高清免费看看 | 精品96久久久久久中文字幕无 | 精品一区电影 | 在线91视频 | 亚洲 中文 欧美 日韩vr 在线 | 久久精品久久久久久久 | 日韩久久精品一区二区三区下载 | 久久久久久毛片 | 日韩视频免费播放 | 久久天| 国模一二三区 | 色吊丝在线永久观看最新版本 | 九九九热| 91在线播放视频 | 久久在现 | 日本中文字幕在线视频 | 久久精品中文 | 伊人婷婷色| 亚洲一区二区三区精品在线观看 | 国产精品人人做人人爽人人添 | 日本精品久久 | 免费日韩在线 | 激情综合亚洲精品 | 色综合色综合久久综合频道88 | 久久精品99精品国产香蕉 | www.亚洲精品 | 97小视频| a黄色片在线观看 | 97成人免费 | 国产精品21区 | 三级性生活视频 | 色综合婷婷久久 | 天天摸天天舔天天操 | 欧美激情亚洲综合 | 欧美乱码精品一区 | 国产精品大片在线观看 | 中文字幕日韩一区二区三区不卡 | 国产精品久久久久久久久婷婷 | 在线精品视频免费播放 | 午夜精品久久久久久久99无限制 | 91自拍91| 91精品伦理 | 99久久精品免费看国产四区 | 精品一区在线 | 日日干日日色 | 国内精品亚洲 | 韩国一区二区三区在线观看 | 免费99精品国产自在在线 | 特级毛片网站 | 精品久久久久一区二区国产 | 在线观看中文字幕2021 | 欧美日韩视频免费 | 国产成人av综合色 | 欧美人人 | 91麻豆免费版 | 成人理论电影 | 九九九视频精品 | 精品久久久久免费极品大片 | 免费色视频在线 | 9999在线 | 久久蜜臀av | 人人玩人人添人人澡超碰 | 综合久久精品 | 欧美一区二区三区在线播放 | 国产精品久久久久久久免费观看 | 91亚洲欧美激情 | 成人在线观看av | 午夜视频在线瓜伦 | 日本在线观看视频一区 | 成人免费在线观看入口 | 亚洲精品国产精品国产 | 精品在线播放 | 久久免费视频一区 | 久久一视频 | 性色av香蕉一区二区 | www.av中文字幕.com | 国产精品一区二区免费在线观看 | 激情视频在线高清看 | 91麻豆国产福利在线观看 | 午夜精品久久久久久久99 | 日本精品一区二区三区在线观看 | 婷婷亚洲五月色综合 | 日韩com| 欧美日韩一区二区三区视频 | 97精品超碰一区二区三区 | 精品资源在线 | 99视频在线免费看 | av观看久久久 | 色婷婷综合视频在线观看 | 欧美网址在线观看 | .国产精品成人自产拍在线观看6 | 91精品国产乱码在线观看 | 国产精品福利无圣光在线一区 | 日韩黄色网络 | 日韩免费播放 | 欧美在线一级片 | 九九久久久久久久久激情 | 色丁香婷婷 | 精品九九九九 | 久久任你操 | www.成人久久| 亚洲黄色免费观看 | www免费在线观看 | 中文字幕一区二区三区四区久久 | 成人一区二区三区在线 | 日本在线观看中文字幕无线观看 | 精品在线二区 | www.xxxx欧美| 丁香五婷| 麻豆91在线观看 | 日韩在线视频在线观看 | 国产精品一区二区无线 | 欧美a级成人淫片免费看 | 九九色在线观看 | 999久久国精品免费观看网站 | 99免费在线视频 | 97在线观看免费高清完整版在线观看 | 美女免费视频一区二区 | 国产精品久久久久999 | 久久se视频 | 免费视频一二三 | 亚洲欧美va | 五月天激情视频在线观看 | 中文字幕在线免费观看 | 西西www4444大胆在线 | 国产一级二级av | 狠狠狠色丁香婷婷综合久久五月 | 日韩一区二区三免费高清在线观看 | 丁香视频全集免费观看 | 91丝袜美腿| 色福利网 | 国产亚洲在线视频 | 国产精品久久久久久久久婷婷 | 黄色影院在线免费观看 | 91原创在线观看 | 欧美性脚交 | 日日躁夜夜躁xxxxaaaa | 亚洲一区二区麻豆 | 青春草免费在线视频 | 日日夜夜精品视频天天综合网 | 2024国产精品视频 | 国产精品九九久久99视频 | 超碰官网 | 91网在线| 97视频免费观看 | 日韩免费b| 在线看黄色av | 天天射网 | 99精品在线观看视频 | 久久精品视频一 | 久草在线在线精品观看 | 国产日本亚洲高清 | 黄色视屏在线免费观看 | 日韩一区二区免费播放 | 亚洲狠狠丁香婷婷综合久久久 | 日日添夜夜添 | 久草在 | www.色午夜 | 中文字幕av在线不卡 | 91av视频播放 | 91九色porn在线资源 | 一本一道久久a久久精品蜜桃 | 99中文字幕 | 欧美综合久久 | 国产99久久久欧美黑人 | 91视频亚洲| 欧美性生爱 | 精品在线视频观看 | 日韩欧美99 | 久久免费试看 | 日韩1页| 四虎影视精品 | 色噜噜狠狠色综合中国 | 日本不卡123 | av免费片 | 在线日本看片免费人成视久网 | 国产精品理论片在线观看 | 国产婷婷精品av在线 | 激情婷婷综合 | 国产一区网址 | 少妇18xxxx性xxxx片 | 很黄很黄的网站免费的 | 亚洲一区二区三区四区在线视频 | 伊人国产视频 | 欧美日韩精品区 | 国产色婷婷精品综合在线手机播放 | 欧美一级日韩三级 | 丁香五月缴情综合网 | 免费观看视频黄 | 免费观看视频的网站 | 三级黄色片在线观看 | 亚洲成人二区 | 99热这里有精品 | 国产成人一级电影 | 自拍超碰在线 | 美女视频黄网站 | 午夜视频免费播放 | 国产a国产a国产a | 在线观看免费成人av | 91在线蜜桃臀 | 波多野结衣电影一区 | 国产黄色免费在线观看 | 久久久久久久99精品免费观看 | 国产xxxx性hd极品 | 99久久精品日本一区二区免费 | 欧美日韩在线视频观看 | 美女免费电影 | 五月情婷婷 | 亚洲精品久久久久久久蜜桃 | 天天艹 | 手机看片中文字幕 | 亚洲成年人在线播放 | 国内久久看 | 日韩在线电影观看 | 97在线观看视频国产 | 黄色av一区二区 | 夜夜夜精品 | 亚洲精品国偷自产在线91正片 | 亚洲人天堂 | 亚洲精品在线视频观看 | 久久99热久久99精品 | 天天操夜夜叫 | 天天操夜夜摸 | 最近字幕在线观看第一季 | 国产专区在线播放 | 五月婷婷在线观看视频 | 欧美激情一区不卡 | 亚州精品成人 | 国产99在线| 精品国产视频在线观看 | 999久久久 | 亚洲成人av影片 | 成人亚洲网 | 国产又粗又猛又黄又爽 | 热久久在线视频 | 黄色毛片在线 | 亚洲视频电影在线 | 天天色天天艹 | 欧美精品九九99久久 | 97视频免费在线看 | 日韩在线观看第一页 | 国产精品1024 | 九色精品免费永久在线 | 精品一二区 | 国产精品美女免费看 | 五月天婷婷免费视频 | 久久只精品99品免费久23小说 | 国产精品igao视频网入口 | 国产精品原创 | 色偷偷97 | 久久另类视频 | 日韩a欧美 | 91女神的呻吟细腰翘臀美女 | 日本精品视频免费观看 | 亚洲精品一区二区三区新线路 | 国产福利一区二区三区在线观看 | 婷婷丁香av | 亚州国产精品久久久 | 91久久精品一区二区三区 | 天天草天天插 | 丁香激情综合国产 | 亚洲精品国产精品乱码不99热 | 国产精品片 | 在线中文字幕视频 | 中文字幕999 | 欧美日韩午夜爽爽 | 国产精品网站 | 国产资源免费 | 九九久久久久久久久激情 | 亚洲精品毛片一级91精品 | 久久久香蕉视频 | 高清在线一区 | 国产伦理久久精品久久久久_ | av网站地址| 99久久精品视频免费 | 高清av中文字幕 | 成人黄色在线观看视频 | 久久不射电影网 | 99视频在线免费 | 999超碰 | 国内精品毛片 | 精品免费久久久久久 | 日韩精品你懂的 | 国产999在线观看 | 在线观看免费一区 | av超碰在线| 成人激情开心网 | 亚洲一区尤物 | 91视频在线| 欧美热久久| 91九色精品国产 | 免费在线国产 | 日韩视频一区二区三区在线播放免费观看 | 日本性生活免费看 | 亚洲成人二区 | 在线电影日韩 | 高潮久久久久久久久 | 久久夜夜夜 | 久久女同性恋中文字幕 | 成年人免费在线看 | 五月天婷婷在线视频 | 日韩久久久久久 | 色的网站在线观看 | 日本少妇久久久 | 久久99久久99精品免观看粉嫩 | 精品国产理论片 | 久久久久久久久久久久久影院 | 精品 激情 | 欧美日韩一区二区免费在线观看 | 激情综合网五月激情 | 91看片淫黄大片在线播放 | 中文字幕高清av | 99日韩精品| 亚洲国产一区在线观看 | 免费黄在线观看 | av电影免费在线 | 日日成人网 | 日韩精品五月天 | 久久高清国产视频 | 国产美女精品视频 | 日日夜av| 亚洲第一av在线播放 | 免费看麻豆 | 国产一区二区不卡视频 | 国产精品自产拍 | 亚洲综合黄色 | 涩涩网站在线播放 | 国产精品免费视频网站 | 天天天天天天操 | 国产二区电影 | a极黄色片 | 日韩 国产| 免费观看版 | 九九免费精品视频 | 黄色a视频免费 | 精品视频久久久久久 | 天天操天天添 | 亚洲国产午夜视频 | 久久久黄视频 | 国产免费久久久久 | 超碰97国产在线 | 五月亚洲综合 | 婷婷色伊人| 日韩av在线一区二区 | 激情五月***国产精品 | 精品成人网 | 成人av资源在线 | 精品视频成人 | 992tv在线成人免费观看 | 视频在线99 | 久久电影日韩 | 91影视成人| 国产免费观看久久黄 | 99视频在线免费 | 999久久久久 | 五月天久久激情 | av片一区二区 | 六月天色婷婷 | 国产亚洲va综合人人澡精品 | 亚洲精色| av在线电影网站 | 日韩视频免费看 | 欧美精品v国产精品v日韩精品 | 成人在线视频论坛 | 超碰在线94| 国产视频2 | 日韩免费观看高清 | 国产精品久久久久久久久免费看 | 国产一级二级三级视频 | 日韩欧美在线观看一区二区三区 | 最新av网址在线观看 | 久草视频精品 | 日韩欧美一二三 | av网址在线播放 | www免费视频com━ | 97在线精品视频 | 久草免费在线观看视频 | 国产精品九九九九九九 | 成人av中文字幕在线观看 | 婷婷中文在线 | 国产精品一码二码三码在线 | 国产精品久久久毛片 | 毛片一区二区 | 午夜视频播放 | 91精品国产成人观看 | 日韩午夜大片 | 亚洲精品在线资源 | 99精品欧美一区二区三区 | 久久综合网色—综合色88 | 国产精品久久久久久久久久直播 | 欧美极度另类性三渗透 | 97综合网| 美女视频黄频 | 91自拍视频在线观看 | 超碰人人做 | 国产伦理久久精品久久久久_ | av大片免费 | 视频一区二区三区视频 | 欧美了一区在线观看 | av免费在线观看网站 | 国产精品久久久久久久久久久久久久 | 丁香六月在线观看 | 中文字幕在线观看网站 | 亚洲网久久 | 中文字幕在线第一页 | 日韩视频三区 | 欧美激情视频免费看 | 精品日韩视频 | 亚洲精品美女久久久 | 国产精品一区一区三区 | 在线播放精品一区二区三区 | 狠狠地日 | 日本大尺码专区mv | 国产麻豆精品一区 | 在线亚洲播放 | 人人爱爱| 久久全国免费视频 | 久久综合免费视频 | 国产亚洲精品成人 | 日韩在线高清免费视频 | 免费a v视频 | 999久久国产 | 激情综合网天天干 | 亚洲欧美视频一区二区三区 | 狠狠色丁香婷婷综合久久片 | 欧美另类z0zx | 欧美激情视频一二区 | 狠狠狠色丁香综合久久天下网 | 一级片视频免费观看 | 九九九国产 | 国产h在线播放 | 高清av在线免费观看 | 国产一区二区久久 | 亚洲一区二区三区精品在线观看 | 在线观看播放av | 国产精品18久久久久久久久久久久 | 五月婷婷在线视频 | 国内精品久久久久影院优 | 午夜国产在线 | 中国黄色一级大片 | 午夜精品久久久久久 | 最近中文字幕在线播放 | 久草在线最新免费 | 午夜精品久久久久久久99水蜜桃 | 337p西西人体大胆瓣开下部 | 亚洲精品高清视频在线观看 | 狠狠操天天射 | av黄色免费看 | 久久久www| 国产在线a不卡 | 国产精品一区二区久久精品爱涩 | 日日干 天天干 | 日韩在线观看视频一区二区三区 | 亚洲成aⅴ人在线观看 | 奇米网网址 | 亚洲爽爽网 | 日韩精品一区二区三区中文字幕 | 成人理论在线观看 | 国产成人黄色网址 | 日韩欧美一区二区三区免费观看 | 狠狠插狠狠干 | 日韩高清免费无专码区 | 国产精品久久久久久婷婷天堂 | www五月天| 91九色最新地址 | 亚洲欧洲成人 | 国产黄色在线看 | 免费h精品视频在线播放 | 国产精品mv在线观看 | 天堂网一区二区 |