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

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

淺談MyBatis 如何執行一條 SQL語句

瀏覽:66日期:2023-10-19 11:51:54
目錄前言基礎組件工作流程初步使用詳細流程獲取 MapperProxy 對象緩存執行方法構造參數獲取需要執行的 SQL 對象執行 SQL 語句總結前言

Mybatis 是 Java 開發中比較常用的 ORM 框架。在日常工作中,我們都是直接通過 Spring Boot 自動配置,并直接使用,但是卻不知道 Mybatis 是如何執行一條 SQL 語句的,而這篇文章就是來揭開 Mybatis 的神秘面紗。

基礎組件

我們要理解 Mybatis 的執行過程,就必須先了解 Mybatis 中都有哪一些重要的類,這些類的職責都是什么?

SqlSession

我們都很熟悉,它對外提供用戶和數據庫之間交互需要使用的方法,隱藏了底層的細節。它默認是實現類是 DefaultSqlSession

Executor

這個是執行器,SqlSession 中對數據庫的操作都是委托給它。它有多個實現類,可以使用不同的功能。

淺談MyBatis 如何執行一條 SQL語句

Configuration

它是一個很重要的配置類,它包含了 Mybatis 的所有有用信息,包括 xml 配置,動態 sql 語句等等,我們到處都可以看到這個類的身影。

MapperProxy

這是一個很重要的代理類,它代理的就是 Mybatis 中映射 SQL 的接口。也就是我們常寫的 Dao 接口。

工作流程初步使用

首先,我們需要得到一個 SqlSessionFactory 對象,該對象的作用是可以獲取 SqlSession 對象。

// 讀取配置InputStream resourceAsStream = Resources.getResourceAsStream('config.xml');SqlSessionFactoryBuilder sqlSessionFactoryBuilder = new SqlSessionFactoryBuilder();// 創建一個 SqlSessionFactory 對象SqlSessionFactory sqlSessionFactory = sqlSessionFactoryBuilder.build(resourceAsStream);

當我們得到一個 SqlSessionFactory 對象之后,就可以通過它的 openSession 方法得到一個 SqlSession 對象。

SqlSession sqlSession = sqlSessionFactory.openSession(true);

最后,我們通過 SqlSession 對象獲取 Mapper ,從而可以從數據庫獲取數據。

// 獲取 Mapper 對象HeroMapper mapper = sqlSession.getMapper(HeroMapper.class);// 執行方法,從數據庫中獲取數據Hero hero = mapper.selectById(1);詳細流程獲取 MapperProxy 對象

我們現在主要關注的就是 getMapper 方法,該方法為我們創建一個代理對象,該代理對象為我們執行 SQL 語句提供了重要的支持。

// SqlSession 對象@Overridepublic <T> T getMapper(Class<T> type) { return configuration.getMapper(type, this);}

getMapper 方法里面委托 Configuration 對象去獲取對應的 Mapper 代理對象,之前說過 Configuration 對象里面包含了 Mybatis 中所有重要的信息,其中就包括我們需要的 Mapper 代理對象,而這些信息都是在讀取配置信息的時候完成的,也就是執行sqlSessionFactoryBuilder.build 方法。

// Configuration 對象public <T> T getMapper(Class<T> type, SqlSession sqlSession) { return mapperRegistry.getMapper(type, sqlSession);}

我們可以看到它又將獲取 Mapper 代理對象的操作委托給了 MapperRegistry 對象(擱著俄羅斯套娃呢?),這個 MapperRegistry 對象里面就存放了我們想要的 Mapper 代理對象,如果你這么想,就錯了,實際上,它存放的并不是我們想要的 Mapper 代理對象,而是 Mapper 代理對象的工廠,Mybatis 這里使用到了工廠模式。

public class MapperRegistry { private final Configuration config; private final Map<Class<?>, MapperProxyFactory<?>> knownMappers = new HashMap<>(); public MapperRegistry(Configuration config) { this.config = config; } @SuppressWarnings('unchecked') public <T> T getMapper(Class<T> type, SqlSession sqlSession) { final MapperProxyFactory<T> mapperProxyFactory = (MapperProxyFactory<T>) knownMappers.get(type); if (mapperProxyFactory == null) { throw new BindingException('Type ' + type + ' is not known to the MapperRegistry.'); } try { return mapperProxyFactory.newInstance(sqlSession); } catch (Exception e) { throw new BindingException('Error getting mapper instance. Cause: ' + e, e); } } public <T> void addMapper(Class<T> type) { if (type.isInterface()) { if (hasMapper(type)) {throw new BindingException('Type ' + type + ' is already known to the MapperRegistry.'); } boolean loadCompleted = false; try {knownMappers.put(type, new MapperProxyFactory<>(type));// It’s important that the type is added before the parser is run// otherwise the binding may automatically be attempted by the// mapper parser. If the type is already known, it won’t try.MapperAnnotationBuilder parser = new MapperAnnotationBuilder(config, type);parser.parse();loadCompleted = true; } finally {if (!loadCompleted) { knownMappers.remove(type);} } } }}

我只保留了 getMapper 方法和 addMapper 方法。

在 getMapper 方法中,它獲取的是 MapperProxyFactory 對象,我們通過名稱可以得出這是一個 Mapper 代理對象工廠,但是我們是要得到一個 MapperProxy 對象,而不是一個工廠對象,我們再來看 getMapper 方法,它通過 mapperProxyFactory.newInstance 來創建代理對象。

protected T newInstance(MapperProxy<T> mapperProxy) { return (T) Proxy.newProxyInstance(mapperInterface.getClassLoader(), new Class[] { mapperInterface }, mapperProxy);}public T newInstance(SqlSession sqlSession) { final MapperProxy<T> mapperProxy = new MapperProxy<>(sqlSession, mapperInterface, methodCache); return newInstance(mapperProxy);}

創建了一個 MapperProxy 對象,并且通過 Proxy.newProxyInstance 方法(不會還有人不知道這是 JDK 動態代理吧),創建一個代理對象處理,這個代理對象就是我們想要的結果。這里沒有體現出來代理了哪個對象啊?其實 mapperInterface 這是一個成員變量,它引用了需要被代理的對象。而這個成員變量實在創建 MapperProxyFactory 對象的時候賦值的,所以我們每一個需要被代理的接口,在 Mybatis 中都會為它生成一個 MapperProxyFactory 對象,該對象的作用就是為了創建所需要的代理對象。

淺談MyBatis 如何執行一條 SQL語句

緩存執行方法

當我們獲取到代理對象 mapper 之后,就可以執行它里面的方法。這里使用一個例子:

// Myabtis 所需要的接口public interface HeroMapper { Hero selectById(Integer id);}

// HeroMapper 接口所對應的 xml 文件<?xml version='1.0' encoding='UTF-8' ?><!DOCTYPE mapper PUBLIC '-//mybatis.org//DTD Mapper 3.0//EN' 'http://mybatis.org/dtd/mybatis-3-mapper.dtd'><mapper namespace='test.HeroMapper'> <select resultType='test.Hero'>select * from hero where id = #{id} </select></mapper>

我們執行 selectById 方法,獲取一個用戶的信息。

// 獲取 Mapper 對象HeroMapper mapper = sqlSession.getMapper(HeroMapper.class);// 執行方法,從數據庫中獲取數據Hero hero = mapper.selectById(1);

通過上面的解析已經知道,這里的 mapper 是一個代理對象的引用,而這個代理類則是 MapperProxy,所以我們主要是去了解 MapperProxy 這個代理類做了什么事情。

public class MapperProxy<T> implements InvocationHandler, Serializable { private final SqlSession sqlSession; private final Class<T> mapperInterface; private final Map<Method, MapperMethodInvoker> methodCache; public MapperProxy(SqlSession sqlSession, Class<T> mapperInterface, Map<Method, MapperMethodInvoker> methodCache) { this.sqlSession = sqlSession; this.mapperInterface = mapperInterface; this.methodCache = methodCache; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { try { if (Object.class.equals(method.getDeclaringClass())) {return method.invoke(this, args); } else {return cachedInvoker(method).invoke(proxy, method, args, sqlSession); } } catch (Throwable t) { throw ExceptionUtil.unwrapThrowable(t); } } private MapperMethodInvoker cachedInvoker(Method method) throws Throwable { return methodCache.computeIfAbsent(method, m -> { return new PlainMethodInvoker(new MapperMethod(mapperInterface, method, sqlSession.getConfiguration())); } } private static class PlainMethodInvoker implements MapperMethodInvoker { private final MapperMethod mapperMethod; public PlainMethodInvoker(MapperMethod mapperMethod) { super(); this.mapperMethod = mapperMethod; } @Override public Object invoke(Object proxy, Method method, Object[] args, SqlSession sqlSession) throws Throwable { return mapperMethod.execute(sqlSession, args); } }}

代理對象執行方法時都是直接執行 invoke() 方法,在這個方法中,我們主要就看一條語句 cachedInvoker(method).invoke(proxy, method, args, sqlSession);

我們首先看 cachedInvoker 方法,它的參數是 Method 類型,所以這個 method 表示的就是我們執行的方法 HeroMapper.selectById,它首先從緩存中獲取是否之前已經創建過一個該方法的方法執行器 PlainMethodInvoker 對象,其實這只是一個包裝類,可有可無,在工程上來說,有了這個包裝類,會更加易于維護。而這個執行器里面只有一個成員對象,這個成員對象就是 MapperMethod,并且這個 MapperMethod 的構造函數中需要傳遞 HeroMapper、HeroMapper.selectById、Cofiguration 這三個參數。

以上步驟都執行完成之后,接下來我們可以看到執行了 PlainMethodInvoker 的 invoke 方法,而它又將真正的操作委托給了 MapperMethod,執行 MapperMethod 下的 execute 方法,這個方法就是本文章的重點所在。

淺談MyBatis 如何執行一條 SQL語句

構造參數

從上面的解析可以知道,最后會執行到這個方法之中。

public Object execute(SqlSession sqlSession, Object[] args) { Object result; switch (command.getType()) { case INSERT: {Object param = method.convertArgsToSqlCommandParam(args);result = rowCountResult(sqlSession.insert(command.getName(), param));break; } case UPDATE: {Object param = method.convertArgsToSqlCommandParam(args);result = rowCountResult(sqlSession.update(command.getName(), param));break; } case DELETE: {Object param = method.convertArgsToSqlCommandParam(args);result = rowCountResult(sqlSession.delete(command.getName(), param));break; } case SELECT:if (method.returnsVoid() && method.hasResultHandler()) { executeWithResultHandler(sqlSession, args); result = null;} else if (method.returnsMany()) { result = executeForMany(sqlSession, args);} else if (method.returnsMap()) { result = executeForMap(sqlSession, args);} else if (method.returnsCursor()) { result = executeForCursor(sqlSession, args);} else { Object param = method.convertArgsToSqlCommandParam(args); result = sqlSession.selectOne(command.getName(), param); if (method.returnsOptional() && (result == null || !method.getReturnType().equals(result.getClass()))) { result = Optional.ofNullable(result); }}break; case FLUSH:result = sqlSession.flushStatements();break; default:throw new BindingException('Unknown execution method for: ' + command.getName()); } return result; }

這個方法中,我們可以看到熟悉的幾個關鍵字:select、update、delete、insert,這個就是為了找到執行方式,我們因為是 select 語句,所以分支會走向 select,并且最終會執行到 sqlSession.selectOne 方法中,所以最終饒了一大圈,依然還是回到了我們一開始就提到的 SqlSession 對象中。在這個方法中,首先會構造參數,也就是我們看到的 convertArgsToSqlCommandParam 方法,它的內部執行方式是按照如下方式來轉換參數的:

使用 @param 自定義命名amethod(@Param int a, @Param int b) 則會構造 map -> [{'a', a_arg}, {'b', b_arg}, {'param1', a_arg}, {'param2', b_arg}],a 和 param1 是對參數 a 的命名,a_arg 是傳遞的實際的值。雖然只有兩個參數,但是最后卻會在 Map 存在四個鍵值對,因為 Mybatis 最后自己會生成以 param 為前綴的參數名稱,名稱按照參數的位置進行命名。

不使用 @param

amethod(int a, int b),則會構造 map -> [{'arg0', a_arg}, {'arg1', b_arg}, {'param1', a_arg}, {'param2', b_arg}],因為沒有對參數進行自定義命名,所以 Myabtis 就對參數取了一個默認的名稱,以 arg 為前綴,位置為后綴進行命名。

在參數只有一個,并且參數為集合的情況下,會存放多個鍵值對:

amethod(Collection<Integer> a),這種情況下,會構造 map -> [{'arg0', a_arg}, {'collection', a_arg}] amethod(List<Integer> a),這種情況下,會構造 map -> [{'arg0', a_arg}, {'collection', a_arg}, {'list', a_arg}] amethod(Integer[] a),這種情況下,會構造 map -> [{'arg0', a_arg}, {'array', a_arg}] 但是,如果有兩個參數,那么就不會這么存放,而是按照常規的方式: amethod(List<Integer> a,List<Integer> b) 則會構造 map -> [{'arg0', a_arg}, {'arg1', b_arg}, {'param1', a_arg}, {'param2', b_arg}] amethod(List<Integer> a,int b) 則會構造 map -> [{'arg0', a_arg}, {'arg1', b_arg}, {'param1', a_arg}, {'param2', b_arg}]

不會作為參數的對象在 Mybatis 中有兩個特殊的對象:RowBounds、ResultHandler,這兩個對象如果作為參數則不會放入到 map 中,但是會占據位置。

amethod(int a,RowBounds rb, int b),這種情況下,會構造 map -> [{'arg0', a_arg}, {'arg2', b_arg}, {'param1', a_arg}, {'param2', b_arg}]

注意這里的 b 參數的命名分別是 arg2 和 param2,arg2 是因為它的位置在參數的第 3 位,而 param2 則是因為它是第 2 個有效參數。

獲取需要執行的 SQL 對象

參數構造完成之后,我們就需要尋找需要執行的 SQL 語句了。

@Override public <T> T selectOne(String statement, Object parameter) { // Popular vote was to return null on 0 results and throw exception on too many. List<T> list = this.selectList(statement, parameter); if (list.size() == 1) { return list.get(0); } else if (list.size() > 1) { throw new TooManyResultsException('Expected one result (or null) to be returned by selectOne(), but found: ' + list.size()); } else { return null; } }

這里的 statement 雖然是 String 類型的,但是它并不是真正的 SQL 語句,它是一個尋找對應 MapperStatement 對象的名稱,在我們的例子中,它就是 test.HeroMapper.selectById ,Mybatis 通過這個名稱可以尋找到包含了 SQL 語句的對象。

我們跟蹤代碼的執行,最后會來到下面這個方法,這是一個包含三個參數的重載方法。

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

在第四行代碼中,可以得知它通過 statement 從 Configuration 對象中獲取了一個 MapperStatement 對象, MapperStatement 對象包含的信息是由 <select>、<update>、<delete> 、<insert> 元素提供的,我們在這些元素中定義的信息都會保存在該對象中,如:Sql 語句、resultMap、fetchSize 等等。

執行 SQL 語句

獲取到包含 SQL 語句信息的對象之后,就會交給 Execute 執行器對象去執行后續的處理,也就是 executor.query 方法。

@Overridepublic <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);}

獲取需要自行的 Sql 語句,然后創建一個緩存使用的 key,用于二級緩存。

@Overridepublic <E> List<E> query(MappedStatement ms, Object parameter, RowBounds rowBounds, ResultHandler resultHandler, CacheKey key, BoundSql boundSql) throws SQLException { // .... // 跟緩存有關,如果緩存中存在數據,則直接從緩存中返回,否則從數據庫中查詢 list = queryFromDatabase(ms, parameter, rowBounds, resultHandler, key, boundSql); return list;}

最后會執行到一個 doQuery 方法

@Overridepublic <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.query(stmt, resultHandler); } finally {closeStatement(stmt); }}

這段代碼創建了一個 Statement 對象的處理器 StatementHandler,這個處理器主要的工作就是完成 JDBC 中 PrepareStatement 對象的一些準備工作,包括:創建 PrepareStatement 對象,設置需要執行的 sql 語句,為 sql 語句中的參數賦值。完成這些工作之后,就開始從數據庫獲取數據了。

@Overridepublic <E> List<E> query(Statement statement, ResultHandler resultHandler) throws SQLException { PreparedStatement ps = (PreparedStatement) statement; ps.execute(); return resultSetHandler.handleResultSets(ps);}

第四行代碼即執行對應的 Sql 查詢,后續則是對結果進行處理。

總結

Mybatis 通過 MapperProxy 代理了我們的 Dao 接口類,以此來幫助我們執行預定義的 Sql 語句,通過 Cache 來緩存對應的執行結果,通過 StatementHandler 創建 PrepareStatement 對象,通過 jdbc 執行 SQL 操作。

到此這篇關于淺談MyBatis 如何執行一條 SQL語句的文章就介紹到這了,更多相關MyBatis 執行SQL語句內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!

標簽: Mybatis 數據庫
相關文章:
主站蜘蛛池模板: 999精品视频 | 国产精品高颜值在线观看 | 亚洲精品一区二区三区 | 色资源在线 | 日韩国产在线观看 | 久久毛片 | 天天操天天干天天爽 | 久久高清亚洲 | 久久网一区二区三区 | 国产日韩欧美在线 | 欧美国产日韩一区二区 | 国产精品27页 | 国产综合精品一区二区三区 | 成人精品一区二区 | 中文天堂av | 国产精品一区二区无线 | 欧美99| 黄色大片网站在线观看 | 欧美一区亚洲二区 | 免费a视频 | 午夜国产一级 | 成人黄色片在线观看 | 在线观看免费视频黄 | 久在线观看 | 国产精品久久久久久久久免费丝袜 | 国产91视频在线观看 | 人人草人人干 | 久久精品国产亚洲blacked | 欧洲毛片 | 日韩在线一区二区三区 | 97伦理电影网 | 国产成人免费视频网站高清观看视频 | 国产精品免费观看 | 欧美一级做a爰片免费视频 亚洲精品一区在线观看 | av免费黄色 | 国产乱码精品一区二区三区忘忧草 | h视频免费 | 日韩精品一区二区三区 | 久久久久久网站 | 电影午夜精品一区二区三区 | www.久久.com| 日韩中文字幕在线视频 | 国产在线不卡 | 国产淫片在线观看 | 亚洲欧美国产另类 | 日韩在线观看一区 | 欧美精品免费在线 | 国产成人精品一区二 | 久久亚洲一区 | 欧美成人一区二区三区片免费 | 国产精品综合视频 | 亚洲aⅴ| 精品伦精品一区二区三区视频 | 日韩欧美在线视频 | 欧美一级片免费播放 | 国产成人精品一区二区三区视频 | 国产精品国产精品国产专区不片 | 国产在线精品一区二区 | 精品亚洲一区二区三区在线观看 | 久久一| 日韩美女av在线 | 日本狠狠干| 成人影院网站ww555久久精品 | 亚洲精品久久久久久久久久久久久 | 日韩成人在线播放 | 亚洲一区二区三区 | 日韩6699人妻熟女毛片 | 日韩av在线中文字幕 | 在线观看一区 | 国产一区在线观看视频 | 四虎免看黄 | 日本一级在线观看 | 精品久久中文字幕 | 久久aⅴ乱码一区二区三区 91综合网 | 国产精品欧美一区二区三区 | 亚洲美女久久 | 成人a毛片 | 日韩精品一区二区在线观看视频 | 欧美一级在线 | 成人欧美一区二区三区在线播放 | 久久精品一 | 在线观看精品自拍私拍 | 国产综合久久久 | 美女精品视频在线 | 国变精品美女久久久久av爽 | 日韩在线视频一区 | 国产精品ssss在线亚洲 | 亚洲视频一区二区在线 | 日本在线视频不卡 | 欧美精品区| 国产精品久久久久久久久久久久久 | 冷水浴在线观看 | 99精品欧美一区二区三区综合在线 | 草久久av | 日本一区二区高清不卡 | 国产精品成人一区二区 | 成人在线www| 久久99国产一区二区三区 | 欧美日韩一区在线 | 亚洲一区二区三区免费在线观看 | 久久这里只有精品首页 | 精品不卡 | 国产成人精品一区二区 | 日本小视频网站 | 亚洲激情视频在线播放 | 日本理伦片午夜理伦片 | 超碰97人人人人人蜜桃 | 澳门av | 亚洲国产高清在线 | 午夜国产视频 | 一区二区三区免费 | 久久久久亚洲av毛片大全 | 在线成人亚洲 | 欧美日韩国产在线观看 | 天天干天天草 | 午夜影院免费观看 | 国产免费拔擦拔擦8x高清在线人 | 久久精品欧美一区二区三区不卡 | 国产精品成av人在线视午夜片 | 99久久99热这里只有精品 | 国产精品久久久久久一区二区三区 | 亚洲黄网在线观看 | heyzo在线观看 | 久在线视频播放免费视频 | 久久99一区二区 | 伊人二区| 国产精品亚洲一区 | 日韩欧美在线播放视频 | 日韩精品在线网站 | 国产精品国产三级国产a | 国产精品久久久久影院色老大 | 成人精品视频免费在线观看 | 国产精品成人一区二区 | 免费看黄色大片 | 日韩超级大片免费看国产国产播放器 | 97久久久国产精品 | 成人亚洲精品 | 国产精品一区二区三区四区 | 日本好好热视频 | 欧美日韩激情在线 | 北条麻妃一区二区三区在线 | 在线观看第一页 | 亚洲国产精品久久久久久久 | 成年人黄色免费视频 | 日韩精品一区在线 | 国产在线精品一区二区 | 久久福利| 精品国产91亚洲一区二区三区www | 久久爱电影 | 免费观看一区二区三区 | 色站综合| 在线国产一区 | 成人精品一区二区三区 | 国产人体视频 | 国产精品久久久久久久9999 | 欧美日韩国产一区二区 | 红色av社区 | 久久一精品 | 亚洲一区| 欧美午夜一区二区三区免费大片 | 中文字幕第一页在线 | 国产欧美日韩中文字幕 | av免费网站在线观看 | 在线免费黄色小视频 | 欧美精品导航 | 国产视频一区二区在线 | 欧美a网站| 亚洲精品国品乱码久久久久 | 亚洲一区日韩 | 91视频入口 | 亚洲视频综合 | 国产超碰人人模人人爽人人添 | 天天看片天天操 | 亚洲综合天堂网 | 久久r免费视频 | 中文字幕日韩视频 | 亚洲永久免费 | 久久韩剧网 | 久久久91精品国产一区二区三区 | 午夜免费剧场 | 99久久久成人国产精品 | 777色狠狠一区二区三区 | 亚洲国内精品 | 好姑娘影视在线观看高清 | 日韩有码在线播放 | 好看的一级毛片 | 一色屋精品久久久久久久久久 | 亚洲欧美在线免费 | 国产亚洲一区二区三区在线观看 | 国产精品久久久久久久 | 中文字幕一区二区三 | 看黄色.com| 国产一级免费视频 | 久久亚洲一区二区三区四区 | 国产xxx在线观看 | 天天看天天摸天天操 | 日韩精品一区二区三区在线 | 国产成人毛片 | 久久成人免费视频 | 一级毛片免费完整视频 | 亚洲毛片网站 | 综合网在线 | 亚洲午夜视频在线观看 | 99精品国产高清在线观看 | 综合 欧美 亚洲日本 | 亚洲精品在线播放 | 欧美日韩精品久久 | 精品国产污网站污在线观看15 | 91精品在线观看入口 | 成人超碰在线 | 欧美日韩精品综合 | 国产综合精品一区二区三区 | 精品亚洲视频在线 | 中文字幕 视频一区 | 国产精品成人国产乱一区 | 国产精品久久久久久亚洲调教 | 天天爽天天操 | 国产成人涩涩涩视频在线观看 | 日韩一区二区在线观看 | 91电影在线观看 | 国产一区二区在线播放 | 天堂va在线高清一区 | 一区二区观看 | 国产成人精品一区二区 | 欧美高清国产 | 成人国产精品久久久 | 久久精品免费一区二区三区 | 玖玖操| 亚洲精品成人免费 | 一级片在线观看视频 | 日韩精品999 | 中文字幕在线三区 | 美日韩精品 | 黄色一级毛片 | 欧美久久综合 | 精品色区 | 国产午夜手机精彩视频 | 日韩欧美国产精品综合嫩v 亚洲欧美日韩在线 | 欧美精品一区在线发布 | 久久精品网| 久久久一区二区三区 | 国产欧美一区二区精品忘忧草 | 日韩中文视频 | 一区二区日韩视频 | 欧美成人精品一区二区 | 国产主播久久 | 亚洲精品久久久一区二区三区 | 野狼在线社区2017入口 | 亚洲一区二区中文字幕在线观看 | 亚洲精品综合在线 | 国产精品入口久久 | 一区二区视频 | 国产精品91av | 九九热精 | 伊人网站 | 国产成人精品一区二区三区视频 | 精品免费视频 | 成人综合在线观看 | 91精品国产日韩91久久久久久 | 欧美精品激情 | 国内精品在线视频 | 欧美激情在线免费观看 | 黄p在线看 | 中文字幕在线观看的电影 | 亚洲免费人成在线视频观看 | 在线精品亚洲欧美日韩国产 | 1区在线| 精品一区二区三区在线观看 | 欧美一级二级视频 | 成人午夜免费网站 | 欧美日韩在线精品 | 狠狠入ady亚洲精品经典电影 | 国产日韩精品入口 | 国产精品毛片久久久久久久 | 国产精品久久久久久亚洲调教 | 日韩高清黄色 | 99精品欧美一区二区三区 | www..99热| 亚洲 欧美 日韩 精品 | 国产亚洲视频在线观看 | 欧美色图亚洲自拍 | 视频一区二区三区在线观看 | 日本一区二区三区四区 | 色69av| 亚洲日本欧美日韩高观看 | 欧美区在线 | 精品影院 | 国产不卡在线观看 | 亚洲精品久久久久久久久 | 国产在线不卡视频 | 久久久噜噜噜www成人网 | 日韩欧美在线视频 | 国产亚洲精品久久久 | 色综合免费视频 | 欧美狠狠操 | 青青草久久 | 国产精品久久久久一区二区三区 | 久久精品久久精品 | 国产中文视频 | 亚洲欧美在线观看 | 久国产精品 | 久久一区 | 91国内精品 | 在线观看日韩av | 成人在线一区二区 | 欧美成人一区二区三区片免费 | 国产成人小视频 | 91精品国产综合久久香蕉922 | 狠狠av| 色婷婷av久久久久久久 | 精品二区视频 | 国产91黄色 | 国产专区一区 | 中文字幕视频一区 | 91香蕉视频在线观看 | 在线视频亚洲 | 在线精品一区 | 精品自拍视频 | 美女国产 | 日韩欧美成人影院 | 亚洲一区免费视频 | 99精品热| 五月婷婷综合激情网 | 免费看的av | 欧美二区视频 | 免费观看av毛片 | 国产一区网站 | 久久精品日产高清版的功能介绍 | 亚洲精品美女在线观看 | 亚洲精品国产高清 | 中文字幕在线免费播放 | 国产精品一区二区三区免费 | 一区二区三区免费网站 | 亚洲精品久久久久久国产精华液 | 97视频在线 | 国产欧美在线观看 | 久久亚洲网| 精品久 | 精品一区二区视频 | 91九色在线观看 | 国产精品久久久久久久久 | 亚洲综合大片69999 | 精品欧美乱码久久久久久1区2区 | 国产精品久热 | 日韩不卡一区 | 亚洲精品视频在线 | 日韩在线视频中文字幕 | 欧美激情一区二区三区在线观看 | 亚洲国产一区二区三区四区 | 久久国产精品一区二区三区 | 日本一级毛片视频 | 精品视频一区二区三区 | 国产91麻豆视频 | 日本99精品| 成人爽a毛片一区二区免费 美女高潮久久久 | 在线中文字幕观看 | 日韩一区二区在线观看 | 成人免费毛片aaaaaa片 | 国产一区二区三区久久久久久久久 | 久久99这里只有精品 | 四虎中文字幕 | 色橹橹欧美在线观看视频高清 | 最近免费中文字幕大全免费版视频 | 日韩一区精品 | 欧美在线观看一区二区 | 男人天堂亚洲 | 欧美精品一区二区视频 | 日韩中文不卡 | 国产精品二区一区 | 91色视频在线观看 | 国产精品 日韩 | 男人阁久久 | 一区二区精品视频 | 欧美精品a∨在线观看不卡 欧美日韩中文字幕在线播放 | 日韩av在线中文字幕 | 麻豆freexxxx性91精品 | 国产精品久久久久精 | 亚洲一区二区三区四区五区中文 | 中文字幕精品三级久久久 | 成人在线免费 | 国产午夜精品美女视频明星a级 | 日韩欧美在线播放 | 久热精品国产 | www.国产精 | 亚洲欧美第一页 | 日韩国产欧美一区 | www婷婷 | 久国久产久精永久网页 | 日韩精品在线视频 | 一区二区视频 | 国产 欧美 日韩 一区 | 精品一区二区三区免费看 | 免费毛片一区二区三区久久久 | 国产乱码精品1区2区3区 | 九九热这里只有精品6 | 国产精品久久777777 | 簧片毛片| 欧美精品久久久 | 久久久999精品视频 五月天婷婷在线视频 | 欧美日韩第一页 | 99re国产| 亚洲午夜精品一区二区三区他趣 | 日韩免费一区二区 | 国产高清精品一区 | 亚洲欧美激情在线 | 在线成人国产 | 精品国产一区二区国模嫣然 | 日韩精品一区二区三区在线观看 | 久久精视频 | 成人久久精品 | www.久久.com | 黄在线看 | 午夜精品久久久久99蜜 | 最新免费av网站 | 欧洲美女性开放视频 | 97精品 | 91精品久久久久久久久 | 91精品久久久久久久久久小网站 | 国产999精品久久久久久麻豆 | h片在线免费观看 | 久久福利 | 欧美精品激情 | 亚洲综合激情网 | 夜夜草av | 欧美视频精品 | 午夜影院在线免费观看 | 亚洲欧美福利视频 | av在线一区二区 | 少妇激烈床戏视频 | 成人免费视频观看视频 | 日操视频 | 国产精品自产拍在线观看 | 天天曰 | 亚洲三区在线观看 | 欧美高清hd | 国产日韩一区二区三区 | 欧美日韩亚洲综合 | 欧美激情欧美激情在线五月 | 亚洲一区二区免费在线观看 | 国产一区二区黑人欧美xxxx | 久久久蜜桃| 久热精品视频在线播放 | 亚洲黄色免费观看 | 亚洲二区在线 | 国产日韩亚洲欧美 | 亚洲视频一区二区在线 | 91在线 | 亚洲| 九九色综合 | 91在线视频一区 | 91影院 | 欧洲美女7788成人免费视频 | 亚洲综合色视频在线观看 | 日韩午夜在线视频 | 午夜窝窝 | 中文字幕亚洲欧美日韩在线不卡 | 欧美激情欧美激情在线五月 | 97成人精品视频在线观看 | 亚洲三级在线 | 在线视频二区 | 久久人人爽爽爽人久久久 | 在线视频中文字幕 | 亚洲国产成人久久一区二区三区 | 综合亚洲精品 | 国产精品久久久久久久久免费桃花 | 久久9色| 国产精品色 | 女人夜夜春 | 国产美女中出 | 亚洲综合无码一区二区 | 亚洲电影在线观看 | 夜夜爽99久久国产综合精品女不卡 | 国产探花在线精品一区二区 | 一级在线| 91午夜在线| 欧美久久久久久久久久 | 亚洲视频在线看 | av资源中文在线 | 欧美一区二区三区黄色 | 日韩欧美一级 | 蜜桃中文字幕 | 欧美一级艳片视频免费观看 | 丝袜美腿一区二区三区 | 日韩在线免费 | 黄色一级片黄色一级片 | 精品伦精品一区二区三区视频 | 久久精品久久综合 | 久久久www成人免费精品 | 中文字幕精品视频在线观看 | 国产精品一区二区三区在线播放 | 在线成人免费视频 | 中文字幕在线观看第一页 | 成人h在线 | 国产一区二区精品久久 | 精品久久精品久久 | 亚洲啊v在线 | 欧美专区在线 | 久久久久久亚洲国产 | 91丝袜| 国产v日产∨综合v精品视频 | 亚洲成人第一 | 天天操天天拍 | 精品国产91亚洲一区二区三区www | 欧美日韩国产在线播放 | 国产精品久久久久久久久免费桃花 | 美女在线国产 | 亚洲男人网 | 中文字幕免费在线 | 亚洲国产精品成人 | 黄视频在线播放 | 日韩在线免费 | 日日搞夜夜操 | 狠狠干天天干 | 91精产国品一二三区在线观看 | 国产69精品99久久久久久宅男 | 国产精品久久久久久久久免费高清 | 国产精品久久久久久久久免费丝袜 | 红色av社区 | 日韩精品av一区二区三区 | 影视在线观看 | 精品电影| 97精品 | 一区在线视频 | 欧美视频一区二区三区在线观看 | 午夜日韩 | 91精品久久久久久久久久 | 看毛片网站 | 精品久久一区二区三区 | 国产欧美精品一区二区三区 | 免费观看一区二区三区毛片软件 | 日日摸天天爽天天爽视频 | 999在线视频免费观看 | 国产精品一区av | 天天干女人网 | 狠狠夜夜| 日韩电影一区二区在线观看 | 毛片大全| 亚洲成人精品区 | 欧美二区在线 | 国产三级一区二区 | 精品国产伦一区二区三区观看说明 | 一区二区三区的视频 | 欧美日韩精品一区 | 伊人久久国产 | 欧美一区二区在线视频 | 97人人看| 欧美精品一区在线 | 天天操天天草 | 精品国产31久久久久久 | 欧美视频三区 | 国产精品一区二区三区免费视频 | 久久久精品 | 久色视频在线观看 | 99免费在线观看视频 | 久久久久久一区 | 一级片欧美 | 一区二区三区四区免费观看 | 美女张开腿视频网站免费 | 国产欧美日韩一区 | 欧美日韩视频一区二区 | 中文字幕一区二区三区四区 | 国产日韩在线视频 | 在线观看免费视频a | www天天干 | 精品欧美黑人一区二区三区 | 精品国产区 | 久久亚洲一区 | 男女网站视频 | 国产成人在线视频 | 久久精精品 | 一级片在线观看 | 男人的天堂在线视频 | 午夜欧美一区二区三区在线播放 | 成人动慢| 1000部精品久久久久久久久 | 国产成人高清在线 | 国产在线一区二区三区 | 欧美一区在线看 | 成人日韩在线观看 | 成人毛片在线视频 | 日韩在线播放一区二区三区 | 亚洲一区成人 | 2021最新热播中文字幕-第1页-看片视频 青青青久草 | 999在线观看精品免费不卡网站 | 91精品国产色综合久久 | 日韩一区在线观看视频 | 色婷婷久久 | 国产精品久久久久久久午夜片 | 欧美成人精品在线观看 | 一级色网站| 久久久精品电影 | 山岸逢花在线观看无删减 | 亚洲高清在线观看视频 | 成年人看的羞羞网站 | 91久久久精品视频 | 国产探花在线看 | 亚洲精品久久久久久久久久久 | 99精品欧美一区二区三区综合在线 | 新99热| 天堂在线中文字幕 | 亚洲三区在线观看 | 国产一级一级特黄女人精品毛片 | 国产精品久久久久久久久免费丝袜 | 精品伦精品一区二区三区视频 | 久草免费福利 | 久久久久久成人 | 国产成人精品免高潮在线观看 | 国产精品久久久久久久岛一牛影视 | 亚洲精品网址 | 欧美在线二区 | 国产中文字幕一区二区三区 | 国产精品视频一区二区免费不卡 | 久久精选视频 | 久久精品国产免费 | 久久精品国产一区二区三区不卡 | 欧美视频精品在线 |