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

首頁 > 開發(fā) > Java > 正文

Executor攔截器高級(jí)教程QueryInterceptor的規(guī)范

2024-07-14 08:43:22
字體:
供稿:網(wǎng)友

Executor 攔截器高級(jí)教程 - QueryInterceptor 規(guī)范

這篇文檔涉及下面幾個(gè)方面

  • 1. Executor query 方法介紹
  • 2. 攔截器配置和調(diào)用順序
  • 3. 攔截 query 方法的技巧
  • 4. 攔截 query 方法的規(guī)范
  • 5. 如何配置不同的 Executor 插件

1. Executor query 方法介紹

在 MyBatis 的攔截器的文檔部分,我們知道 Executor 中的 query 方法可以被攔截,如果你真正寫過這個(gè)方法的攔截器,你可能會(huì)知道在 Executor 中的 query 方法有兩個(gè):

<E> List<E> query(   MappedStatement ms,    Object parameter,    RowBounds rowBounds,    ResultHandler resultHandler,    CacheKey cacheKey,    BoundSql boundSql) throws SQLException;<E> List<E> query(   MappedStatement ms,    Object parameter,    RowBounds rowBounds,    ResultHandler resultHandler) throws SQLException;

這兩個(gè)方法的區(qū)別是第一個(gè)方法多兩個(gè)參數(shù) CacheKey 和 BoundSql,在多數(shù)情況下,我們用攔截器的目的就是針對(duì) SQL 做處理,如果能夠攔截第一個(gè)方法,可以直接得到 BoundSql 對(duì)象,就會(huì)很容易的得到執(zhí)行的 SQL,也可以對(duì) SQL 做處理。

雖然想的很好,但是 MyBatis 提供的 Exctutor 實(shí)現(xiàn)中,參數(shù)多的這個(gè) query 方法都是被少的這個(gè) query 方法在內(nèi)部進(jìn)行調(diào)用的。

CachingExecutor中:

public <E> List<E> query(    MappedStatement ms,     Object parameter,     RowBounds rowBounds,     ResultHandler resultHandler) throws SQLException {  BoundSql boundSql = ms.getBoundSql(parameterObject);  CacheKey key = createCacheKey(ms, parameterObject, rowBounds, boundSql);  return query(ms, parameterObject, rowBounds, resultHandler, key, boundSql);}

BaseExecutor中:

public <E> List<E> query(    MappedStatement ms,     Object parameter,     RowBounds rowBounds,     ResultHandler resultHandler) throws SQLException {  BoundSql boundSql = ms.getBoundSql(parameter);  CacheKey key = createCacheKey(ms, parameter, rowBounds, boundSql);  return query(ms, parameter, rowBounds, resultHandler, key, boundSql);}

上面這兩個(gè)方法一樣。由于第一個(gè) query 方法在這里是內(nèi)部調(diào)用,并且我們所有的攔截器都是層層代理的CachingExecutor或基于BaseExecutor的實(shí)現(xiàn)類,所以我們能攔截的就是參數(shù)少的這個(gè)方法。

分頁插件開始從Executor攔截開始就一直是攔截的參數(shù)少的這個(gè)方法。但是從5.0 版本開始,query 的這兩個(gè)方法都可以被攔截了。在講這個(gè)原理之前,我們先了解一下攔截器的執(zhí)行順序。

2. 攔截器配置和調(diào)用順序

攔截器的調(diào)用順序分為兩大種,第一種是攔截的不同對(duì)象,例如攔截 Executor 和 攔截 StatementHandler 就屬于不同的攔截對(duì)象,這兩類的攔截器在整體執(zhí)行的邏輯上是不同的,在 Executor 中的 query 方法執(zhí)行過程中,會(huì)調(diào)用下面的代碼:

public <E> List<E> doQuery(    MappedStatement ms,     Object parameter,     RowBounds rowBounds,     ResultHandler resultHandler,     BoundSql boundSql) throws SQLException {  Statement stmt = null;  try {     Configuration configuration = ms.getConfiguration();     StatementHandler handler = configuration.newStatementHandler(wrapper, ms, parameter, rowBounds, resultHandler, boundSql);     stmt = prepareStatement(handler, ms.getStatementLog());     return handler.<E>query(stmt, resultHandler);  } finally {    closeStatement(stmt);  }}

在這段代碼中,才會(huì)輪到 StatementHandler 去執(zhí)行,StatementHandler 屬于 Executor 執(zhí)行過程中的一個(gè)子過程。所以這兩種不同類別的插件在配置時(shí),一定是先執(zhí)行 Executor 的攔截器,然后才會(huì)輪到 StatementHandler。所以這種情況下配置攔截器的順序就不重要了,在 MyBatis 邏輯上就已經(jīng)控制了先后順序。

第二種攔截器的順序就是指攔截同一種對(duì)象的同一個(gè)方法,例如都攔截 Executor 的 query 方法,這時(shí)你配置攔截器的順序就會(huì)對(duì)這里有影響了。假設(shè)有如下幾個(gè)攔截器,都是攔截的 Executor 的 query 方法。

<plugins>  <plugin interceptor="com.github.pagehelper.ExecutorQueryInterceptor1"/>  <plugin interceptor="com.github.pagehelper.ExecutorQueryInterceptor2"/>  <plugin interceptor="com.github.pagehelper.ExecutorQueryInterceptor3"/></plugins>

org.apache.ibatis.session.Configuration中有如下方法:

public void addInterceptor(Interceptor interceptor) {  interceptorChain.addInterceptor(interceptor);}

MyBatis 會(huì)按照攔截器配置的順序依次添加到interceptorChain中,其內(nèi)部就是List<Interceptor> interceptors。再看 Configuration中創(chuàng)建 Executor 的代碼:

public Executor newExecutor(Transaction transaction, ExecutorType executorType) {  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);  }  executor = (Executor) interceptorChain.pluginAll(executor);  return executor;}

在調(diào)用 interceptorChain.pluginAll 之前,executor 就是前一節(jié)中的 CachingExecutor 或基于 BaseExecutor 的實(shí)現(xiàn)類。然后看 interceptorChain.pluginAll 方法:

public Object pluginAll(Object target) {  for (Interceptor interceptor : interceptors) {    target = interceptor.plugin(target);  }  return target;}

前面我們配置攔截器的順序是1,2,3。在這里也會(huì)按照 1,2,3 的順序被層層代理,代理后的結(jié)構(gòu)如下:

Interceptor3:{  Interceptor2: {    Interceptor1: {      target: Executor    }  }}

從這個(gè)結(jié)構(gòu)應(yīng)該就很容易能看出來,將來執(zhí)行的時(shí)候肯定是按照 3>2>1>Executor>1>2>3 的順序去執(zhí)行的??赡苡行┤瞬恢罏槭裁?>2>1>Executor之后會(huì)有1>2>3,這是因?yàn)槭褂么頃r(shí),調(diào)用完代理方法后,還能繼續(xù)進(jìn)行其他處理。處理結(jié)束后,將代理方法的返回值繼續(xù)往外返回即可。例如:

Interceptor3 前置處理   Object result = Interceptor2..query(4個(gè)參數(shù)方法);   Interceptor3 后續(xù)處理  return result;

對(duì)于 Interceptor2.invoke 方法也是相同的邏輯:

Interceptor2 前置處理   Object result = Interceptor1..query(4個(gè)參數(shù)方法);   Interceptor2 后續(xù)處理  return result;

同理 Interceptor1.invoke :

Interceptor1 前置處理   Object result = executor.query(4個(gè)參數(shù)方法);   Interceptor1 后續(xù)處理  return result;

疊加到一起后,如下:

Interceptor3 前置處理Interceptor2 前置處理Interceptor1 前置處理 Object result = executor.query(4個(gè)參數(shù)方法);   Interceptor1 后續(xù)處理  Interceptor2 后續(xù)處理 Interceptor3 后續(xù)處理  return result;

所以這個(gè)順序就是 3>2>1>Executor>1>2>3。

在你弄清楚這個(gè)邏輯后,再繼續(xù)往下看,因?yàn)楹竺娴募记蓵?huì)顛覆這個(gè)邏輯,所以才會(huì)有后面的規(guī)范以及如何配置不同的插件。

3. 攔截 query 方法的技巧

上一節(jié)的內(nèi)容中,對(duì)攔截器的用法是最常見的一種用法,所以才會(huì)出現(xiàn)這種都能理解的執(zhí)行順序。但是分頁插件 5.0 不是這樣,這個(gè)插件顛覆了這種順序,這種顛覆其實(shí)也很普通,這也是本節(jié)要說的技巧。

在我寫作 MyBatis 技術(shù)書籍的過程中(還沒寫完,已經(jīng)因?yàn)榉猪摬寮加昧藥字艿膶懽鲿r(shí)間),我就在考慮為什么不能攔截第一個(gè)query(6個(gè)參數(shù)的)方法,如果能攔截這個(gè)方法,就可以直接拿到 BoundSql,然后處理 SQL 就很容易實(shí)現(xiàn)其他的操作。

在第1 節(jié)介紹為什么第一個(gè)query方法不能被攔截時(shí),是因?yàn)橄旅孢@段代碼:

public <E> List<E> query(    MappedStatement ms,     Object parameter,     RowBounds rowBounds,     ResultHandler resultHandler) throws SQLException {  BoundSql boundSql = ms.getBoundSql(parameter);  CacheKey key = createCacheKey(ms, parameter, rowBounds, boundSql);  return query(ms, parameter, rowBounds, resultHandler, key, boundSql);}

既然CachingExecutor或基于BaseExecutor的實(shí)現(xiàn)類只是這么簡單的調(diào)用兩個(gè)方法得到了BoundSql 和Cachekey,我們?yōu)槭裁床恢苯犹娲麄兡兀?/p>

所以我們可以有類似下面的攔截器用法:

@Intercepts(@Signature(type = Executor.class, method = "query", args = {MappedStatement.class, Object.class, RowBounds.class, ResultHandler.class}))public class QueryInterceptor implements Interceptor {  @Override  public Object intercept(Invocation invocation) throws Throwable {    Object[] args = invocation.getArgs();    MappedStatement ms = (MappedStatement) args[0];    Object parameterObject = args[1];    RowBounds rowBounds = (RowBounds) args[2];    ResultHandler resultHandler = (ResultHandler) args[3];    Executor executor = (Executor) invocation.getTarget();    BoundSql boundSql = ms.getBoundSql(parameterObject);    //可以對(duì)參數(shù)做各種處理    CacheKey cacheKey = executor.createCacheKey(ms, parameterObject, rowBounds, boundSql);    return executor.query(ms, parameterObject, rowBounds, resultHandler, cacheKey, boundSql);  }  @Override  public Object plugin(Object target) {    return Plugin.wrap(target, this);  }  @Override  public void setProperties(Properties properties) {  }}

這個(gè)攔截器直接替代了原有 Executor 的部分邏輯,直接去調(diào)用了 6 個(gè)參數(shù)的方法,因而導(dǎo)致 4 個(gè)參數(shù)的后續(xù)方法被跳過了。但是由于這里的 executor 是代理對(duì)象,所以 6 個(gè)參數(shù)的 query 方法可以被代理了,這就擾亂了上一節(jié)中的執(zhí)行順序。

在上一節(jié)攔截器的例子中,做簡單修改,將 ExecutorQueryInterceptor2 換成上面的 QueryInterceptor,配置如下:

<plugins>  <plugin interceptor="com.github.pagehelper.ExecutorQueryInterceptor1"/>  <plugin interceptor="com.github.pagehelper.QueryInterceptor"/>  <plugin interceptor="com.github.pagehelper.ExecutorQueryInterceptor3"/></plugins>

代理后的結(jié)構(gòu)如下:

Interceptor3:{  QueryInterceptor: {    Interceptor1: {      target: Executor    }  }}

這時(shí),調(diào)用順序就變了,Interceptor3 執(zhí)行順序如下:

Interceptor3 前置處理   Object result = QueryInterceptor.query(4個(gè)參數(shù)方法);   Interceptor3 后續(xù)處理  return result;

QueryInterceptor.invoke 執(zhí)行邏輯如下:

Interceptor2 前置處理   Object result = executor.query(6個(gè)參數(shù)方法);   Interceptor2 后續(xù)處理  return result;

在 QueryInterceptor 中,沒有繼續(xù)執(zhí)行 4個(gè)參數(shù)方法,而是執(zhí)行了 6 個(gè)參數(shù)方法。但是 Interceptor1 攔截的 4 個(gè)參數(shù)的方法,所以 Interceptor1 就被跳過去了,整體的執(zhí)行邏輯就變成下面這樣了:

Interceptor3 前置處理Interceptor2 前置處理Object result = executor.query(6個(gè)參數(shù)方法);   Interceptor2 后續(xù)處理 Interceptor3 后續(xù)處理  return result;

如果 Interceptor1 攔截的是 6 個(gè)參數(shù)的方法,因?yàn)?QueryInterceptor 獲取的是 Interceptor1 代理的 executor 對(duì)象,那么 Interceptor1 就會(huì)被 QueryInterceptor 繼續(xù)執(zhí)行下去。

分頁插件就是類似 QueryInterceptor 的執(zhí)行邏輯,所以當(dāng)你使用 5.0 版本之后的插件時(shí),如果你還需要配置其他 Executor 的 query 插件,你就會(huì)遇到一些問題(可以解決,繼續(xù)往下看)。

如果你是自己開發(fā)的插件,那么你按照下一節(jié)的規(guī)范去開發(fā)也不會(huì)遇到問題。如果你使用的其他人提供的插件,按照第 5 節(jié)的配置順序也能解決問題。

4. 攔截 query 方法的規(guī)范

QueryInterceptor 的邏輯就是進(jìn)去的是 4 個(gè)參數(shù)的方法,出去的是 6 個(gè)參數(shù)的方法。這種處理方法不僅僅不方便和一般的 Excutor 攔截器搭配使用,當(dāng)出現(xiàn)兩個(gè)以上類似 QueryInterceptor 的插件時(shí),由于接口變了,類似 QueryInterceptor 插件也無法連貫的執(zhí)行下去。因而有必要解決這個(gè)問題。解決的辦法就是使用統(tǒng)一的規(guī)范。經(jīng)過規(guī)范后 QueryInterceptor 如下:

@Intercepts(  {    @Signature(type = Executor.class, method = "query", args = {MappedStatement.class, Object.class, RowBounds.class, ResultHandler.class}),    @Signature(type = Executor.class, method = "query", args = {MappedStatement.class, Object.class, RowBounds.class, ResultHandler.class, CacheKey.class, BoundSql.class}),  })public class QueryInterceptor implements Interceptor {  @Override  public Object intercept(Invocation invocation) throws Throwable {    Object[] args = invocation.getArgs();    MappedStatement ms = (MappedStatement) args[0];    Object parameterObject = args[1];    RowBounds rowBounds = (RowBounds) args[2];    ResultHandler resultHandler = (ResultHandler) args[3];    Executor executor = (Executor) invocation.getTarget();    CacheKey cacheKey;    BoundSql boundSql;    //由于邏輯關(guān)系,只會(huì)進(jìn)入一次    if(args.length == 4){      //4 個(gè)參數(shù)時(shí)      boundSql = ms.getBoundSql(parameterObject);      cacheKey = executor.createCacheKey(ms, parameterObject, rowBounds, boundSql);    } else {      //6 個(gè)參數(shù)時(shí)      cacheKey = (CacheKey) args[4];      boundSql = (BoundSql) args[5];    }    //TODO 自己要進(jìn)行的各種處理    //注:下面的方法可以根據(jù)自己的邏輯調(diào)用多次,在分頁插件中,count 和 page 各調(diào)用了一次    return executor.query(ms, parameterObject, rowBounds, resultHandler, cacheKey, boundSql);  }  @Override  public Object plugin(Object target) {    return Plugin.wrap(target, this);  }  @Override  public void setProperties(Properties properties) {  }}

注意兩個(gè)變化,第一個(gè)就是攔截器簽名同時(shí)攔截了 4 個(gè) 和 6 個(gè)參數(shù)的方法,這樣不管那個(gè)插件在前在后都會(huì)被執(zhí)行。

第二個(gè)變化就是這段代碼:

CacheKey cacheKey;BoundSql boundSql;//由于邏輯關(guān)系,只會(huì)進(jìn)入一次if(args.length == 4){  //4 個(gè)參數(shù)時(shí)  boundSql = ms.getBoundSql(parameterObject);  cacheKey = executor.createCacheKey(ms, parameterObject, rowBounds, boundSql);} else {  //6 個(gè)參數(shù)時(shí)  cacheKey = (CacheKey) args[4];  boundSql = (BoundSql) args[5];}

如果這個(gè)插件配置的靠后,是通過 4 個(gè)參數(shù)方法進(jìn)來的,我們就獲取這兩個(gè)對(duì)象。如果這個(gè)插件配置的靠前,已經(jīng)被別的攔截器處理成 6 個(gè)參數(shù)的方法了,那么我們直接從 args 中取出這兩個(gè)參數(shù)直接使用即可。取出這兩個(gè)參數(shù)就保證了當(dāng)其他攔截器對(duì)這兩個(gè)參數(shù)做過處理時(shí),這兩個(gè)參數(shù)在這里會(huì)繼續(xù)生效。

假設(shè)有個(gè)排序插件和分頁插件,排序插件將 BoundSql 修改為帶排序的 SQL 后,SQL 會(huì)繼續(xù)交給分頁插件使用。分頁插件的分頁 SQL 執(zhí)行時(shí),會(huì)保留排序去執(zhí)行,這樣的規(guī)范就保證了兩個(gè)插件都能正常的執(zhí)行下去。

所以如果大家想要使用這種方式去實(shí)現(xiàn)攔截器,建議大家遵守這個(gè)規(guī)范。

這個(gè)規(guī)范對(duì)于已經(jīng)存在的插件來說就沒法控制了,但是仍然可以通過配置順序來解決。

5. 如何配置不同的 Executor 插件

當(dāng)引入類似 QueryInterceptor 插件時(shí),由于擾亂了原有的插件執(zhí)行方式,當(dāng)配置 Executor 順序不對(duì)時(shí)會(huì)導(dǎo)致插件無法生效。

第 4 節(jié)中的例子:

<plugins>  <plugin interceptor="com.github.pagehelper.ExecutorQueryInterceptor1"/>  <plugin interceptor="com.github.pagehelper.QueryInterceptor"/>  <plugin interceptor="com.github.pagehelper.ExecutorQueryInterceptor3"/></plugins>

首先執(zhí)行順序?yàn)?3>Query>1>Executor,由于 Query 是 4 或 6 個(gè)參數(shù)進(jìn)來,6 個(gè)參數(shù)出去。所以在 Query 前面執(zhí)行的攔截器必須是 4 個(gè)的(Query 規(guī)范攔截器先后都能執(zhí)行,需要根據(jù)邏輯配置先后)參數(shù)的,在 Query 后面執(zhí)行的攔截器必須是 6 個(gè)參數(shù)的。

這個(gè)順序?qū)?yīng)到配置順序時(shí),也就是 4 個(gè)參數(shù)的配置在 QueryInterceptor 攔截器的下面,6 個(gè)參數(shù)的配置在 QueryInterceptor 攔截器的上面。按照這個(gè)順序進(jìn)行配置時(shí),就能保證攔截器都執(zhí)行。

如果你想獲得如分頁插件(QueryInterceptor 規(guī)范)執(zhí)行的 SQL,你就得按照 QueryInterceptor 規(guī)范去實(shí)現(xiàn),否則只能配置在分頁插件的下面,也就只能獲得分頁處理前的 SQL。

總結(jié)

以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對(duì)VeVb武林網(wǎng)的支持。


注:相關(guān)教程知識(shí)閱讀請(qǐng)移步到JAVA教程頻道。
發(fā)表評(píng)論 共有條評(píng)論
用戶名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
欧美性猛交xxxx免费看漫画| 中文字幕人妻一区二区三区视频| 美女激情网站| 亚洲精品影院在线观看| 欧美日韩美女在线观看| 青青草华人在线视频| eeuss影院一区二区三区| 国产精品丝袜在线播放| 在线观看黄色国产| 2019最新中文字幕| 欧美v国产在线一区二区三区| 亚洲图片激情小说| 成人免费视频国产免费观看| 欧产日产国产精品视频| 波兰性xxxxx极品hd| 黄频免费在线观看| 欧美日韩中文字幕视频| 激情婷婷久久| 国产色无码精品视频国产| 日韩av资源| 超碰97国产在线| 秋霞午夜鲁丝一区二区| 日韩一区欧美| 日本在线视频中文有码| 久久伊99综合婷婷久久伊| 久久99国产精品久久99小说| 国产精品videossex国产高清| 久久久久99精品成人片三人毛片| b站大片免费直播| 成人免费一区二区三区视频网站| 欧美一区二区三区高清视频| 国产福利一区二区在线精品| 欧美吞精做爰啪啪高潮| 国产精品三级av在线播放| 香蕉久久aⅴ一区二区三区| 精品国产18久久久久久二百| 川上优av中文字幕一区二区| 亚洲一区二区精品在线观看| 天堂在线观看免费视频| 熟妇人妻久久中文字幕| 先锋av在线资源| 亚洲精品偷拍视频| 校园春色另类视频| 青青草草视频| 日本中文字幕视频一区| 香港三级经典全部种子下载| 国产精品乱码一区二区三区| 亚洲免费伊人电影| 乱一区二区三区在线播放| 亚洲天堂网站在线观看视频| 91成人抖音| 黄色手机在线视频| 国产高潮国产高潮久久久91| 亚洲精品国产精品乱码在线观看| 九色91视频| 裸体丰满少妇做受久久99精品| 亚洲精品国产成人久久av盗摄| 2019中文字幕在线免费观看| 7777久久亚洲中文字幕| 激情综合网婷婷| 国产成人麻豆免费观看| 老司机深夜福利在线观看| caoporen人人| 国产精品剧情一区二区三区| 风间由美一二三区av片| 亚洲91视频| 国产高清欧美| 美女视频免费一区| 精品国产鲁一鲁一区二区三区| www国产免费| 精品一区在线播放| 精品产品国产在线不卡| 91在线观看免费高清完整版在线观看| 国产精品国产av| 国产综合动作在线观看| 久久久水蜜桃av免费网站| 日韩欧美大尺度| 国产毛片一区二区三区| 欧美亚洲精品天堂| 国产66精品久久久久999小说| 成人免费黄色网址| 日韩精品视频在线观看视频| 91av入口| 中文字幕日韩欧美在线视频| 美女网站在线观看| 污污视频在线免费观看| 黄色网免费观看| 中文字幕日本欧美| 中文字幕在线观看不卡| 亚洲 高清 成人 动漫| 国产精品久久二区| 精品久久久久久久久久久久包黑料| 日本女优北野望在线电影| 欧美丝袜在线观看| 国产精品毛片在线看| 亚洲熟妇av日韩熟妇在线| www.jizz在线观看| 国产亚洲精品久久久优势| 精品三级av在线| 欧美日韩成人在线播放| 国产成人在线观看免费网站| 国产精品无码无卡无需播放器| 在线播放一级片| 亚洲成色在线综合网站2018站| 国产激情视频一区二区| 亚洲不卡1卡2卡三卡2021麻豆| 成人精品gif动图一区| 国产成人综合欧美精品久久| 日韩深夜福利网站| 色播五月激情五月| 巨乳女教师的诱惑| 国产视频青青| jizz视频| 国产精品99久久99久久久| 色天天久久综合婷婷女18| 国产精品自在| 亚洲欧美丝袜| 香蕉久久一区二区不卡无毒影院| 欧美高清性xxxxhdvideosex| 丝袜足脚交91精品| 欧美一区二区三区性视频| 国产精品乱码一区二区三区视频| 欧美成人黑人猛交| 久久xxxx精品视频| 最新成人av网站| 中文字幕在线不卡| 三上悠亚在线观看| 欧美日韩国产色| 日韩欧美在线中字| 国产成人综合一区二区三区| 午夜不卡av在线| 亚洲一区二区在线观| 国产日韩在线不卡| 九九免费精品视频| 国产精品主播直播| 久久精品亚洲麻豆av一区二区| www欧美xxxx| 蜜臀av在线观看| 97激碰免费视频| 色综合视频一区二区三区44| 亚洲成人黄色网| 亚洲av首页在线| 亚洲欧美日韩天堂一区二区| 亚洲一区二区影院| 国产精品久久久久久久久电影网| 四季av日韩精品一区| 北岛玲日韩精品一区二区三区| 午夜视黄欧洲亚洲| 九七电影院97理论片久久tvb| 国产精品免费视频xxxx| 51成人做爰www免费看网站| 久久免费大视频| 成人免费一级视频| 91麻豆国产视频| 国产在线观看无码免费视频| 99久久免费国产精品| 91成人国产| 亚洲最大黄色| 一区二区三区欧美在线| 中文字幕自拍vr一区二区三区| 91影视免费在线观看| 美女久久久久久久久久| 日本久久久久亚洲中字幕| 朝桐光一区二区| 日韩电影在线播放| 狠狠久久亚洲欧美| 日韩区国产区| 日韩一级二级三级精品视频| 精品中文字幕在线观看| 激情五月深爱五月| 欧美日韩色视频| 亚洲欧美一级| 久久性感美女视频| 先锋影音av在线| 亚洲精品乱码久久久久久久久| 亚洲欧洲日韩av| 成人短视频app| 女教师淫辱の教室蜜臀av软件| 亚洲欧洲日本专区| 波多野结衣a v在线| 亚洲国产成人91porn| 日韩高清在线一区| 要久久电视剧全集免费| 亚洲欧美激情另类校园| 国产麻豆剧传媒精品国产av| 国产盗摄x88av| 欧美图片自拍偷拍| 国产男女猛烈无遮挡在线喷水| 影音先锋欧美在线| 三级影片在线观看欧美日韩一区二区| 992tv人人草| 亚洲网一区二区三区| 国产a级全部精品| 国产精品日韩电影| 国产丝袜控视频在线观看| 男女爱爱福利视频| 97久久网站| 精品国语对白精品自拍视| 在线国产一区| 欧美亚洲色图校园春色| 日韩**中文字幕毛片| 午夜国产精品理论片久久影院| 日本免费在线观看视频| 大胆日韩av| 久久国产精品久久精品国产| 色网址在线观看| 神马午夜dy888| 亚洲码无人客一区二区三区| 都市激情亚洲综合| 国产无遮挡又黄又爽| 国产999精品在线观看| 精品一区二区国语对白| 一起草最新网址| xf在线a精品一区二区视频网站| 国产黄a三级三级看三级| 欧美大尺度在线观看| 久本草在线中文字幕亚洲欧美| 久久久99国产精品免费| 日韩欧美在线视频播放| 中文字幕亚洲在线观看| 麻豆精品久久久久久久99蜜桃| 久久精品久久精品| 五福影院新址进入www1378| 永久免费看片直接| 国产婷婷一区二区三区| 麻豆网站在线观看| 亚洲欧美中日韩| 亚洲国产另类久久精品| 午夜影院免费| 日韩大陆毛片av| 中文字幕一区二区视频| 偷拍亚洲精品| 97免费在线视频| 国产高清精品软件丝瓜软件| 日本亚洲不卡| 免费看涩涩视频软件| 18岁成人毛片| 亚洲精品自拍偷拍| 99riav国产精品| 韩国三级一区| 免费观看h片| 黄色av片三级三级三级免费看| 两根大肉大捧一进一出好爽视频| 国产精品不卡一区二区三区在线观看| 欧美日韩在线免费视频| 国产天堂素人系列在线视频| 国产综合精品久久久久成人av| 男人插曲女人的视频| 电影在线观看一区二区| 一本一本久久a久久精品综合麻豆| 欧美一区二区在线看| 欧美一级国产精品| 91丨九色丨黑人外教| 国产精品欧美一级免费| 久久久久久免费精品| 91精品国产综合久久久久| 都市激情国产精品| 精品国产一区二区三区性色av| 国内精品视频在线| 色中文字幕在线| 18精品爽视频在线观看| 天堂中文字幕一二区| 菠萝蜜视频在线观看入口| 韩国精品一区二区三区六区色诱| 欧洲亚洲国产日韩| 国产精品丝袜久久久久久app| 国内精品久久久久久野外| 在线色视频观看| 男女无套免费网站| 亚洲人在线观看视频| 国产精品伦子伦免费视频| 欧美日韩中文一区二区| 是的av在线| 婷婷色在线观看| 欧美性猛交xxxxx免费看| 亚洲黄色精品| 国产成人自拍网站| 国产 日韩 欧美在线| 五月婷婷伊人网| 日本三级视频在线播放| av av在线| 亚洲综合好骚| 欧美日韩免费在线观看| 欧美性xxxx极品hd欧美风情| 男人的天堂avav| 欧美va亚洲va香蕉在线| 国产精品pans私拍| 欧美熟妇精品一区二区蜜桃视频| 四季av一区二区凹凸精品| 中国1级黄色片| 在线免费观看黄| 免费一级在线观看播放网址| 欧美在线观看网站| 日韩性生活视频| 欧美视频在线免费| 欧美大交乱xxxxbbbb| 女优一区二区三区| 亚洲乱码中文字幕综合| wwwww黄色| 国产亚洲精久久久久久| 高清日韩一区| 精品欧美激情精品一区| av电影在线不卡| 波多野结衣视频免费观看| 日本三级小视频| 美女免费久久| 91成人免费观看| 熟女俱乐部一区二区| 最新版sss视频在线| 人禽交欧美网站| 国产精东传媒成人av电影| 国产日韩精品一区| 狠狠久久亚洲欧美| 超碰一区二区| 久久综合伊人77777麻豆| 亚洲产国偷v产偷v自拍涩爱| 日韩电影一区| 欧美呦呦网站| 极品白嫩的小少妇| 日日夜夜一区二区| 精品免费在线视频| 欧美一区日本一区韩国一区| 久久频这里精品99香蕉| 欧美三级日韩三级国产三级| 少妇激情一区二区三区| 永久免费不卡在线观看黄网站| 亚洲欧美国产另类| 成人久久精品视频|