久久福利_99r_国产日韩在线视频_直接看av的网站_中文欧美日韩_久久一

您的位置:首頁技術文章
文章詳情頁

MyBatis攔截器的原理與使用

瀏覽:57日期:2023-10-19 09:07:07
目錄一、攔截對象和接口實現示例二、攔截器注冊的三種方式1.XML注冊2.配置類注冊3.注解方式三、ParameterHandler參數改寫-修改時間和修改人統一插入四、通過StatementHandler改寫SQL一、攔截對象和接口實現示例

MyBatis攔截器的作用是在于Dao到DB中間進行額外的處理。大部分情況下通過mybatis的xml配置sql都可以達到想要的DB操作效果,然而存在一些類似或者相同的查詢條件或者查詢要求,這些可以通過攔截器的實現可以提升開發效率,比如:分頁、插入和更新時間/人、數據權限、SQL監控日志等。

Mybatis支持四種對象攔截Executor、StatementHandler、PameterHandler和ResultSetHandler Executor:攔截執行器的方法。 StatementHandler:攔截Sql語法構建的處理。 ParameterHandler:攔截參數的處理。 ResultHandler:攔截結果集的處理。

public interface Executor { ResultHandler NO_RESULT_HANDLER = null; int update(MappedStatement var1, Object var2) throws SQLException; <E> List<E> query(MappedStatement var1, Object var2, RowBounds var3, ResultHandler var4, CacheKey var5, BoundSql var6) throws SQLException; <E> List<E> query(MappedStatement var1, Object var2, RowBounds var3, ResultHandler var4) throws SQLException; <E> Cursor<E> queryCursor(MappedStatement var1, Object var2, RowBounds var3) throws SQLException; List<BatchResult> flushStatements() throws SQLException; void commit(boolean var1) throws SQLException; void rollback(boolean var1) throws SQLException; CacheKey createCacheKey(MappedStatement var1, Object var2, RowBounds var3, BoundSql var4); boolean isCached(MappedStatement var1, CacheKey var2); void clearLocalCache(); void deferLoad(MappedStatement var1, MetaObject var2, String var3, CacheKey var4, Class<?> var5); Transaction getTransaction(); void close(boolean var1); boolean isClosed(); void setExecutorWrapper(Executor var1);}public interface StatementHandler { Statement prepare(Connection var1, Integer var2) throws SQLException; void parameterize(Statement var1) throws SQLException; void batch(Statement var1) throws SQLException; int update(Statement var1) throws SQLException; <E> List<E> query(Statement var1, ResultHandler var2) throws SQLException; <E> Cursor<E> queryCursor(Statement var1) throws SQLException; BoundSql getBoundSql(); ParameterHandler getParameterHandler();}public interface ParameterHandler { Object getParameterObject(); void setParameters(PreparedStatement var1) throws SQLException;}public interface ResultHandler<T> { void handleResult(ResultContext<? extends T> var1);}

攔截的執行順序是Executor->StatementHandler->ParameterHandler->ResultHandler

MyBatis提供的攔截器接口:

public interface Interceptor { Object intercept(Invocation var1) throws Throwable; default Object plugin(Object target) {return Plugin.wrap(target, this); } default void setProperties(Properties properties) {}}

Object intercept方法用于攔截器的實現;

Object plugin方法用于判斷執行攔截器的類型;

void setProperties方法用于獲取配置項的屬性。

攔截對象和攔截器接口的結合,自定義的攔截器類需要實現攔截器接口,并通過注解@Intercepts和參數@Signature來聲明要攔截的對象。

@Signature參數type是攔截對象,method是攔截的方法,即上面的四個類對應的方法,args是攔截方法對應的參數(方法存在重載因此需要指明參數個數和類型)

@Intercepts可以有多個@Signature,即一個攔截器實現類可以同時攔截多個對象及方法,示例如下:

Executor->intercept StatementHandler->intercept ParameterHandler->intercept ResultHandler->intercept

@Intercepts({@Signature(type = Executor.class,method = 'query',args = {MappedStatement.class, Object.class, RowBounds.class, ResultHandler.class})})public class SelectPlugin implements Interceptor { @Override public Object intercept(Invocation invocation) throws Throwable {if (invocation.getTarget() instanceof Executor) { System.out.println('SelectPlugin');}return invocation.proceed(); } @Override public Object plugin(Object target) {if (target instanceof Executor) { return Plugin.wrap(target, this);}return target; } @Override public void setProperties(Properties properties) {}}@Intercepts({@Signature(type = StatementHandler.class, method = 'prepare', args = {Connection.class, Integer.class})})public class StatementPlugin implements Interceptor { @Override public Object intercept(Invocation invocation) throws Throwable {if (invocation.getTarget() instanceof StatementHandler) { System.out.println('StatementPlugin');}return invocation.proceed(); } @Override public Object plugin(Object target) {if (target instanceof StatementHandler) { return Plugin.wrap(target, this);}return target; } @Override public void setProperties(Properties properties) {}}@Intercepts({@Signature(type = ParameterHandler.class,method = 'setParameters',args = {PreparedStatement.class})})public class ParameterPlugin implements Interceptor { @Override public Object intercept(Invocation invocation) throws Throwable {if (invocation.getTarget() instanceof ParameterHandler) { System.out.println('ParameterPlugin');}return invocation.proceed(); } @Override public Object plugin(Object target) {if (target instanceof ParameterHandler) { return Plugin.wrap(target, this);}return target; } @Override public void setProperties(Properties properties) {}}@Intercepts({@Signature(type = ResultHandler.class,method = 'handleResult',args = {ResultContext.class})})public class ResultPlugin implements Interceptor { @Override public Object intercept(Invocation invocation) throws Throwable {if (invocation.getTarget() instanceof ResultHandler) { System.out.println('ResultPlugin');}return invocation.proceed(); } @Override public Object plugin(Object target) {if (target instanceof ResultHandler) { return Plugin.wrap(target, this);}return target; } @Override public void setProperties(Properties properties) {}}二、攔截器注冊的三種方式

前面介紹了Mybatis的攔截對象及其接口的實現方式,那么在項目中如何注冊攔截器呢?本文中給出三種注冊方式。

1.XML注冊

xml注冊是最基本的方式,是通過在Mybatis配置文件中plugins元素來進行注冊的。一個plugin對應著一個攔截器,在plugin元素可以指定property子元素,在注冊定義攔截器時把對應攔截器的所有property通過Interceptor的setProperties方法注入給攔截器。因此攔截器注冊xml方式如下:

<?xml version='1.0' encoding='UTF-8' ?><!DOCTYPE configuration PUBLIC '-//mybatis.org//DTD Config 3.0//EN' 'http://mybatis.org/dtd/mybatis-3-config.dtd'><configuration> <!-- ...... --> <plugins> <plugin interceptor='com.tiantian.mybatis.interceptor.MyInterceptor'> <property name='prop1' value='prop1'/> <property name='prop2' value='prop2'/> </plugin> </plugins> <!-- ...... --></configuration>2.配置類注冊

配置類注冊是指通過Mybatis的配置類中聲明注冊攔截器,配置類注冊也可以通過Properties類給Interceptor的setProperties方法注入參數。具體參考如下:

@Configurationpublic class MyBatisConfig { @Bean public String MyBatisInterceptor(SqlSessionFactory sqlSessionFactory) {UpdatePlugin executorInterceptor = new UpdatePlugin();Properties properties = new Properties();properties.setProperty('prop1', 'value1');// 給攔截器添加自定義參數executorInterceptor.setProperties(properties);sqlSessionFactory.getConfiguration().addInterceptor(executorInterceptor);sqlSessionFactory.getConfiguration().addInterceptor(new StatementPlugin());sqlSessionFactory.getConfiguration().addInterceptor(new ResultPlugin());sqlSessionFactory.getConfiguration().addInterceptor(new ParameterPlugin());// sqlSessionFactory.getConfiguration().addInterceptor(new SelectPlugin());return 'interceptor'; } // 與sqlSessionFactory.getConfiguration().addInterceptor(new SelectPlugin());效果一致 @Bean public SelectPlugin SelectInterceptor() {SelectPlugin interceptor = new SelectPlugin();Properties properties = new Properties();// 調用properties.setProperty方法給攔截器設置自定義參數interceptor.setProperties(properties);return interceptor; }}3.注解方式

通過@Component注解方式是最簡單的方式,在不需要轉遞自定義參數時可以使用,方便快捷。

@Component@Intercepts({@Signature(type = Executor.class,method = 'query',args = {MappedStatement.class, Object.class, RowBounds.class, ResultHandler.class})})public class SelectPlugin implements Interceptor { @Override public Object intercept(Invocation invocation) throws Throwable {if (invocation.getTarget() instanceof Executor) { System.out.println('SelectPlugin');}return invocation.proceed(); } @Override public Object plugin(Object target) {if (target instanceof Executor) { return Plugin.wrap(target, this);}return target; } @Override public void setProperties(Properties properties) { }}三、ParameterHandler參數改寫-修改時間和修改人統一插入

針對具體的攔截器實現進行描述。日常編碼需求中會碰到修改時需要插入修改的時間和人員,如果要用xml的方式去寫非常麻煩,而通過攔截器的方式可以快速實現全局的插入修改時間和人員。先看代碼:

@Component@Intercepts({@Signature(type = ParameterHandler.class, method = 'setParameters', args = {PreparedStatement.class}),})public class MyBatisInterceptor implements Interceptor { @Override public Object intercept(Invocation invocation) throws Throwable {// 參數代理if (invocation.getTarget() instanceof ParameterHandler) { System.out.println('ParameterHandler'); // 自動添加操作員信息 autoAddOperatorInfo(invocation);}return invocation.proceed(); } @Override public Object plugin(Object target) {return Plugin.wrap(target, this); } @Override public void setProperties(Properties properties) { } /** * 自動添加操作員信息 * * @param invocation 代理對象 * @throws Throwable 異常 */ private void autoAddOperatorInfo(Invocation invocation) throws Throwable {System.out.println('autoInsertCreatorInfo');// 獲取代理的參數對象ParameterHandlerParameterHandler ph = (ParameterHandler) invocation.getTarget();// 通過MetaObject獲取ParameterHandler的反射內容MetaObject metaObject = MetaObject.forObject(ph,SystemMetaObject.DEFAULT_OBJECT_FACTORY,SystemMetaObject.DEFAULT_OBJECT_WRAPPER_FACTORY,new DefaultReflectorFactory());// 通過MetaObject反射的內容獲取MappedStatementMappedStatement mappedStatement = (MappedStatement) metaObject.getValue('mappedStatement');// 當sql類型為INSERT或UPDATE時,自動插入操作員信息if (mappedStatement.getSqlCommandType() == SqlCommandType.INSERT ||mappedStatement.getSqlCommandType() == SqlCommandType.UPDATE) { // 獲取參數對象 Object obj = ph.getParameterObject(); if (null != obj) {// 通過反射獲取參數對象的屬性Field[] fields = obj.getClass().getDeclaredFields();// 遍歷參數對象的屬性for (Field f : fields) { // 如果sql是INSERT,且存在createdAt屬性 if ('createdAt'.equals(f.getName()) && mappedStatement.getSqlCommandType() == SqlCommandType.INSERT) {// 設置允許訪問反射屬性f.setAccessible(true);// 如果沒有設置createdAt屬性,則自動為createdAt屬性添加當前的時間if (null == f.get(obj)) { // 設置createdAt屬性為當前時間 f.set(obj, LocalDateTime.now());} } // 如果sql是INSERT,且存在createdBy屬性 if ('createdBy'.equals(f.getName()) && mappedStatement.getSqlCommandType() == SqlCommandType.INSERT) {// 設置允許訪問反射屬性f.setAccessible(true);// 如果沒有設置createdBy屬性,則自動為createdBy屬性添加當前登錄的人員if (null == f.get(obj)) { // 設置createdBy屬性為當前登錄的人員 f.set(obj, 0);} } // sql為INSERT或UPDATE時均需要設置updatedAt屬性 if ('updatedAt'.equals(f.getName())) {f.setAccessible(true);if (null == f.get(obj)) { f.set(obj, LocalDateTime.now());} } // sql為INSERT或UPDATE時均需要設置updatedBy屬性 if ('updatedBy'.equals(f.getName())) {f.setAccessible(true);if (null == f.get(obj)) { f.set(obj, 0);} }}// 通過反射獲取ParameterHandler的parameterObject屬性Field parameterObject = ph.getClass().getDeclaredField('parameterObject');// 設置允許訪問parameterObject屬性parameterObject.setAccessible(true);// 將上面設置的新參數對象設置到ParameterHandler的parameterObject屬性parameterObject.set(ph, obj); }} }}

攔截器的接口實現參考前文,這里著重介紹autoAddOperatorInfo方法里的相關類。

1.ParameterHandler

接口源碼:

public interface ParameterHandler { Object getParameterObject(); void setParameters(PreparedStatement var1) throws SQLException; }

提供兩個方法:

getParameterObject是獲取參數對象,可能存在null,需要注意null指針。

setParameters是控制如何設置SQL參數,即sql語句中配置的java對象和jdbc類型對應的關系,例如#{id,jdbcType=INTEGER},id默認類型是javaType=class java.lang.Integer。

該接口有一個默認的實現類,源碼如下:

public class DefaultParameterHandler implements ParameterHandler { private final TypeHandlerRegistry typeHandlerRegistry; private final MappedStatement mappedStatement; private final Object parameterObject; private final BoundSql boundSql; private final Configuration configuration; public DefaultParameterHandler(MappedStatement mappedStatement, Object parameterObject, BoundSql boundSql) {this.mappedStatement = mappedStatement;this.configuration = mappedStatement.getConfiguration();this.typeHandlerRegistry = mappedStatement.getConfiguration().getTypeHandlerRegistry();this.parameterObject = parameterObject;this.boundSql = boundSql; } public Object getParameterObject() {return this.parameterObject; } public void setParameters(PreparedStatement ps) {ErrorContext.instance().activity('setting parameters').object(this.mappedStatement.getParameterMap().getId());List<ParameterMapping> parameterMappings = this.boundSql.getParameterMappings();if (parameterMappings != null) { for(int i = 0; i < parameterMappings.size(); ++i) {ParameterMapping parameterMapping = (ParameterMapping)parameterMappings.get(i);if (parameterMapping.getMode() != ParameterMode.OUT) { String propertyName = parameterMapping.getProperty(); Object value; if (this.boundSql.hasAdditionalParameter(propertyName)) {value = this.boundSql.getAdditionalParameter(propertyName); } else if (this.parameterObject == null) {value = null; } else if (this.typeHandlerRegistry.hasTypeHandler(this.parameterObject.getClass())) {value = this.parameterObject; } else {MetaObject metaObject = this.configuration.newMetaObject(this.parameterObject);value = metaObject.getValue(propertyName); } TypeHandler typeHandler = parameterMapping.getTypeHandler(); JdbcType jdbcType = parameterMapping.getJdbcType(); if (value == null && jdbcType == null) {jdbcType = this.configuration.getJdbcTypeForNull(); } try {typeHandler.setParameter(ps, i + 1, value, jdbcType); } catch (SQLException | TypeException var10) {throw new TypeException('Could not set parameters for mapping: ' + parameterMapping + '. Cause: ' + var10, var10); }} }} }}

通過DefaultParameterHandler實現類我們知道通過ParameterHandler可以獲取到哪些屬性和方法,其中包括我們下面一個重要的類MappedStatement。

2.MappedStatement

MyBatis的mapper文件中的每個select/update/insert/delete標簽會被解析器解析成一個對應的MappedStatement對象,也就是一個MappedStatement對象描述一條SQL語句。MappedStatement對象屬性如下:

// mapper配置文件名 private String resource; // mybatis的全局信息,如jdbc private Configuration configuration; // 節點的id屬性加命名空間,如:com.example.mybatis.dao.UserMapper.selectByExample private String id; private Integer fetchSize; private Integer timeout; private StatementType statementType; private ResultSetType resultSetType; private SqlSource sqlSource; private Cache cache; private ParameterMap parameterMap; private List<ResultMap> resultMaps; private boolean flushCacheRequired; private boolean useCache; private boolean resultOrdered; // sql語句的類型:select、update、delete、insert private SqlCommandType sqlCommandType; private KeyGenerator keyGenerator; private String[] keyProperties; private String[] keyColumns; private boolean hasNestedResultMaps; private String databaseId; private Log statementLog; private LanguageDriver lang; private String[] resultSets;

在本例中通過MappedStatement對象的sqlCommandType來判斷當前的sql類型是insert、update來進行下一步的操作。

四、通過StatementHandler改寫SQL

StatementHandler是用于封裝JDBC Statement操作,負責對JDBC Statement的操作,如設置參數,并將Statement結果集轉換成List集合。

實現代碼如下:

刪除注解標記

@Target({ElementType.METHOD}) //表示注解的使用范圍@Retention(RetentionPolicy.RUNTIME) //注解的保存時間@Documented //文檔顯示public @interface DeletedAt { boolean has() default true;}

Dao層添加刪除注解,為false時不添加刪除標志

@Mapper public interface AdminProjectDao { @DeletedAt(has = false) List<AdminProjectPo> selectProjects(AdminProjectPo po); }

攔截器通過刪除注解標記判斷是否添加刪除標志

@Component@Intercepts({@Signature(type = StatementHandler.class, method = 'prepare', args = {Connection.class, Integer.class}),})public class MyBatisInterceptor implements Interceptor { @Override public Object intercept(Invocation invocation) throws Throwable {if (invocation.getTarget() instanceof StatementHandler) { System.out.println('StatementHandler'); checkHasDeletedAtField(invocation);}return invocation.proceed(); } @Override public Object plugin(Object target) {return Plugin.wrap(target, this); } @Override public void setProperties(Properties properties) { } /** * 檢查查詢是否需要添加刪除標志字段 * * @param invocation 代理對象 * @throws Throwable 異常 */ private void checkHasDeletedAtField(Invocation invocation) throws Throwable {System.out.println('checkHasDeletedAtField');StatementHandler statementHandler = (StatementHandler) invocation.getTarget();// 通過MetaObject訪問對象的屬性MetaObject metaObject = MetaObject.forObject(statementHandler,SystemMetaObject.DEFAULT_OBJECT_FACTORY,SystemMetaObject.DEFAULT_OBJECT_WRAPPER_FACTORY,new DefaultReflectorFactory());// 獲取成員變量mappedStatementMappedStatement mappedStatement = (MappedStatement) metaObject.getValue('delegate.mappedStatement');// 如果sql類型是查詢if (mappedStatement.getSqlCommandType() == SqlCommandType.SELECT) { // 獲取刪除注解標志 DeletedAt annotation = null; String id = mappedStatement.getId(); String className = id.substring(0, id.lastIndexOf('.')); String methodName = id.substring(id.lastIndexOf('.') + 1); Class<?> aClass = Class.forName(className); Method[] declaredMethods = aClass.getDeclaredMethods(); for (Method declaredMethod : declaredMethods) {declaredMethod.setAccessible(true);//方法名相同,并且注解是DeletedAtif (methodName.equals(declaredMethod.getName()) && declaredMethod.isAnnotationPresent(DeletedAt.class)) { annotation = declaredMethod.getAnnotation(DeletedAt.class);} } // 如果注解不存在或者注解為true(默認為true) 則為mysql語句增加刪除標志 if (annotation == null || annotation.has()) {BoundSql boundSql = statementHandler.getBoundSql();//獲取到原始sql語句String sql = boundSql.getSql();//通過反射修改sql語句Field field = boundSql.getClass().getDeclaredField('sql');field.setAccessible(true);String newSql = sql.replaceAll('9=9', '9=9 and deleted_at is null ');field.set(boundSql, newSql); }} }}

在SQL語句替換上需要能識別到要被替換的內容,因此在xml的sql語句中加入特殊標志'9=9',該標志不影響原來SQL的執行結果,不同的過濾條件可以設置不同的標志,是一個比較巧妙的替換方式。

以上就是MyBatis攔截器的原理與使用的詳細內容,更多關于MyBatis攔截器的資料請關注好吧啦網其它相關文章!

標簽: Mybatis 數據庫
相關文章:
主站蜘蛛池模板: 久久se精品一区精品二区 | 奇米影视首页 | 精品久久久久一区二区国产 | 日本中文字幕一区 | 一区二区三区中文字幕 | 韩日精品一区 | 日韩网站免费观看 | 欧美精品一区二区在线观看 | 亚洲欧美日韩在线 | 国产一极片 | 国产高清视频一区二区 | 精品国产三级 | 日日干夜夜骑 | av在线免费观看一区二区 | 亚洲天堂久久 | 日本在线观看 | 荡女妇边被c边呻吟视频 | 视频一区免费观看 | 亚洲一区二区三区四区五区中文 | 国产在线激情视频 | 精品久久久久久久 | 成人国产精品一级毛片视频 | 天天草天天色 | 国产成人精品午夜视频' | 精品国产九九 | 99视频这里有精品 | 日韩毛片免费视频一级特黄 | 亚洲免费在线观看 | 在线视频不卡一区 | 亚洲免费在线视频 | 99精彩视频 | 色婷婷基地 | 久久国产一区视频 | 久久久久久免费看 | 亚洲欧美影院 | 黄色天堂在线观看 | 日韩精品 电影一区 亚洲 | 成人一区二区三区久久精品嫩草 | 欧美操穴| 成人免费淫片aa视频免费 | 91一区 | 久久夜色精品国产 | 中文字幕日韩欧美一区二区三区 | 久草热8精品视频在线观看 欧美全黄 | 国产亚洲精品久久久456 | 久国产| 欧美精品在线视频 | 91在线视频福利 | 日韩精品一区二区在线观看 | 精品国产乱码久久久久久久软件 | 午夜精品在线 | 亚洲一区二区av | 国产91在线 | 亚洲 | 在线视频 91 | 夜夜夜操 | 国内成人精品2018免费看 | 亚洲成人国产精品 | 麻豆av在线播放 | 成人久久久 | 国产91在线 | 亚洲 | 欧美成人精品一区二区男人看 | 国产成人综合在线 | 国产成人精品免高潮在线观看 | 亚洲精选久久 | 国产亚洲精品一区二区 | av日韩在线看 | av电影天堂网 | 国产精品亚洲欧美日韩一区在线 | 在线免费色视频 | 色婷婷国产精品综合在线观看 | 亚洲 自拍 另类 欧美 丝袜 | 九九色综合 | 97国产精品视频人人做人人爱 | 成年人精品视频 | 午夜电影 | 黄色一级片视频播放 | 精品国产欧美一区二区 | 成人av网页| 99久久精品国产一区二区成人 | 久久久一区二区三区 | 噜噜噜噜狠狠狠7777视频 | 国产成人久久精品麻豆二区 | 亚洲一区国产精品 | 九九99热 | 亚洲一区二区 | 欧美一区久久 | 国产美女高潮一区二区三区 | 亚洲视频一区二区三区 | 日韩成人免费 | www国产亚洲精品 | 草久网| 在线观看毛片网站 | 亚洲视频一区二区三区四区 | 国产一区中文字幕 | 91社区在线高清 | 国产精品一卡二卡三卡 | 影音先锋成人资源网 | 亚洲高清视频网站 | 国产成人亚洲综合 | www久久99 | 久久国产高清 | 97伦理在线| 国产福利在线观看 | 国产精品一二三区 | 国色天香成人网 | 亚洲黄色成人 | 亚洲成人av | 激情福利视频 | 黄网站色大毛片 | 国产视频亚洲 | 久久久人成影片一区二区三区 | 欧美在线国产 | 欧美精品第一页 | 日本久久久一区二区三区 | 久久一区国产 | 91久久久久久久久 | 久久99精品久久久久久琪琪 | 免费大片黄在线观看 | 蜜桃av一区二区三区 | 色婷婷国产精品综合在线观看 | jjzz18国产 | 亚洲每日更新 | 欧美人体一区二区三区 | 日韩精品在线免费观看 | 国产成人精品在线 | 亚洲一级黄色 | 成av在线| 亚洲精品久久久久久久久久久 | 欧美在线网站 | 99亚洲精品 | 日本 国产 欧美 | 国产一级免费视频 | 免费av在线网站 | 日韩一区二区在线观看 | 午夜精品久久久久久久久久久久久 | 欧美精品1区2区3区 精品国产欧美一区二区 | 日韩精品一二三 | 91麻豆精品国产91久久久更新时间 | 免费成人在线观看视频 | 国产精品久久久久久久竹霞 | 久久九九国产精品 | 视频一区免费观看 | 一区二区三区在线观看视频 | 日韩一区电影 | 成人在线网址 | 国产成人精品无人区一区 | 亚洲精品成人av | 亚洲电影一区二区 | baoyu123成人免费看视频 | 成人在线| 精品美女在线观看视频在线观看 | 一级一级国产片 | 亚洲国产精品一区二区www | 91电影院 | 天天天操 | 古典武侠第一页久久777 | 婷婷亚洲五月 | 亚洲精选久久 | 五月天婷婷激情视频 | 中文字幕亚洲欧美日韩在线不卡 | 不卡视频一区 | 呦一呦二在线精品视频 | 欧美三级影院 | 国产日韩一区二区三区 | 欧美性猛交一区二区三区精品 | 国产免费黄色大片 | 久热精品在线视频 | 成人精品| 成人午夜电影网 | 日韩免费在线观看视频 | 免费在线看a| 亚洲精品国精品久久99热 | 一区免费视频 | 美女一区 | 精品国产三级 | 欧美日韩黄色一级片 | 成人精品久久久 | 欧美午夜视频 | 干干日日 | 欧美xxxx在线| 久久精品欧美一区二区三区不卡 | 人人射av| 操操操日日日 | 色吧久久 | 正在播放国产精品 | 青草成人免费视频 | 午夜精品91 | 国产成人网 | 二区在线视频 | 国产一区二 | 国产福利在线视频 | 国产精品精品视频一区二区三区 | 久久蜜桃精品一区二区三区综合网 | 久草精品视频 | av毛片| 亚洲高清在线 | 狠狠躁夜夜躁人人爽天天天天97 | 国产精品亚洲视频 | 99亚洲| aaa久久| 国产xxx护士爽免费看 | caoporn国产精品免费公开 | 啪啪tv网站免费入口 | 国产午夜精品美女视频明星a级 | 一二三区精品 | 婷婷综合 | 日韩欧美中文字幕在线视频 | 亚洲精品视频大全 | 久久久久久久久国产 | 欧美日韩一区二区中文字幕 | 国产精品极品美女在线观看免费 | 午夜精品在线 | 久久天堂电影 | 亚洲精品久久久久国产 | 精品日韩av| 国产一区二区三区免费视频 | 中国国产一级毛片 | 日韩三级网| jizz欧美最大| 国产欧美一区二区三区在线看 | 极品白嫩少妇无套内谢 | 国产精品女教师av久久 | 福利久久久 | 国产在线不卡 | av免费网站在线观看 | 日韩a视频 | 99热精品免费 | 九九免费视频 | 精品免费国产一区二区三区 | 国产中文字幕一区 | 国产精品一区二 | 日本大人吃奶视频xxxx | 国产精品国产三级国产a | 日韩欧美在线观看视频 | 黄色一级免费大片 | 国产精一区二区 | 一区二区在线免费观看 | 国产精品久久久久久久久久久新郎 | 欧美日韩国产一区二区三区 | 99视频网| 成人1区 | 黄色国产视频 | 永久av | 欧美日韩一区二区三区 | 国产黑丝在线 | 精品久久一级片 | 成人在线一区二区 | 亚洲国产区 | 中文字幕在线观看 | 欧美一区2区三区3区公司 | 91中文字幕在线 | 国产精品久久久久久妇女6080 | 女同理伦片在线观看禁男之园 | 日本福利网站 | 亚洲日韩欧美一区二区在线 | 在线免费成人 | 久久久久久久久一区二区三区 | 日韩精品一区二区三区老鸭窝 | 免费观看a视频 | 国产成人在线视频 | 亚洲一区二区三区精品视频 | 91精品国产综合久久精品 | 天天天天天天天操 | 亚洲综合精品视频 | 国产欧美精品一区二区三区四区 | 天天干夜操 | 免费国产视频在线观看 | a久久久 | 玖玖操| 国产精品久久久久久久久费观看 | 久久国产精品影视 | 在线色网站| 一区二区手机在线 | 国产v日产∨综合v精品视频 | 成人国产免费视频 | 午夜影院免费版 | www97影院 | 欧美日韩免费一区二区三区 | 欧美色综合 | 亚洲综合视频 | 欧美精品一区二区三区在线 | 久久久一二三 | 久久a国产 | 一区二区三区日韩在线 | 久久不射电影网 | 久久婷婷香蕉 | h片在线| 91精品国产91久久久久久久久久久久 | 色视频网站在线观看 | 精品美女在线观看视频在线观看 | 综合一区二区三区 | 91大神xh98hx在线播放 | 亚洲国产精品一区二区三区 | 久久久久亚洲一区二区三区 | 丁香五月网久久综合 | 久久久久久久久国产 | 午夜在线观看视频网站 | 成人黄色在线观看 | 性视频黄色 | 色欧美片视频在线观看 | www久| 亚洲成人免费视频 | 最新免费av网站 | 精品国产一区二区在线 | 91久久久久久久久久久久久久久久 | 国产一级视频免费播放 | 欧美国产三级 | 日韩一区二区在线免费观看 | 久久精品无码一区二区三区 | 成人av电影网址 | 亚洲h视频 | 99热这里有精品 | 人人射人人草 | 一本一道久久精品综合 | www.狠狠干| 日韩性xxx | 一区二区精品 | 福利在线看 | 亚洲一区二区高清视频 | av成人在线观看 | 国产一级淫免费播放m | 久久久成人精品 | 91精品国产综合久久久久久蜜月 | 激情欧美一区二区三区中文字幕 | 夜夜操操 | 搡女人真爽免费午夜网站 | 性高湖久久久久久久久 | 欧美在线免费 | 一本一道久久a久久精品逆3p | 国产又粗又长又硬又猛电影 | 欧美在线视频播放 | 国产精品视频播放 | 一区二区三区国产 | 99久久99久久精品 | 久久精品网 | 精品久久久久国产 | 久久黄色 | 亚洲欧美一区二区三区在线 | www伊人| 国产人妖一区 | 国产精品久久7777 | segui88久久综合9999 | 欧美色性| 一区二区三区视频免费在线观看 | 国产一区二区三区久久久 | 精品一区二区三区免费视频 | 欧美 日韩 | 鲁一鲁影院 | 亚洲自拍偷拍精品 | 亚洲国产成人在线视频 | 免费在线成人av | 中文字幕在线观看视频一区 | 国产一区www | 精品一区视频 | 羞羞视频在线播放 | 欧美日韩专区 | 97av在线| 国产一区二区三区欧美 | 在线观看欧美一区 | 97色在线视频 | 中出片| 九九热有精品 | 国产一区二区欧美 | 色婷婷网 | 国产美女精品 | 99精品电影 | 国产亚洲欧美在线 | 亚洲一区二区三区免费观看 | 波多野结衣 一区二区三区 精品精品久久 | 中文字幕第66页 | 国产视频二 | 夜夜操操操 | 欧美日韩一区二区三区在线观看 | 午夜免费剧场 | 久久久久久成人精品 | 亚洲精品二区 | av免费在线观看网站 | 午夜影院毛片 | 国产精品免费av | 久一区二区三区 | 国产精品女教师av久久 | 国产亚洲欧美一区二区 | 日韩一区二区在线免费 | 国厂黄色片 | 国内久久精品 | 久久va| 爱爱视频网站 | 国产视频导航 | 中文字幕91 | 久久伊| 日韩成人中文字幕 | 国产一区二区自拍 | 欧美日韩在线二区 | 久草视频播放 | 日日插日日操 | 久久人 | 日韩久久精品 | 99re国产精品视频 | 激情国产| 三级网站在线播放 | 国产一区二区三区四区视频 | 国产精品成人3p一区二区三区 | 亚洲精品久久久久久久久久久 | 免费的一级毛片 | 久久久av亚洲男天堂 | 99久久综合 | 中文字幕三区 | 毛片一区 | 亚洲美女视频在线观看 | 国产午夜久久久久 | www国产亚洲精品久久网站 | 国产激情偷乱视频一区二区三区 | 国产亚洲精品美女久久久久久久久久 | 国产999精品久久久久久 | 中文字幕黄色 | 日韩一区二区三区在线观看 | 欧美日韩一区二区三区不卡视频 | 最新日韩欧美 | av午夜电影 | 裸体的日本在线观看 | 成人aaaa| 成人超碰在线 | 精品www| 国产精品国产自产拍高清 | 亚洲男人的天堂在线观看 | 欧美极品一区二区 | 亚洲一区二区三区免费观看 | 亚洲一区二区在线播放 | 日本a视频| 亚洲一区 | 亚洲国产精品成人无久久精品 | 久久久久91| 91麻豆精品国产91久久久久久 | 日韩在线免费视频 | 日韩免费 | 亚洲欧美久久久 | 欧日韩不卡在线视频 | 日韩精品小视频 | 国产精品一区二区三区四区 | 免费一级 国产 | 玖玖精品| 国产91网 | 亚洲国产成人av好男人在线观看 | 一区二区三区四区视频 | 国产三级精品在线 | а天堂中文最新一区二区三区 | www.99热 | 日韩国产二区 | av网站观看 | 久久国产精品视频 | 久久精品亚洲一区二区 | 久久免费黄色网址 | 综合久久综合久久 | 一区二区三区四区在线 | 一区二区日本 | 国产成人高清 | 99精品久久久久久久免费 | 精品国产一区二区三区久久久蜜月 | 中文字幕一区二区三区日韩精品 | 国产成人一区二区三区影院在线 | 黄在线免费观看 | 毛片网站在线观看 | 国产一区2区 | 中文字幕在线三区 | 农村末发育av片四区五区 | 成人免费视频网 | 久久亚洲一区二区三区四区 | 成人狠狠干 | 美女久久| 99精品九九| 日韩一本| 国产精品久久久久久中文字 | 91高清视频在线观看 | 国产色在线 | 国产精品丝袜视频 | 日本精品免费观看 | 国产在线视频在线 | 神马久久久久久 | 国产亚洲欧美在线 | 81精品国产乱码久久久久久 | 黄色二区 | 久久精品国产99 | 日韩欧美精品一区二区三区 | 欧美综合一区二区 | 成人免费精品视频 | 综合二区| 国产精品精品视频一区二区三区 | 青青草久 | 国内精品一区二区三区视频 | 小情侣高清国产在线播放 | 国产精品久久精品 | 精品国产一区二区三区久久久久久 | 人干人人 | 精品自拍视频 | 国产精品三级在线 | 欧美精品一区二区久久 | www.色综合 | www.国产欧美 | 久久久久亚洲国产 | 国产91精品一区二区绿帽 | 成人小视频在线播放 | 欧美激情在线精品一区二区三区 | 日韩在线成人 | 亚洲综合社区 | 国产亚洲精品久久久久久久 | 国产欧美日韩精品一区 | 亚洲国产精品久久 | 在线观看视频一区二区三区 | 精品国产一区在线 | 免费av手机在线观看 | 女同理伦片在线观看禁男之园 | 日韩精品免费在线观看 | 精品国产31久久久久久 | 国产精品久久久久久久久久三级 | 狠狠狠色丁香婷婷综合久久五月 | 99热最新网站 | 69性欧美高清影院 | 一区二区免费在线观看 | 亚洲午夜在线 | 综合精品久久久 | 国产精品久久久久久吹潮 | 国产综合久久久久久鬼色 | 91.xxx.高清在线| 亚洲不卡免费视频 | 亚洲啊v在线 | 日韩免费视频一区二区 | av在线成人 | 欧美高清视频在线观看 | 成年免费观看视频 | 午夜影院a| 成人午夜视频在线观看 | 欧美久久久久久 | 久久久久久久影院 | 国产一区二区精品在线 | 午夜天堂精品久久久久 | 久久久999成人 | 国产日韩一区 | 综合久久99 | 超碰人人操 | 福利片中文字幕 | 青青久久| 国产精品美女久久久久久久久久久 | 黄色片在线免费观看 | 亚洲日本中文 | 亚洲高清一区二区三区 | 亚洲国产成人在线视频 | 亚洲精品久久久久久久久久久久久 | 日本在线视频不卡 | a网站在线观看 | 在线免费观看日韩视频 | 国产一级片在线播放 | 成人欧美一区二区三区黑人孕妇 | 国产一区国产二区在线观看 | 午夜影院在线观看 | 一级一级一级毛片 | 欧美成人精品在线观看 | 一区二区激情 | 欧美亚洲91 | 国产美女高潮一区二区三区 | a视频在线观看免费 | 波多野结衣精品 | 欧美久久a| 99久热在线精品视频观看 | 久久网页 | 青青草av| 玖玖国产精品视频 | jlzzjlzz亚洲日本少妇 | 成人午夜免费视频 | 日韩精品一 | jizz欧美最大| 欧美精品成人一区二区三区四区 | 婷五月综合 | 97av在线| 在线观看国产视频 | 一区二区三区四区在线 | 精品欧美激情在线观看 | 成人高清视频在线观看 | 国产欧美日韩精品一区 | 成人国产精品久久 | 福利在线播放 | 国产精品一二 | 亚洲视频免费 | 最新国产在线 | 国产一级毛片在线视频 | 色5月婷婷丁香六月 | 一区二区在线电影 | 国产成人免费 | 天天干天天爱天天 | 精品一区二区在线观看 | 日韩精品一区二区三区中文字幕 | 久久久久久久久久久精 | 日本免费视频在线观看 | chengrenzaixian| 欧美人牲 | 一区二区三区在线不卡 | 成人av观看| 先锋影音在线观看 | 91影院在线观看 | 一级a毛片免费 | 欧美日韩精品一区二区在线播放 | 欧美综合视频在线观看 | 99精品国产高清一区二区麻豆 | 人人草人人干 | 色免费视频 | 中文字幕第31页 | 精品久久一区 | 中文字幕免费中文 | 国产成人在线一区二区 | 伊人色综合久久久天天蜜桃 | 免费毛片网站 | 亚洲综合首页 | 成人福利在线观看 | 国产成人精品久久 | 久久精品国产一区 | 欧美一区二区三区免费观看视频 |