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

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

全局程序集缓存gac中安装程序集_我就不信2W字把源码拆的这么碎,你还不明白mybatis缓存...

發布時間:2025/6/17 编程问答 75 豆豆
生活随笔 收集整理的這篇文章主要介紹了 全局程序集缓存gac中安装程序集_我就不信2W字把源码拆的这么碎,你还不明白mybatis缓存... 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

前言

不知道大家看到這張圖感覺怎么樣,不是難,一共也沒有幾個組件,但是真的讓我想當頭疼,因為在面試的時候,就這張圖,對,你沒看錯,就這幾個組件,那是讓我相當難受啊

MyBatis中SQL執行的整體過程

在 SqlSession 中,會將執行 SQL 的過程交由Executor執行器去執行,過程大致如下:

1、通過DefaultSqlSessionFactory創建與數據庫交互的 SqlSession “會話”,其內部會創建一個Executor執行器對象
2、然后Executor執行器通過StatementHandler創建對應的java.sql.Statement對象,并通過ParameterHandler設置參數,然后執行數據庫相關操作
如果是數據庫更新操作,則可能需要通過KeyGenerator先設置自增鍵,然后返回受影響的 行數
如果是數據庫查詢操作,則需要將數據庫返回的ResultSet結果集對象包裝ResultSetWrapper,然后通過DefaultResultSetHandler對結果集進行映射,最后返回 Java 對象

上面還涉及到一級緩存、二級緩存和延遲加載等其他處理過程,下面我們來看一下具體的執行過程

SQL執行過程(一)之Executor

在MyBatis的SQL執行過程中,Executor執行器擔當著一個重要的角色,相關操作都需要通過它來執行,相當于一個調度器,把SQL語句交給它,它來調用各個組件執行操作

其中一級緩存和二級緩存都是在Executor執行器中完成的

Executor執行器接口的實現類如下圖所示:

org.apache.ibatis.executor.BaseExecutor:實現Executor接口,提供骨架方法,支持一級緩存,指定幾個抽象的方法交由不同的子類去實現
org.apache.ibatis.executor.SimpleExecutor:繼承 BaseExecutor 抽象類,簡單的 Executor 實現類(默認)
org.apache.ibatis.executor.ReuseExecutor:繼承 BaseExecutor 抽象類,可重用的 Executor 實現類,相比SimpleExecutor,在Statement執行完操作后不會立即關閉,而是緩存起來,執行的SQL作為key,下次執行相同的SQL時優先從緩存中獲取Statement對象
org.apache.ibatis.executor.BatchExecutor:繼承 BaseExecutor 抽象類,支持批量執行的 Executor 實現類
org.apache.ibatis.executor.CachingExecutor:實現 Executor 接口,支持二級緩存的 Executor 的實現類,實際采用了裝飾器模式,裝飾對象為左邊三個Executor類

Executor

org.apache.ibatis.executor.Executor:執行器接口,代碼如下:

public interface Executor {/*** ResultHandler 空對象*/ResultHandler NO_RESULT_HANDLER = null;/*** 更新或者插入或者刪除* 由傳入的 MappedStatement 的 SQL 所決定*/int update(MappedStatement ms, Object parameter) throws SQLException;/*** 查詢,帶 ResultHandler + CacheKey + BoundSql*/<E> List<E> query(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler,CacheKey cacheKey, BoundSql boundSql) throws SQLException;/*** 查詢,帶 ResultHandler*/<E> List<E> query(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler)throws SQLException;/*** 查詢,返回 Cursor 游標*/<E> Cursor<E> queryCursor(MappedStatement ms, Object parameter, RowBounds rowBounds) throws SQLException;/*** 刷入批處理語句*/List<BatchResult> flushStatements() throws SQLException;/*** 提交事務*/void commit(boolean required) throws SQLException;/*** 回滾事務*/void rollback(boolean required) throws SQLException;/*** 創建 CacheKey 對象*/CacheKey createCacheKey(MappedStatement ms, Object parameterObject, RowBounds rowBounds, BoundSql boundSql);/*** 判斷是否緩存*/boolean isCached(MappedStatement ms, CacheKey key);/*** 清除本地緩存*/void clearLocalCache();/*** 延遲加載*/void deferLoad(MappedStatement ms, MetaObject resultObject, String property, CacheKey key, Class<?> targetType);/*** 獲得事務*/Transaction getTransaction();/*** 關閉事務*/void close(boolean forceRollback);/*** 判斷事務是否關閉*/boolean isClosed();/*** 設置包裝的 Executor 對象*/void setExecutorWrapper(Executor executor); }

執行器接口定義了操作數據庫的相關方法:

  • 數據庫的讀和寫操作
  • 事務相關
  • 緩存相關
  • 設置延遲加載
  • 設置包裝的 Executor 對象

BaseExecutor

org.apache.ibatis.executor.BaseExecutor:實現Executor接口,提供骨架方法,指定幾個抽象的方法交由不同的子類去實現,例如:

protected abstract int doUpdate(MappedStatement ms, Object parameter) throws SQLException;protected abstract List<BatchResult> doFlushStatements(boolean isRollback) throws SQLException;protected abstract <E> List<E> doQuery(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, BoundSql boundSql) throws SQLException;protected abstract <E> Cursor<E> doQueryCursor(MappedStatement ms, Object parameter, RowBounds rowBounds,BoundSql boundSql) throws SQLException;

上面這四個方法交由不同的子類去實現,分別是:更新數據庫、刷入批處理語句、查詢數據庫和查詢數據返回游標

構造方法

public abstract class BaseExecutor implements Executor {private static final Log log = LogFactory.getLog(BaseExecutor.class);/*** 事務對象*/protected Transaction transaction;/*** 包裝的 Executor 對象*/protected Executor wrapper;/*** DeferredLoad(延遲加載)隊列*/protected ConcurrentLinkedQueue<DeferredLoad> deferredLoads;/*** 本地緩存,即一級緩存,內部就是一個 HashMap 對象*/protected PerpetualCache localCache;/*** 本地輸出類型參數的緩存,和存儲過程有關*/protected PerpetualCache localOutputParameterCache;/*** 全局配置*/protected Configuration configuration;/*** 記錄當前會話正在查詢的數量*/protected int queryStack;/*** 是否關閉*/private boolean closed;protected BaseExecutor(Configuration configuration, Transaction transaction) {this.transaction = transaction;this.deferredLoads = new ConcurrentLinkedQueue<>();this.localCache = new PerpetualCache("LocalCache");this.localOutputParameterCache = new PerpetualCache("LocalOutputParameterCache");this.closed = false;this.configuration = configuration;this.wrapper = this;} }

其中上面的屬性可根據注釋進行查看

一級緩存

這里提一下localCache屬性,本地緩存,用于一級緩存,MyBatis的一級緩存是什么呢?

每當我們使用 MyBatis 開啟一次和數據庫的會話,MyBatis 都會創建出一個 SqlSession 對象,表示與數據庫的一次會話,而每個 SqlSession 都會創建一個 Executor 對象
在對數據庫的一次會話中,我們有可能會反復地執行完全相同的查詢語句,每一次查詢都會訪問一次數據庫,如果在極短的時間內做了完全相同的查詢,那么它們的結果極有可能完全相同,由于查詢一次數據庫的代價很大,如果不采取一些措施的話,可能造成很大的資源浪費
為了解決這一問題,減少資源的浪費,MyBatis 會在每一次 SqlSession 會話對象中建立一個簡單的緩存,將每次查詢到的結果緩存起來,當下次查詢的時候,如果之前已有完全一樣的查詢,則會先嘗試從這個簡單的緩存中獲取結果返回給用戶,不需要再進行一次數據庫查詢了 注意,這個“簡單的緩存”就是一級緩存,且默認開啟,無法“關閉”
MyBatis 的一次會話:在一個 SqlSession 會話對象中創建一個localCache本地緩存,對于每一次查詢,都會根據查詢條件嘗試去localCache本地緩存中獲取緩存數據,如果存在,就直接從緩存中取出數據然后返回給用戶,否則訪問數據庫進行查詢,將查詢結果存入緩存并返回給用戶(如果設置的緩存區域為STATEMENT,默認為SESSION,在一次會話中所有查詢執行后會清空當前 SqlSession 會話中的localCache本地緩存,相當于“關閉”了一級緩存)
所有的數據庫更新操作都會清空當前 SqlSession 會話中的本地緩存
如上描述,MyBatis的一級緩存在多個 SqlSession 會話時,可能導致數據的不一致性,某一個 SqlSession 更新了數據而其他 SqlSession 無法獲取到更新后的數據,出現數據不一致性,這種情況是不允許出現了,所以我們通常選擇“關閉”一級緩存

clearLocalCache方法

clearLocalCache()方法,清空一級(本地)緩存,如果全局配置中設置的localCacheScope緩存區域為STATEMENT(默認為SESSION),則在每一次查詢后會調用該方法,相當于關閉了一級緩存,代碼如下:

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

createCacheKey方法

createCacheKey(MappedStatement ms, Object parameterObject, RowBounds rowBounds, BoundSql boundSql)方法,根據本地查詢的相關信息創建一個CacheKey緩存key對象,代碼如下:

@Override public CacheKey createCacheKey(MappedStatement ms, Object parameterObject, RowBounds rowBounds, BoundSql boundSql) {if (closed) {throw new ExecutorException("Executor was closed.");}// <1> 創建 CacheKey 對象CacheKey cacheKey = new CacheKey();// <2> 設置 id、offset、limit、sql 到 CacheKey 對象中cacheKey.update(ms.getId());cacheKey.update(rowBounds.getOffset());cacheKey.update(rowBounds.getLimit());cacheKey.update(boundSql.getSql());// <3> 設置 ParameterMapping 數組的元素對應的每個 value 到 CacheKey 對象中List<ParameterMapping> parameterMappings = boundSql.getParameterMappings();TypeHandlerRegistry typeHandlerRegistry = ms.getConfiguration().getTypeHandlerRegistry();// mimic DefaultParameterHandler logicfor (ParameterMapping parameterMapping : parameterMappings) {if (parameterMapping.getMode() != ParameterMode.OUT) { // 該參數需要作為入參Object value;String propertyName = parameterMapping.getProperty();/** 獲取該屬性值*/if (boundSql.hasAdditionalParameter(propertyName)) {// 從附加參數中獲取value = boundSql.getAdditionalParameter(propertyName);} else if (parameterObject == null) {// 入參對象為空則直接返回 nullvalue = null;} else if (typeHandlerRegistry.hasTypeHandler(parameterObject.getClass())) {// 入參有對應的類型處理器則直接返回該參數value = parameterObject;} else {// 從入參對象中獲取該屬性的值MetaObject metaObject = configuration.newMetaObject(parameterObject);value = metaObject.getValue(propertyName);}cacheKey.update(value);}}// <4> 設置 Environment.id 到 CacheKey 對象中if (configuration.getEnvironment() != null) {// issue #176cacheKey.update(configuration.getEnvironment().getId());}return cacheKey; }
  • 創建一個CacheKey實例對象
  • 將入參中的id、offset、limit、sql,通過CacheKey的update方法添加到其中,它的方法如下:public void update(Object object) { // 方法參數 object 的 hashcode int baseHashCode = object == null ? 1 : ArrayUtil.hashCode(object); this.count++; // checksum 為 baseHashCode 的求和 this.checksum += baseHashCode; // 計算新的 hashcode 值 baseHashCode *= this.count; this.hashcode = this.multiplier * this.hashcode + baseHashCode; // 添加 object 到 updateList 中 this.updateList.add(object); }
  • 獲取本次查詢的入參值,通過CacheKey的update方法添加到其中
  • 獲取本次環境的Environment.id,通過CacheKey的update方法添加到其中
  • 返回CacheKey實例對象,這樣就可以為本次查詢生成一個唯一的緩存key對象,可以看看CacheKey重寫的equal方法:
  • @Override public boolean equals(Object object) {if (this == object) {return true;} if (!(object instanceof CacheKey)) { return false; } final CacheKey cacheKey = (CacheKey) object;if (hashcode != cacheKey.hashcode) { return false;} if (checksum != cacheKey.checksum) { return false;} if (count != cacheKey.count) {return false; } for (int i = 0; i < updateList.size(); i++) { Object thisObject = updateList.get(i); Object thatObject = cacheKey.updateList.get(i); if (!ArrayUtil.equals(thisObject, thatObject)) { return false; } } return true; }

    query相關方法

    查詢數據庫因為涉及到一級緩存,所以這里有多層方法,最終訪問數據庫的doQuery方法是交由子類去實現的,總共分為三層:

    1、根據入參獲取BoundSql和CacheKey對象,然后再去調用查詢方法
    2、涉及到一級緩存和延遲加載的處理,緩存未命中則再去調用查詢數據庫的方法
    3、保存一些信息供一級緩存使用,內部調用doQuery方法執行數據庫的讀操作

    接下來我們分別來看看這三個方法

    ① 數據庫查詢操作的入口

    代碼格式:query(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler)方法,代碼如下

    @Override public <E> List<E> query(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler)throws SQLException {// <1> 獲得 BoundSql 對象BoundSql boundSql = ms.getBoundSql(parameter);// <2> 創建 CacheKey 對象CacheKey key = createCacheKey(ms, parameter, rowBounds, boundSql);// <3> 查詢return query(ms, parameter, rowBounds, resultHandler, key, boundSql); }
  • 通過MappedStatement對象根據入參獲取BoundSql對象,如果是動態SQL則需要進行解析,獲取到最終的SQL,替換成?占位符
  • 調用createCacheKey方法為本次查詢創建一個CacheKey對象
  • 繼續調用query(...)方法執行查詢
  • ② 處理數據庫查詢操作,涉及到一級緩存

    代碼格式:query(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, CacheKey key, BoundSql boundSql)方法,代碼如下:

    @Override public <E> List<E> query(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler,CacheKey key, BoundSql boundSql) throws SQLException {ErrorContext.instance().resource(ms.getResource()).activity("executing a query").object(ms.getId());// <1> 已經關閉,則拋出 ExecutorException 異常if (closed) {throw new ExecutorException("Executor was closed.");}// <2> 清空本地緩存,如果 queryStack 為零,并且要求清空本地緩存(配置了 flushCache = true)if (queryStack == 0 && ms.isFlushCacheRequired()) {clearLocalCache();}List<E> list;try {// <3> queryStack + 1queryStack++;// <4> 從一級緩存中,獲取查詢結果list = resultHandler == null ? (List<E>) localCache.getObject(key) : null;if (list != null) { // <4.1> 獲取到,則進行處理// 處理緩存存儲過程的結果handleLocallyCachedOutputParameters(ms, key, parameter, boundSql);} else { // <4.2> 獲得不到,則從數據庫中查詢list = queryFromDatabase(ms, parameter, rowBounds, resultHandler, key, boundSql);}} finally {// <5> queryStack - 1queryStack--;}if (queryStack == 0) { // <6> 如果當前會話的所有查詢執行完了// <6.1> 執行延遲加載for (DeferredLoad deferredLoad : deferredLoads) {deferredLoad.load();}// issue #601// <6.2> 清空 deferredLoadsdeferredLoads.clear();// <6.3> 如果緩存級別是 LocalCacheScope.STATEMENT ,則進行清理if (configuration.getLocalCacheScope() == LocalCacheScope.STATEMENT) {// issue #482clearLocalCache();}}// <7> 返回查詢結果return list; }
  • 當前會話已經被關閉則拋出異常
  • 如果queryStack為0(表示是當前會話只有本次查詢而沒有其他的查詢了),并且要求清空本地緩存(配置了flushCache=true),那么直接清空一級(本地)緩存
  • 當前會話正在查詢的數量加一,queryStack++
  • 從localCache一級緩存獲取緩存的查詢結果如果有緩存數據,則需要處理儲存過程的情況,將需要作為出參(OUT)的參數設置到本次查詢的入參的屬性中如果沒有緩存數據,則調用queryFromDatabase方法,執行數據庫查詢操作
  • 當前會話正在查詢的數量減一,queryStack--
  • 如果當前會話所有查詢都執行完執行當前會話中的所有的延遲加載deferredLoads,這種延遲加載屬于查詢后的延遲,和后續講到的獲取屬性時再加載不同,這里的延遲加載是在哪里生成的呢?在DefaultResultSetHandler中進行結果映射時,如果某個屬性配置的是子查詢,并且本次的子查詢在一級緩存中有緩存數據,那么將會創建一個DeferredLoad對象保存在deferredLoads中,該屬性值先設置為DEFERRED延遲加載對象(final修飾的Object對象),待當前會話所有的查詢結束后,也就是當前執行步驟,則會從一級緩存獲取到數據設置到返回結果中清空所有的延遲加載deferredLoads對象如果全局配置的緩存級別為STATEMENT(默認為SESSION),則清空當前會話中一級緩存的所有數據
  • 返回查詢結果
  • ③ 執行數據庫查詢操作

    代碼格式:queryFromDatabase(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, CacheKey key, BoundSql boundSql)方法,代碼如下:

    private <E> List<E> queryFromDatabase(MappedStatement ms, Object parameter, RowBounds rowBounds,ResultHandler resultHandler, CacheKey key, BoundSql boundSql) throws SQLException {List<E> list;// <1> 在緩存中,添加正在執行的占位符對象,因為正在執行的查詢不允許提前加載需要延遲加載的屬性,可見 DeferredLoad#canLoad() 方法localCache.putObject(key, EXECUTION_PLACEHOLDER);try {// <2> 執行讀操作list = doQuery(ms, parameter, rowBounds, resultHandler, boundSql);} finally {// <3> 從緩存中,移除占位對象localCache.removeObject(key);}// <4> 添加到緩存中localCache.putObject(key, list);// <5> 如果是存儲過程,則將入參信息保存保存,跟一級緩存處理存儲過程相關if (ms.getStatementType() == StatementType.CALLABLE) {localOutputParameterCache.putObject(key, parameter);}// <6> 返回查詢結果return list; }
  • 在緩存中,添加正在執行的EXECUTION_PLACEHOLDER占位符對象,因為正在執行的查詢不允許提前加載需要延遲加載的屬性,可見 DeferredLoad#canLoad() 方法
  • 調用查詢數據庫doQuery方法,該方法交由子類實現
  • 刪除第1步添加的占位符
  • 將查詢結果添加到localCache一級緩存中
  • 如果是存儲過程,則將入參信息保存保存,跟一級緩存處理存儲過程相關,可見上面的第②個方法的第4.1步
  • 返回查詢結果
  • update方法

    update(MappedStatement ms, Object parameter)方法,執行更新數據庫的操作,代碼如下:

    @Override public int update(MappedStatement ms, Object parameter) throws SQLException {ErrorContext.instance().resource(ms.getResource()).activity("executing an update").object(ms.getId());// <1> 已經關閉,則拋出 ExecutorException 異常if (closed) {throw new ExecutorException("Executor was closed.");}// <2> 清空本地緩存clearLocalCache();// <3> 執行寫操作return doUpdate(ms, parameter); }
  • 當前會話已經被關閉則拋出異常
  • 清空當前會話中一級緩存的所有數據
  • 調用更新數據庫doUpdate方法,該方法交由子類實現
  • 其他方法

    除了上面介紹的幾個重要的方法以外,還有其他很多方法,例如獲取當前事務,提交事務,回滾事務,關閉會話等等,這里我就不一一列出來了,大家可以自行閱讀類源碼,我下面會進行涉及,但是不會那么精細

    二級緩存

    問題

    在講到的一級緩存中,緩存數據僅在當前的 SqlSession 會話中進行共享,可能會導致多個 SqlSession 出現數據不一致性的問題

    如果需要在多個 SqlSession 之間需要共享緩存數據,則需要使用到二級緩存

    開啟二級緩存后,會使用CachingExecutor對象裝飾其他的Executor類,這樣會先在CachingExecutor進行二級緩存的查詢,緩存未命中則進入裝飾的對象中,進行一級緩存的查詢

    流程如下圖所示:

    在全局配置對象中cacheEnabled是否開啟緩存屬性默認為true,可以在mybatis-config.xml配置文件中添加以下配置關閉:

    <configuration><settings><setting name="cacheEnabled" value="false" /></settings> </configuration>

    我們來看看MyBatis是如何實現二級緩存的

    CachingExecutor

    org.apache.ibatis.executor.CachingExecutor:實現 Executor 接口,支持二級緩存的 Executor 的實現類

    構造方法

    public class CachingExecutor implements Executor {/*** 被委托的 Executor 對象*/private final Executor delegate;/*** TransactionalCacheManager 對象*/private final TransactionalCacheManager tcm = new TransactionalCacheManager();public CachingExecutor(Executor delegate) {this.delegate = delegate;// 設置 delegate 被當前執行器所包裝delegate.setExecutorWrapper(this);} }
    • delegate 屬性,為被委托的Executor對象,具體的數據庫操作都是交由它去執行
    • tcm 屬性,TransactionalCacheManager對象,支持事務的緩存管理器,因為二級緩存是支持跨 SqlSession 共享的,此處需要考慮事務,那么,必然需要做到事務提交時,才將當前事務中查詢時產生的緩存,同步到二級緩存中,所以需要通過TransactionalCacheManager來實現

    query方法

    處理數據庫查詢操作的方法,涉及到二級緩存,會將Cache二級緩存對象裝飾成TransactionalCache對象并存放在TransactionalCacheManager管理器中,代碼如下:

    @Override public <E> List<E> query(MappedStatement ms, Object parameterObject, RowBounds rowBounds,ResultHandler resultHandler, CacheKey key, BoundSql boundSql) throws SQLException {// <1> 獲取 Cache 二級緩存對象Cache cache = ms.getCache();// <2> 如果配置了二級緩存if (cache != null) {// <2.1> 如果需要清空緩存,則進行清空flushCacheIfRequired(ms);// <2.2> 如果當前操作需要使用緩存(默認開啟)if (ms.isUseCache() && resultHandler == null) {// <2.2.1> 如果是存儲過程相關操作,保證所有的參數模式為 ParameterMode.INensureNoOutParams(ms, boundSql);// <2.2.2> 從二級緩存中獲取結果,會裝飾成 TransactionalCacheList<E> list = (List<E>) tcm.getObject(cache, key);if (list == null) {// <2.2.3> 如果不存在,則從數據庫中查詢list = delegate.query(ms, parameterObject, rowBounds, resultHandler, key, boundSql);// <2.2.4> 將緩存結果保存至 TransactionalCachetcm.putObject(cache, key, list); // issue #578 and #116}// <2.2.5> 直接返回結果return list;}}// <3> 沒有使用二級緩存,則調用委托對象的方法return delegate.query(ms, parameterObject, rowBounds, resultHandler, key, boundSql); }
  • 獲取Cache二級緩存對象
  • 如果該對象不為空,表示配置了二級緩存如果需要清空緩存,則進行清空如果當前操作需要使用緩存(默認開啟)如果是存儲過程相關操作,保證所有的參數模式為http://ParameterMode.IN通過TransactionalCacheManager從二級緩存中獲取結果,會裝飾成TransactionalCach對象如果緩存未命中,則調用委托對象的query方法將緩存結果保存至TransactionalCache對象中,并未真正的保存至Cache二級緩存中,需要待事務提交才會保存過去,其中緩存未命中的也會設置緩存結果為null直接返回結果
  • 沒有使用二級緩存,則調用委托對象的方法
  • update方法

    @Override public int update(MappedStatement ms, Object parameterObject) throws SQLException {// 如果需要清空緩存,則進行清空flushCacheIfRequired(ms);// 執行 delegate 對應的方法return delegate.update(ms, parameterObject); }private void flushCacheIfRequired(MappedStatement ms) {Cache cache = ms.getCache();if (cache != null && ms.isFlushCacheRequired()) {tcm.clear(cache);} }

    數據庫的更新操作,如果配置了需要清空緩存,則清空二級緩存

    這里就和一級緩存不同,一級緩存是所有的更新操作都會清空一級緩存

    commit方法

    @Override public void commit(boolean required) throws SQLException {// 執行 delegate 對應的方法delegate.commit(required);// 提交 TransactionalCacheManagertcm.commit(); }

    在事務提交后,通過TransactionalCacheManager二級緩存管理器,將本次事務生成的緩存數據從TransactionalCach中設置到正真的Cache二級緩存中

    rollback方法

    @Override public void rollback(boolean required) throws SQLException {try {// 執行 delegate 對應的方法delegate.rollback(required);} finally {if (required) {// 回滾 TransactionalCacheManagertcm.rollback();}} }

    在事務回滾后,如果需要的話,通過TransactionalCacheManager二級緩存管理器,將本次事務生成的緩存數據從TransactionalCach中移除

    close方法

    @Override public void close(boolean forceRollback) {try {// issues #499, #524 and #573if (forceRollback) {tcm.rollback();} else {tcm.commit();}} finally {delegate.close(forceRollback);} }

    在事務關閉前,如果是強制回滾操作,則TransactionalCacheManager二級緩存管理器,將本次事務生成的緩存數據從TransactionalCach中移除,否則還是將緩存數據設置到正真的Cache二級緩存中

    TransactionalCacheManager

    org.apache.ibatis.cache.TransactionalCacheManager:二級緩存管理器,因為二級緩存是支持跨 SqlSession 共享的,所以需要通過它來實現,當事務提交時,才將當前事務中查詢時產生的緩存,同步到二級緩存中,代碼如下:

    public class TransactionalCacheManager {/*** Cache 和 TransactionalCache 的映射*/private final Map<Cache, TransactionalCache> transactionalCaches = new HashMap<>();public void clear(Cache cache) {getTransactionalCache(cache).clear();}public Object getObject(Cache cache, CacheKey key) {return getTransactionalCache(cache).getObject(key);}public void putObject(Cache cache, CacheKey key, Object value) {// 首先,獲得 Cache 對應的 TransactionalCache 對象// 然后,添加 KV 到 TransactionalCache 對象中getTransactionalCache(cache).putObject(key, value);}public void commit() {for (TransactionalCache txCache : transactionalCaches.values()) {txCache.commit();}}public void rollback() {for (TransactionalCache txCache : transactionalCaches.values()) {txCache.rollback();}}private TransactionalCache getTransactionalCache(Cache cache) {return transactionalCaches.computeIfAbsent(cache, TransactionalCache::new);}}
    • getTransactionalCache(Cache cache)方法,根據Cache二級緩存對象獲取對應的TransactionalCache對象,如果沒有則創建一個保存起來
    • getObject(Cache cache, CacheKey key)方法,會先調用getTransactionalCache(Cache cache)方法獲取對應的TransactionalCache對象,然后根據CacheKey從該對象中獲取緩存結果
    • putObject(Cache cache, CacheKey key, Object value)方法,同樣也先調用getTransactionalCache(Cache cache)方法獲取對應的TransactionalCache對象,根據該對象將結果進行緩存
    • commit()方法,遍歷transactionalCaches,依次調用TransactionalCache的提交方法
    • rollback()方法,遍歷transactionalCaches,依次調用TransactionalCache的回滾方法

    TransactionalCache

    org.apache.ibatis.cache.decorators.TransactionalCache:用來裝飾二級緩存的對象,作為二級緩存一個事務的緩沖區

    在一個SqlSession會話中,該類包含所有需要添加至二級緩存的的緩存數據,當提交事務后會全部刷出到二級緩存中,或者事務回滾后移除這些緩存數據,代碼如下:

    public class TransactionalCache implements Cache {private static final Log log = LogFactory.getLog(TransactionalCache.class);/*** 委托的 Cache 對象。** 實際上,就是二級緩存 Cache 對象。*/private final Cache delegate;/*** 提交時,清空 {@link #delegate}** 初始時,該值為 false* 清理后{@link #clear()} 時,該值為 true ,表示持續處于清空狀態** 因為可能事務還未提交,所以不能直接清空所有的緩存,而是設置一個標記,獲取緩存的時候返回 null 即可* 先清空下面這個待提交變量,待事務提交的時候才真正的清空緩存**/private boolean clearOnCommit;/*** 待提交的 Key-Value 映射*/private final Map<Object, Object> entriesToAddOnCommit;/*** 查找不到的 KEY 集合*/private final Set<Object> entriesMissedInCache;public TransactionalCache(Cache delegate) {this.delegate = delegate;this.clearOnCommit = false;this.entriesToAddOnCommit = new HashMap<>();this.entriesMissedInCache = new HashSet<>();}@Overridepublic Object getObject(Object key) {// issue #116// <1> 從 delegate 中獲取 key 對應的 valueObject object = delegate.getObject(key);if (object == null) {// <2> 如果不存在,則添加到 entriesMissedInCache 中entriesMissedInCache.add(key);}// issue #146if (clearOnCommit) {// <3> 如果 clearOnCommit 為 true ,表示處于持續清空狀態,則返回 nullreturn null;} else {return object;}}@Overridepublic void putObject(Object key, Object object) {// 暫存 KV 到 entriesToAddOnCommit 中entriesToAddOnCommit.put(key, object);}@Overridepublic void clear() {// <1> 標記 clearOnCommit 為 trueclearOnCommit = true;// <2> 清空 entriesToAddOnCommitentriesToAddOnCommit.clear();}public void commit() {// <1> 如果 clearOnCommit 為 true ,則清空 delegate 緩存if (clearOnCommit) {delegate.clear();}// 將 entriesToAddOnCommit、entriesMissedInCache 刷入 delegate 中flushPendingEntries();// 重置reset();}public void rollback() {// <1> 從 delegate 移除出 entriesMissedInCacheunlockMissedEntries();// <2> 重置reset();}private void reset() {clearOnCommit = false;entriesToAddOnCommit.clear();entriesMissedInCache.clear();}private void flushPendingEntries() {for (Map.Entry<Object, Object> entry : entriesToAddOnCommit.entrySet()) {delegate.putObject(entry.getKey(), entry.getValue());}for (Object entry : entriesMissedInCache) {if (!entriesToAddOnCommit.containsKey(entry)) {delegate.putObject(entry, null);}}}private void unlockMissedEntries() {for (Object entry : entriesMissedInCache) {try {delegate.removeObject(entry);} catch (Exception e) {log.warn("Unexpected exception while notifiying a rollback to the cache adapter."+ "Consider upgrading your cache adapter to the latest version. Cause: " + e);}}} }

    根據上面的注釋查看每個屬性的作用,我們依次來看下面的方法,看看在不同事務之前是如何處理二級緩存的

    • putObject(Object key, Object object)方法,添加緩存數據時,先把緩存數據保存在entriesToAddOnCommit中,這個對象屬于當前事務,事務還未提交,其他事務是不能訪問到的
    • clear()方法,設置clearOnCommit標記為true,告訴當前事務正處于持續清空狀態,先把entriesToAddOnCommit清空,也就是當前事務中還未提交至二級緩存的緩存數據,事務還未提交,不能直接清空二級緩存中的數據,否則影響到其他事務了
    • commit()方法,事務提交后,如果clearOnCommit為true,表示正處于持續清空狀態,需要先把二級緩存中的數據全部清空,然后再把當前事務生成的緩存設置到二級緩存中,然后重置當前對象這里為什么處于清空狀態把二級緩存的數據清空后,還要將當前事務生成的緩存數據再設置到二級緩存中呢?因為當前事務調用clear()方法后可能有新生成了新的緩存數據,而不能把這些忽略掉
    • getObject(Object key)方法先從delegate二級緩存對象中獲取結果如果緩存未命中則將該key添加到entriesMissedInCache屬性中,因為二級緩存也會將緩存未命中的key起來,數據為null如果clearOnCommit為true,即使你緩存命中了也返回null,因為觸發clear()方法的話,本來需要清空二級緩存的,但是事務還未提交,所以先標記一個緩存持續清理的這么一個狀態,這樣相當于在當前事務中既清空了二級緩存數據,也不影響其他事務的二級緩存數據返回獲取到的結果,可能為null

    Executor在哪被創建

    前面對Executor執行器接口以及實現類都有分析過,那么它是在哪創建的呢?

    public class DefaultSqlSessionFactory implements SqlSessionFactory {private final Configuration configuration;@Overridepublic SqlSession openSession() {return openSessionFromDataSource(configuration.getDefaultExecutorType(), null, false);}private SqlSession openSessionFromDataSource(ExecutorType execType, TransactionIsolationLevel level,boolean autoCommit) {Transaction tx = null;try {// 獲得 Environment 對象final Environment environment = configuration.getEnvironment();// 創建 Transaction 對象final TransactionFactory transactionFactory = getTransactionFactoryFromEnvironment(environment);tx = transactionFactory.newTransaction(environment.getDataSource(), level, autoCommit);// 創建 Executor 對象final Executor executor = configuration.newExecutor(tx, execType);// 創建 DefaultSqlSession 對象return new DefaultSqlSession(configuration, executor, autoCommit);} catch (Exception e) {// 如果發生異常,則關閉 Transaction 對象closeTransaction(tx); // may have fetched a connection so lets call close()throw ExceptionFactory.wrapException("Error opening session. Cause: " + e, e);} finally {ErrorContext.instance().reset();}} }

    我們所有的數據庫操作都是在MyBatis的一個SqlSession會話中執行的,在它被創建的時候,會先通過Configuration全局配置對象的newExecutor方法創建一個Executor執行器

    newExecutor(Transaction transaction, ExecutorType executorType)方法,根據執行器類型創建執行Executor執行器,代碼如下:

    public Executor newExecutor(Transaction transaction, ExecutorType executorType) {// <1> 獲得執行器類型executorType = executorType == null ? defaultExecutorType : executorType;executorType = executorType == null ? ExecutorType.SIMPLE : executorType;// <2> 創建對應實現的 Executor 對象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);}// <3> 如果開啟緩存,創建 CachingExecutor 對象,進行包裝if (cacheEnabled) {executor = new CachingExecutor(executor);}// <4> 應用插件executor = (Executor) interceptorChain.pluginAll(executor);return executor; }
  • 獲得執行器類型,默認為SIMPLE
  • 創建對應的Executor對象,默認就是SimpleExecutor執行器了
  • 如果全局配置了開啟二級緩存,則將Executor對象,封裝成CachingExecutor對象
  • 插件鏈應用該對象,在后續會講到
  • 啊,不知不覺2W字了,寫源碼的東西是真的浪費時間的好東西啊,比刷抖音都消耗時間。但是也是真的香,看著自己整理的這些筆記,那是相當有成就感

    最后,希望各位在學習的過程中,能夠重視一下源碼的閱讀,因為源碼中很多的方法其實都已經注釋好了,很方便閱讀

    總結

    以上是生活随笔為你收集整理的全局程序集缓存gac中安装程序集_我就不信2W字把源码拆的这么碎,你还不明白mybatis缓存...的全部內容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

    永久av免费在线观看 | 亚洲国产中文字幕 | 国产传媒一区在线 | 在线观看的av网站 | 国产精品第一 | 久久免费视频在线观看 | 午夜色婷婷 | 久久精品www人人爽人人 | 亚洲免费视频在线观看 | 91精品国产乱码久久桃 | 日韩欧美在线第一页 | 日韩欧美在线中文字幕 | 国产精品免费麻豆入口 | 午夜精品久久久久久久99水蜜桃 | 欧美一级视频免费 | 超碰在线个人 | 国产偷国产偷亚洲清高 | 免费在线观看国产黄 | 国产精品video爽爽爽爽 | 久热免费在线观看 | 日韩视 | 久久激情小视频 | 在线免费高清一区二区三区 | 99久久er热在这里只有精品66 | 99国产在线观看 | 黄色.com | 久久综合色播五月 | 久久久久久蜜桃一区二区 | av电影中文 | 国产99久久精品一区二区永久免费 | 国内久久精品视频 | 免费三级影片 | 国内精品久久久 | 一本色道久久综合亚洲二区三区 | 香蕉视频在线观看免费 | 免费高清在线观看成人 | 久久免费国产精品1 | 日韩精品一区二区在线 | 97色涩| 天天爽天天摸 | 91自拍91| 久久夜色精品国产亚洲aⅴ 91chinesexxx | 免费日韩高清 | 久久玖| 国产99一区视频免费 | 天天爱天天舔 | 天天干天天射天天插 | 91免费在线视频 | 99久久综合狠狠综合久久 | 91人网站| 国产精品视频观看 | 国产一级在线观看 | 欧美日韩视频一区二区三区 | 激情五月婷婷丁香 | 亚洲欧美国产日韩在线观看 | 国产成人综合在线观看 | 97成人精品视频在线播放 | 色在线最新| 丁香六月天 | 久久久久久久久久国产精品 | 黄色大片日本 | 亚洲一级片av | 国产视频精品久久 | 久久久久久久久久久影视 | 91视频免费观看 | 亚洲免费高清视频 | 综合色亚洲 | 亚洲最大免费成人网 | 93久久精品日日躁夜夜躁欧美 | www在线免费观看 | 中文字幕av免费观看 | 日韩色av色资源 | 国产破处在线播放 | 免费黄色小网站 | 99se视频在线观看 | 国产精品毛片久久久 | 国内视频1区 | 天天做天天爱天天爽综合网 | 九九热久久免费视频 | 国产视频在线观看一区 | 五月婷综合网 | 这里只有精品视频在线观看 | 国产不卡免费 | 亚洲精品国产精品国自产观看 | 精品国产一区二区三区噜噜噜 | 国产破处在线视频 | 特级黄色片免费看 | 亚洲最新视频在线播放 | 色综合久久悠悠 | 国产午夜精品一区二区三区欧美 | 超碰个人在线 | 国产精品中文字幕在线 | 国产精品久久久久影院 | 成人av播放 | 99精品国产亚洲 | 色www.| 99久久夜色精品国产亚洲 | 这里只有精品视频在线 | 成人在线视频一区 | 四虎在线观看 | 国产精品久久网站 | 成人免费在线电影 | 中文字幕在线观看第三页 | www久久精品 | 国产精品久久久久久五月尺 | 深夜免费福利网站 | 国产精品久久久久免费a∨ 欧美一级性生活片 | 狠狠色狠狠色综合日日92 | 97超在线 | 国产人成免费视频 | 97国产在线视频 | www.国产在线视频 | 六月天综合网 | 日韩欧美不卡 | 日本久久精 | 在线视频欧美亚洲 | 久久久久久国产精品免费 | 国产黄色片免费 | 国产精品入口麻豆 | 男女激情网址 | 一级黄色片在线 | 亚洲精品在线视频观看 | 国产福利网站 | 在线精品视频免费观看 | 欧美伦理一区二区 | 97av精品 | 婷婷av电影 | 色婷婷久久久综合中文字幕 | av五月婷婷 | 97人人人| 成人久久18免费 | 免费色网站 | 色九九在线 | 日本中文字幕在线免费观看 | 国产 一区二区三区 在线 | 91高清免费在线观看 | 特级毛片网站 | 91精品少妇偷拍99 | av免费看电影 | 2021av在线 | 日韩最新理论电影 | 91在线看片| 亚洲欧美国产精品久久久久 | 日韩精品一区二 | 五月婷婷丁香六月 | 一级片免费观看视频 | 国产精品久久一区二区无卡 | 91福利在线观看 | 91自拍视频在线 | 亚洲人成精品久久久久 | 国产午夜一区二区 | 夜夜躁日日躁狠狠躁 | 日本最大色倩网站www | 日韩在线视频播放 | 激情 一区二区 | 亚洲涩涩涩 | 最近中文字幕免费视频 | 欧美一区二区三区在线视频观看 | 182午夜在线观看 | www.久久色 | 99久久久| 嫩模bbw搡bbbb搡bbbb | 国产精品久久久久一区二区国产 | 青草视频在线免费 | 日本中文字幕一二区观 | 免费看日韩 | 亚洲精品乱码久久久久久蜜桃不爽 | 天天综合视频在线观看 | 美州a亚洲一视本频v色道 | 成人在线视频观看 | 精品久久久久久久久久久久久久久久 | 精品在线你懂的 | 久久综合电影 | 99免费国产| 一级特黄av | 欧美日韩视频网站 | 久久香蕉国产 | 九九视频一区 | 99热国产在线 | 精品专区 | 久久精品国产成人精品 | 天天添夜夜操 | 美女免费视频观看网站 | 久人人| 午夜美女视频 | 黄色大片日本免费大片 | 日本午夜在线亚洲.国产 | 久久99热这里只有精品国产 | 91成人看片| 国产a视频免费观看 | 一级理论片在线观看 | 在线免费中文字幕 | 91视频 - 88av | 美女啪啪图片 | 天天射天天拍 | 久久精品中文字幕一区二区三区 | 福利视频一区二区 | a级免费观看 | 天天干,天天插 | www亚洲精品 | 免费一级片视频 | 天天操夜夜操 | 天天色天天操综合网 | 亚洲 成人 欧美 | 五月婷久久 | 91丨九色丨高潮丰满 | 黄色网址中文字幕 | www.亚洲精品视频 | 中文字幕在线观看91 | 日韩激情久久 | 久久国产精品色婷婷 | 成人激情开心网 | 狠狠色丁香婷婷综合基地 | 中文字幕av有码 | 99久久精品久久亚洲精品 | av电影在线不卡 | 久草在线免费资源 | 国产午夜精品一区 | 九九九九热精品免费视频点播观看 | 国产精品毛片一区二区三区 | 亚洲精品免费在线播放 | 伊人影院av | 精品久久久久亚洲 | 一级黄色片在线免费观看 | 高清一区二区三区av | 五月天免费网站 | 五月天激情在线 | 麻豆精品视频在线 | 久久丁香网 | 国产精品久久电影观看 | 97碰碰精品嫩模在线播放 | 日本3级在线观看 | 一区二区三区四区在线免费观看 | 亚洲精品午夜国产va久久成人 | 91久久国产综合精品女同国语 | 免费看一级特黄a大片 | 日韩欧美在线观看一区二区 | 国产一区二区在线视频观看 | 精品国产免费久久 | 亚洲一区二区视频 | 人人干人人超 | 女人18精品一区二区三区 | 亚洲日本一区二区在线 | 亚洲成年人免费网站 | 国产操在线 | 日韩免费在线一区 | 精品一区二区在线免费观看 | 国产xxxx | 九九热免费视频在线观看 | 97精品国产手机 | 国产精品九九久久99视频 | 天堂麻豆| 久久精品久久99 | 美女久久网站 | 国产精品高 | 天天色图| 成人黄色在线看 | 国产v亚洲v| 99 视频 高清| 成人欧美亚洲 | 欧美日韩二区在线 | 91九色自拍 | 亚洲精品国产品国语在线 | 中文字幕日本电影 | 亚洲精品在线免费播放 | 91亚洲精品国偷拍 | 久久99在线观看 | 日韩在线中文字幕 | 99在线视频播放 | 国产精彩视频 | 91精品国产91久久久久久三级 | 国产精品久久久久久久免费观看 | 婷婷九月丁香 | 日韩精品一区二区在线 | 人人澡人人舔 | 国产高清视频色在线www | 91视频88av| 日韩在线电影 | 国产精品一区二区中文字幕 | 免费网址你懂的 | 国产一级久久 | 欧美日本不卡视频 | 亚洲视频免费视频 | 精品国产午夜 | 一区二区成人国产精品 | 四虎影视欧美 | av福利在线免费观看 | 五月开心婷婷 | av在线免费在线 | 国产精品高潮呻吟久久av无 | 2024国产精品视频 | 五月天最新网址 | 日本黄色免费电影网站 | 三级黄免费看 | 一区二区三区精品在线视频 | 亚洲少妇xxxx | 国产精品日韩高清 | 手机看国产毛片 | 97热久久免费频精品99 | 不卡日韩av| 日日草天天草 | 国产精品一区二区免费 | 免费看黄网站在线 | 五月激情丁香 | 国产精品女人久久久 | 激情久久影院 | 欧美人操人| 日韩精品视频在线观看网址 | 国产精品久久久久久五月尺 | wwwww.国产| 欧美激情视频一区二区三区 | 日本精品视频在线观看 | 又黄又爽又无遮挡免费的网站 | 亚洲精色 | 在线观看aaa | 99麻豆视频 | 中文字幕在线免费看线人 | 国产精品igao视频网网址 | 激情大尺度视频 | 国产亚洲一区二区在线观看 | 亚州精品视频 | 欧美精品中文字幕亚洲专区 | 婷婷激情网站 | 97免费视频在线 | 日韩精品免费在线 | 国产不卡精品视频 | 日韩精品一卡 | 六月丁香婷 | 国产五月婷 | 在线免费观看视频一区 | 国产在线色视频 | 国产传媒一区在线 | 三级在线视频播放 | 色婷婷综合久久久 | 69国产盗摄一区二区三区五区 | 成人在线黄色电影 | 一区二区三区电影在线播 | 成人黄色大片网站 | 国产黄在线播放 | 91桃色在线观看视频 | 久草国产精品 | 91探花在线视频 | 国内精品国产三级国产aⅴ久 | 国产精品午夜8888 | 五月婷婷久久丁香 | 狠狠色丁香婷婷综合久小说久 | zzijzzij亚洲日本少妇熟睡 | 欧美成人aa | 综合国产在线观看 | 综合久久精品 | 欧美激情精品久久久久久免费印度 | www视频在线播放 | 99视频国产精品免费观看 | 成年人视频在线免费观看 | 久久99亚洲精品 | www视频在线免费观看 | 中文字幕在线观看完整版 | 天天操天天玩 | 国产破处视频在线播放 | 国产精品手机在线 | 成人a在线观看高清电影 | 欧美淫视频 | 91精品亚洲影视在线观看 | 成人免费视频观看 | 亚洲在线高清 | 特级a毛片 | 久久婷婷色综合 | 国产视频一 | 亚洲精品乱码久久久久久 | 中文字幕 国产精品 | 99r国产精品 | 欧美黑人巨大xxxxx | 亚洲人成在线电影 | 国产一区二区在线观看视频 | 国产精品久久久久久久av电影 | 超碰97中文 | 天无日天天操天天干 | 天天爱天天 | www.av小说| 亚洲黄色精品 | 欧美日韩二区在线 | 国产精品免费久久 | 亚洲经典中文字幕 | 97人人超碰在线 | 99国产精品久久久久久久久久 | 国产精品免费一区二区 | 国产黄色av| 国产美女网站视频 | 亚洲精品91天天久久人人 | 国产日韩欧美网站 | av电影免费观看 | 日韩av免费观看网站 | 欧美夫妻性生活电影 | 国产一级h | 国产精品18videosex性欧美 | 欧美日韩视频观看 | 337p欧美| 一区二区三区在线视频111 | 婷婷午夜激情 | 国产精品刺激对白麻豆99 | 国产精品igao视频网入口 | 国产精品网红福利 | 操久 | 久久久国产精品久久久 | 亚洲国产精品推荐 | 中文字幕av最新 | 在线观看av中文字幕 | 久草免费电影 | 免费黄色在线播放 | 色婷婷狠狠 | 国产中文字幕大全 | 三级黄色在线观看 | 国产精品成人自产拍在线观看 | 六月色 | 久热免费在线观看 | 亚洲精品国偷拍自产在线观看蜜桃 | 欧美日韩免费一区二区 | 国产精品久久久久久久久久久免费 | 久久,天天综合 | 中文字幕超清在线免费 | 日韩亚洲在线视频 | 天天综合导航 | 夜夜爽88888免费视频4848 | 狠狠操狠狠干2017 | 国产一二区免费视频 | 国产视频一区二区在线观看 | 免费视频a | 久久免费视频99 | 一区二区三区三区在线 | 国产无套视频 | 日日操天天爽 | 日韩在线视频网 | 久久69精品久久久久久久电影好 | 亚洲精品美女视频 | 在线观看亚洲a | 香蕉视频久久久 | 国产精品久久久久久电影 | 国产不卡在线看 | 夜色.com | 国产色区 | 99热都是精品 | 在线播放日韩av | 久久久久久久久久久影视 | 久草综合在线观看 | 久久经典国产 | 久久综合九色综合久久久精品综合 | 国产精品乱码久久久 | 国产成人一区二区精品非洲 | 色香蕉视频 | 日韩欧美第二页 | 国产精品久久一 | 久久免费视频在线观看30 | 欧美日韩1区2区 | 在线观看日本高清mv视频 | 国产成人精品av | 国产亚洲成人网 | 国产精品亚洲综合久久 | 99久久超碰中文字幕伊人 | 国内精品久久久久久久久久 | 国产在线不卡精品 | 婷婷中文字幕在线观看 | 一区二区三区四区五区六区 | 伊人婷婷| 国产精品不卡一区 | 玖玖玖在线观看 | 亚洲精品高清视频 | 午夜精品成人一区二区三区 | 日韩影视大全 | 欧美精品在线观看免费 | 国产免费叼嘿网站免费 | 狠狠干美女 | 久久久精品国产免费观看一区二区 | 91精品国产乱码在线观看 | 亚洲欧美成人网 | 日韩成人xxxx | 精品一区在线 | 久久精品国产成人精品 | 国产精品欧美久久久久天天影视 | 亚洲成 人精品 | 嫩嫩影院理论片 | 亚洲涩涩网 | 国产精品一区二区电影 | 在线导航av | 五月婷婷av在线 | 99精品在线免费视频 | 久久精品免费 | 国产成人免费 | 国产视频91在线 | 黄色小说视频网站 | 亚洲人成人在线 | 九九久久国产精品 | 久久久国产一区二区三区四区小说 | 婷婷中文在线 | 中文av在线免费观看 | 97在线免费视频观看 | 国产精品免费大片视频 | 久久久久久久久久久免费 | 国产一区网 | 中文成人字幕 | 五月激情天 | 韩国av免费观看 | 特级西西444www大胆高清无视频 | 日韩特级黄色片 | 欧美一区二区三区免费观看 | 亚洲精品国产自产拍在线观看 | 欧美片一区二区三区 | 国产色就色 | 免费在线观看av网址 | 国产精品高清一区二区三区 | 日日精品| 久久综合亚洲鲁鲁五月久久 | 成人黄色中文字幕 | 日韩一二三 | 中文字幕在线不卡国产视频 | 成人欧美一区二区三区黑人麻豆 | 97人人澡人人爽人人模亚洲 | 黄色在线观看免费网站 | 美女免费av| 久久99热精品这里久久精品 | 一区二区三区 亚洲 | 亚洲一级黄色 | 亚洲成人精品久久久 | 不卡视频在线 | 亚洲国产中文字幕在线 | 久久精品久久精品久久 | 麻豆视频在线 | 亚洲电影一级黄 | 国产在线超碰 | 91九色蝌蚪视频网站 | 91人人插 | 中文字幕视频一区二区 | 亚洲国产mv | 国产精品免费在线播放 | 美腿丝袜av| 91九色成人 | 日韩在线视频免费观看 | 日韩精品视频免费在线观看 | 69久久99精品久久久久婷婷 | 亚洲欧美日本一区二区三区 | 久久国产网 | 狠狠干天天色 | 欧美一二三区播放 | 日韩在线观看一区二区 | 456免费视频 | 久久爱992xxoo| 国产精品白浆 | 国产精品6| 国产精品久久久久永久免费观看 | 亚洲 欧美 综合 在线 精品 | 国产视频91在线 | 网址你懂的在线观看 | 国产精品一区二区三区99 | 波多野结衣电影一区二区三区 | 国产一级一片免费播放放a 一区二区三区国产欧美 | 欧美一区二视频在线免费观看 | 成人午夜剧场在线观看 | 91成人短视频在线观看 | 在线观看黄色大片 | 丁香婷婷深情五月亚洲 | 在线观看国产www | 日韩高清观看 | 一级免费黄视频 | 人人干97 | 最新精品视频在线 | 国产精品v欧美精品v日韩 | 粉嫩av一区二区三区入口 | 五月天综合婷婷 | 国产资源在线免费观看 | 国产精品久久久久婷婷 | 毛片二区 | 国产精品久久久久久久久搜平片 | 天天做天天射 | 亚洲日韩中文字幕在线播放 | 国产日韩欧美中文 | 国产第一页福利影院 | 五月开心综合 | 在线一区电影 | 波多野结衣在线观看视频 | 日日操狠狠干 | 中国一级片在线观看 | 91在线看黄 | 又黄又爽又无遮挡免费的网站 | 日韩在线免费小视频 | 日本不卡123区 | 亚洲国产无 | 亚洲va欧美va| 成人a级网站| 精品在线观看视频 | 婷婷丁香九月 | 九色91在线视频 | 992tv在线成人免费观看 | 国产资源在线播放 | 国产一区国产精品 | 免费看成人av | 免费人成在线观看 | 国产三级av在线 | 国产日韩亚洲 | 超碰激情在线 | 91香蕉亚洲精品 | 色99之美女主播在线视频 | 91免费在线播放 | 亚洲日本va中文字幕 | 久久久久久久久久久成人 | 性色av一区二区三区在线观看 | 久久精品久久99精品久久 | 中文字幕高清av | 91在线精品秘密一区二区 | 伊色综合久久之综合久久 | 丝袜美腿一区 | 999久久久久| 国产不卡在线观看视频 | 久久草草热国产精品直播 | 久久免费a| 婷婷精品视频 | a亚洲视频 | 国产成人三级在线 | 成年人看片网站 | 日韩欧美在线中文字幕 | 久草视频免费观 | 91视频在线免费观看 | 久久精品国产99国产 | 一级免费观看 | 免费av试看 | 狠狠做六月爱婷婷综合aⅴ 日本高清免费中文字幕 | 中文日韩在线视频 | 中文字幕乱在线伦视频中文字幕乱码在线 | 狠狠干夜夜爽 | 九色91福利 | 成人性生交大片免费看中文网站 | 天天操狠狠操 | 国产精品久久久免费看 | 你操综合 | 97精品国产91久久久久久久 | 国产美女永久免费 | 欧美精品天堂 | 五月婷婷.com | 亚洲黄色小说网 | 99精品网站 | 久久久久网站 | 久久国产精品系列 | 81国产精品久久久久久久久久 | 亚洲精品日韩av | 波多野结衣综合网 | 亚洲日本一区二区在线 | 久久久综合电影 | 欧美精品首页 | 日本女人在线观看 | 99精品在线直播 | 激情综合网在线观看 | 欧美极品少妇xbxb性爽爽视频 | 999亚洲国产996395 | 久久久久成人精品 | 中文字幕久久网 | 亚洲一区在线看 | 久久久黄色免费网站 | 成人黄色国产 | www.亚洲精品在线 | 日女人免费视频 | 国产96在线观看 | 日日天天av| 91精品夜夜| 97国产超碰| 欧美性超爽 | 久久99精品国产99久久 | 99精品国产在热久久 | 97国产在线观看 | 欧美激情在线网站 | 欧美成人黄色 | 国产又黄又爽无遮挡 | 精品久久久久久久久久久久久久久久 | 91人人爽久久涩噜噜噜 | 97超碰资源| 久久激情五月丁香伊人 | 成年人黄色免费网站 | 中文字幕日韩伦理 | 欧美一级视频在线观看 | 中日韩男男gay无套 日韩精品一区二区三区高清免费 | 在线看av网址 | 亚洲最新av在线网址 | 国产丝袜美腿在线 | 久久久福利视频 | 国产黄色大片免费看 | 韩国av一区二区 | 一本一本久久a久久 | 日韩免费电影 | 国产精品久久久久av福利动漫 | 在线观看网站黄 | 精品a级片| 久草热久草视频 | 欧美先锋影音 | 在线看国产视频 | 国产视频1 | 久久视屏网| 久久综合99| 国产剧情久久 | 午夜美女网站 | 久草在线一免费新视频 | 国产 日韩 在线 亚洲 字幕 中文 | 国产免费成人 | 欧美色一色| 国产精品久久久久永久免费 | 91精品一区二区三区蜜桃 | 日韩在线视 | 黄色高清视频在线观看 | 国产又粗又长又硬免费视频 | 欧美日韩综合在线观看 | 人人爱夜夜操 | 91av在线免费观看 | 日韩不卡高清 | 激情综合色综合久久综合 | 成人午夜剧场在线观看 | 免费麻豆网站 | 久久三级毛片 | 久久手机看片 | www国产亚洲精品 | 久久夜av | 国产黄色在线观看 | 中文字幕免费一区二区 | 99久久久成人国产精品 | 韩国精品一区二区三区六区色诱 | 国产黄| 狠狠狠的干 | 亚洲在线日韩 | 欧美aaa级片 | 97精品国自产拍在线观看 | 麻豆传媒视频在线免费观看 | 国产手机免费视频 | av成人黄色 | 国产一级在线视频 | 亚洲视频在线观看 | 黄色网址国产 | 色视频成人在线观看免 | 伊人婷婷久久 | 偷拍精偷拍精品欧洲亚洲网站 | 国产精品99久久99久久久二8 | 国产美女视频网站 | 99视频国产精品免费观看 | 久久一区二区三区国产精品 | 亚洲91av| 成人av教育| 中文字幕在线资源 | 国产成人精品一区在线 | av资源免费在线观看 | 天天操天天干天天玩 | 日韩一级精品 | 国产一区二区不卡视频 | 91久久国产露脸精品国产闺蜜 | 黄色tv视频 | 日韩剧情 | 成人国产电影在线观看 | 丁香花五月 | 成 人 a v天堂 | 久久高清视频免费 | 粉嫩av一区二区三区四区在线观看 | 97在线视 | 国产精品欧美在线 | 欧美成人h版在线观看 | 91av视频播放 | 久草综合视频 | 色婷婷av国产精品 | 日日碰夜夜爽 | 欧美一区在线观看视频 | 国产在线超碰 | 精品国产一区二区三区四 | 国产成人久久av免费高清密臂 | 国产在线最新 | 久艹在线免费观看 | 天天射天天色天天干 | 五月综合色| 中文字幕久久亚洲 | 在线观看不卡视频 | 五月婷婷六月丁香 | 久久狠狠一本精品综合网 | 久草精品在线 | 国产小视频在线免费观看 | av在线播放快速免费阴 | 国产麻豆精品久久 | 亚洲一级片在线观看 | ww亚洲ww亚在线观看 | 超碰午夜| 久久国产精品网站 | 日韩精选在线观看 | 国产精品国产三级国产aⅴ9色 | 夜夜操天天干, | 黄色成人毛片 | 国产一级h | 精品av网站| 欧美亚洲精品在线观看 | 日韩综合精品 | 欧美男同视频网站 | 干天天| 中文字幕日韩高清 | 久久久久久毛片精品免费不卡 | 中文字幕网站视频在线 | 波多野结衣一区二区三区中文字幕 | 日韩精品资源 | 91色综合| 国产精品久久久久久一二三四五 | 色五月色开心色婷婷色丁香 | 99视频在线精品免费观看2 | 精品中文字幕在线观看 | 免费高清在线观看电视网站 | 国产真实精品久久二三区 | 国产精品18久久久久久久久 | av在线播放网址 | 中文字幕有码在线 | 国产成人精品久久久久蜜臀 | 欧美激情综合五月色丁香小说 | 在线三级av | 99在线观看精品 | 久久伊人精品一区二区三区 | 国产精品精品国产婷婷这里av | 久久久高清一区二区三区 | 久久久久日本精品一区二区三区 | 久久国产美女视频 | 国产精品免费观看视频 | 夜夜澡人模人人添人人看 | 91一区二区三区久久久久国产乱 | 久久草草热国产精品直播 | 亚洲97在线 | 欧美激情精品久久久久久变态 | 日本三级大片 | 狠狠色丁香婷婷综合久久片 | 九九九九九九精品任你躁 | 国产精品久久久久久久久久久久 | 国产男女无遮挡猛进猛出在线观看 | 有码一区二区三区 | 操碰av | 久久国产精品一区二区三区四区 | 九九视频精品免费 | 狠狠躁日日躁狂躁夜夜躁 | av在线免费播放网站 | av片中文| 久久综合九色综合欧美狠狠 | 成人久久免费视频 | 天天综合网久久综合网 | 热re99久久精品国产66热 | 国产九九在线 | 婷婷国产v亚洲v欧美久久 | 一区 二区 精品 | 国产精品毛片一区视频播 | 久久电影日韩 | 久久精品视频99 | 碰超在线观看 | 欧美精品九九 | 成人免费观看在线视频 | 久久久久亚洲天堂 | 日韩欧美综合视频 | 亚洲免费精品视频 | 日韩免费视频播放 | 91亚洲夫妻 | 手机av在线网站 | 中文字幕资源在线观看 | 欧美一区二区日韩一区二区 | 操操操干干干 | 久久免费黄色网址 | 人人插人人做 | 黄色免费观看 | 亚洲精品视频免费看 | 欧美日韩aa | 五月激情在线 | 夜夜躁日日躁狠狠久久av | 999成人精品 | 亚洲精品国产精品99久久 | 久久精品国产久精国产 | 日本中文字幕免费观看 | 中文字幕国产视频 | 69视频在线播放 | 国产免费a | 亚洲午夜精品久久久久久久久 | 亚州人成在线播放 | 欧美片一区二区三区 | 国产精品成人久久 | 91亚洲狠狠婷婷综合久久久 | 国产精品一区二区无线 | 欧美色图东方 | 日本三级不卡 | 欧美一级日韩免费不卡 | 亚洲在线网址 | 特级a老妇做爰全过程 | 涩涩网站在线观看 | 在线欧美最极品的av | 欧美日产在线观看 | 91亚洲精品久久久中文字幕 | 青青河边草免费 | 日本精品视频在线 | 97电影在线| 色99之美女主播在线视频 | 亚洲伦理精品 | 久久好看| 最新中文在线视频 | 日本精品视频在线播放 | 97色狠狠| 天天激情综合网 | 欧美人体xx | 久草在线中文888 | 亚洲高清在线视频 | 国产不卡精品视频 | 国产原厂视频在线观看 | 99视频精品全部免费 在线 | 久久久免费高清视频 | 超碰日韩在线 | 欧美 激情 国产 91 在线 | 久草精品视频 | 黄色一级免费 | 久久成人国产精品免费软件 | av.com在线| 天天操夜操 | 四虎国产精品永久在线国在线 | 五月天激情视频在线观看 | 亚洲精品影视在线观看 | 国产精成人品免费观看 | 麻豆国产视频下载 | 一区二区三区久久 | 国产大陆亚洲精品国产 | 亚洲国产久 | 中文字幕在线视频一区二区 | 日韩免费福利 | 久草在线视频精品 | 美女视频黄网站 | 日韩精品一卡 | 久章草在线 | 国产主播大尺度精品福利免费 | 欧美成人黄色 | av一区二区三区在线播放 | 天天综合天天做天天综合 | 国产成人三级在线 | 丁香激情五月婷婷 | 韩国av免费在线观看 | 超碰人人草人人 | 国产精品一区二区三区在线播放 | 18+视频网站链接 | 国产99一区视频免费 | 福利视频入口 | 91精品在线免费 | 免费在线观看毛片网站 | 国产婷婷vvvv激情久 | 成人免费观看网站 | 免费在线播放av电影 | 亚洲精品中文在线 | 久久久久久免费毛片精品 | 久久精品99久久久久久2456 | 在线a人片免费观看视频 | 蜜臀av在线一区二区三区 | 天天操夜夜拍 | 人人草在线视频 | 日韩在线精品一区 | 亚洲色图av | 精品亚洲免费 | 久久伦理影院 | 伊人影院av | 国内精品在线一区 | 91高清免费| 久久久久久久亚洲精品 | 99免费视频 | 久久一二区 | 91麻豆精品国产91久久久久久久久 | 免费av大全| 国产免费一区二区三区最新6 | 黄色字幕网| 九九99视频 | 99久久精品午夜一区二区小说 | 国产18精品乱码免费看 | 色婷婷a | 天天操天天摸天天干 | 2023国产精品自产拍在线观看 | 精品一区中文字幕 | 夜夜操网站 | 人人dvd | 999抗病毒口服液 | 日本激情视频中文字幕 | 成人免费观看网站 | 午夜精品久久久久久久99 | 天天射网 | 日韩av视屏在线观看 | 久久成年人网站 | 日本精品视频免费 | 人人草在线视频 | 美女黄频免费 | 天天爽天天爽 | 美女网色| 91九色国产在线 | 狠狠干.com | 国产69久久久欧美一级 | 免费av在线播放 | 国产精品私人影院 | 丁香婷婷激情网 | 欧美污污网站 | 亚洲色影爱久久精品 | a级国产乱理论片在线观看 特级毛片在线观看 | 欧美日韩在线免费视频 | 超碰人人草 | 亚洲精品在线观看的 | 欧美国产高清 | 视频一区二区视频 | 中文字幕一区二区在线播放 | 999电影免费在线观看2020 | 久久久久婷 |