黄a在线观看-黄a在线-黄a大片-黄色片在线看-黄色毛片免费-黄色大片网站

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

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

瀏覽:80日期: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 數據庫
相關文章:
主站蜘蛛池模板: 亚洲制服丝袜一区二区三区 | 婷婷中文网 | 欧美黄色免费观看 | 久久久一本精品99久久精品66 | 免费大片av | 中文字幕av高清 | www.中文字幕在线观看 | 女女百合高h喷汁呻吟视频 女女百合国产免费网站 | 一区二区三区四区在线视频 | 香蕉成人啪国产精品视频综合网 | 亚洲最新在线视频 | 国产综合免费视频 | 欧美人与动性行为视频 | 熟妇的奶头又大又长奶水视频 | 亚洲网视频 | 五月综合色婷婷 | 亚洲男人的天堂在线播放 | 欧美日韩一二三四区 | 男人激烈吮乳吃奶视频 | 人人草网站 | 色综合av综合无码综合网站 | 久久成人人人人精品欧 | 777亚洲 | 男人天堂亚洲天堂 | 久久久丁香 | 欧美日本一区二区 | 成人小视频在线免费观看 | 大陆熟妇丰满多毛xxxx | 中文字幕免费 | 成人污污视频在线观看 | 国产一区二区视频播放 | 视频这里只有精品 | 午夜又黄又爽 | 97色伦图片 | 91久久精品一区二区二区 | 丁香色欲久久久久久综合网 | 天天精品免费视频 | 亚洲熟妇av一区二区三区宅男 | 欧洲亚洲激情 | 疯狂做受xxxx高潮视频免费 | 五月激情婷婷在线 | 男女下面进入的视频免费午夜 | 国产一区日本 | 国产美女视频91 | 99精产国品一二三产区区别麻豆 | 国产高清一区在线观看 | 精品美女一区 | h网站在线播放 | 日韩最新网址 | 99年国精产品一二二区传媒 | 精品人无码一区二区三区 | 先锋影音av最新资源 | 成年人视频在线看 | 美女男女激情晚上看 | 黄大色黄女片18第一次 | 98婷婷狠狠成人免费视频 | www.日日干| 午夜熟女毛片蜜桃传媒 | 水蜜桃亚洲一二三四在线 | 国产一区二区三区撒尿在线 | 大波大乳videos巨大 | 太深太粗太爽太猛了视频免费观看 | 国产一起色一起爱 | 黄色高潮视频 | 日韩av无码国产精品 | 国产欧美精品一区二区三区-老狼 | a级黄色影院 | 岛国av在线免费 | 日韩毛片无码永久免费看 | 免费看少妇作爱视频 | yp在线观看视频网址入口 | 狠狠色综合久久婷婷 | 国产综合图区 | 九九精品影院 | 欧美视频在线播放 | av黄色在线看 | 午夜成人鲁丝片午夜精品 | 91国偷自产一区二区三区蜜臀 | 国内精品久久久久精免费 | 久久久久久久久久久久国产精品 | 成人欧美一区二区三区黑人 | 欧美影院成年免费版 | 日本黄网站三级三级三级 | 欧洲s码亚洲m码精品一区 | 老熟女 露脸 嗷嗷叫 | 台湾佬中文娱乐22vvvv | 欧美大片在线免费观看 | 国产ts变态重口人妖hd | 亚洲爆乳精品无码一区二区三区 | 国产zzjjzzjj视频全免费 | 乱码午夜-极国产极内射 | 免费美女毛片 | 国模大尺度啪啪 | 男男羞羞视频网站国产 | av网站亚洲 | 美日韩av在线播放 | 国产乱淫av一区二区三区 | 99精品国产免费久久久久久按摩 | 国内精品91 | 亚洲人成未满十八禁网站 | av网站在线不卡 | 操碰视频| 免费看片黄色 | 日日狠狠久久8888偷偷色 | 五月激情综合 | 午夜不卡av| av黄色影院| 国产午夜福利短视频 | 香蕉视频入口 | 国产成人精品一区二三区在线观看 | 91超碰在线观看 | 国产成人精品免费视频 | 一个人看的视频在线观看www | 亚洲一级视频在线观看 | 亚洲激情影院 | 成人手机在线免费视频 | 97热视频| 国产肉体xxxx裸体784大胆 | 中国少妇做爰全过程毛片 | 97碰碰碰免费公开在线视频 | 麻豆一区二区三区蜜桃免费 | 国产中文字幕在线视频 | 亚洲国产一区二区天堂 | 国产一区亚洲二区三区 | 亚洲精品无码专区 | 一区二区三区四区视频 | 欧美日韩1区 | 国产精成人品日日拍夜夜免费 | 99久久久精品免费观看国产 | 狠狠干少妇 | 欧美日韩一区二区三区69堂 | www超碰久久com | 136fldh导航福利视频 | 亚洲欧洲无码一区二区三区 | 黄色正能量网站 | 在线成人免费观看www | 日本少妇一级 | 综合色站导航 | 九九久久精品国产免费看小说 | 狠狠色噜噜狠狠狠狠97首创麻豆 | 国产成人精品在线播放 | 99热一区二区 | 美女精品一区 | 亚洲成色 | 少妇高潮一区二区三区 | 凹凸在线无码免费视频 | 日韩精品一区二区在线观看 | 国产极品粉嫩 | 国产精品天干天干 | 在线观看国产小视频 | 18视频在线观看男男 | 激情久久av一区av二区av三区 | 小h片免费观看久久久久 | 欧美一区二区三区四区五区 | 国产精品国产三级国产潘金莲 | 一本一道av | 超碰97国产| 欧洲一区二区三区四区 | 色姑娘天天操 | 日韩欧美国产三级 | 16一17女人毛片 | 国产精品久久久久久久久福交 | 国产高清视频在线 | 99久久伊人精品综合观看 | 国产精品无码久久av | 亚洲欧洲中文字幕 | 日本一区二区三区精品 | 欧美xxxx性 | 小sao货水好多真紧h无码视频 | 少妇做爰免费视看片 | 国产一区二区视频播放 | 欧美老妇牲交videos | 国内精产品∪v | 女女互揉吃奶揉到高潮视频 | 国产啊v在线观看 | 亚洲a∨精品一区二区三区 gv天堂gv无码男同在线观看 | 国产精品偷伦视频免费还看的 | 日本美女a级片 | 中文字幕一区二区三区乱码图片 | 欧美大片www| 国产一级免费片 | 99热这里有精品 | 亚洲乱码中文字幕久久孕妇黑人 | 少妇黄色一级片 | 青草超碰 | 女人被狂躁到高潮视频免费软件 | 男女做性无遮挡免费视频 | 性一交一乱一乱一乱视频 | 男ji大巴进入女人的视频小说 | 免费动漫av | www亚洲一区 | 九九热在线视频观看这里只有精品 | 国产成人亚洲精品自产在线 | 精品国产一区二区三区av片 | 国产在线a视频 | 成人看的羞羞视频免费观看 | 牛牛a级毛片在线播放 | 全部免费毛片在线播放一个 | 三级久久久 | 久久综合a∨色老头免费观看 | 国产精品一区免费 | 精品国产自在精品国产 | 九九热在线视频免费观看 | 极品少妇被啪到呻吟喷水 | 中文字幕成人网 | 99久久久国产 | 成年女人黄网站色视频免费97 | 最新激情网 | 直接观看黄网站免费视频 | 国产精品午夜无码av体验区 | 久久99精品久久久久久 | 天天插天天操 | 99久久久无码国产精品古装 | 日韩成av人片在线观看 | 男人激烈吮乳吃奶视频 | 91无限观看 | 亚洲精品乱码久久久久久蜜桃图片 | 亚洲黄网站wwwwwwwww | 欧美激情黑白配 | 99精品免费久久久久久久久日本 | 精品国产va久久久久久久冰 | 日韩精品一区二区三区视频播放 | a天堂v| 激情综合网五月婷婷 | av导航网址 | 中国丰满熟妇xxxx性 | 久久久妻 | 日本激情小视频 | 国产视频手机在线观看 | 老女人人体欣赏a√s | 李丽珍裸体午夜理伦片 | 无码h黄动漫在线播放网站 国产精品高潮露脸在线观看 | 亚洲欧洲av综合色无码 | 久久久精品人妻久久影视 | 黄瓜视频在线观看 | 欧美精品1区 | 成人高潮片免费视频欧美 | 亚洲成人久 | 中文字幕国产专区 | 国产欧美一级二级三级在线视频 | 大黑人交xxxx18视频 | 特级黄一级播放 | 中国xxxxx性按摩店 | 精品久久久一二三区播放播放播放视频 | 视频福利在线 | 欧洲熟妇色xxxx欧美老妇多毛 | 国产一及毛片 | 欧洲精品国产 | 国产精品入 | 粗大黑人巨精大战欧美成人 | 久久婷婷五月综合色丁香 | 欧美深夜福利 | 毛片视频大全 | www国产黄色| 国产午夜三级一二三区 | 精品无人乱码一区二区三区 | 免费人成视频在线 | 吃奶揉捏奶头高潮视频在线观看 | 成人免费毛片片v | 色播亚洲视频在线观看 | 18禁黄久久久aaa片广濑美月 | 夜夜躁狠狠躁日日躁2021日韩 | 中国大陆一级片 | 理论黄色片 | 无码专区人妻系列日韩精品少妇 | 久草在线手机视频 | 淫片在线| 国产v亚洲v天堂无码 | 国产二区三区在线 | 最新成人 | www春色 | 99国产欧美久久久精品蜜芽 | 欧美超级乱婬视频播放 | 9l视频自拍九色9l视频大全 | 三个男吃我奶头一边一个视频 | 538任你躁在线精品免费 | 欧美日韩高清 | 精品久久中文字幕97 | a级片久久 | 五月天色丁香 | www色五月| hd国产人妖ts另类视频 | 亚洲精品一区二区三区麻豆 | 久久成人欧美 | 亚洲国产精品网站 | 特级做a爰片毛片免费看无码 | 免费成人进口网站 | av资源在线看 | 日本人麻豆 | 亚洲国产精品成人女人久久 | 亚洲精品久久久狠狠爱小说 | 国产a级精品毛片 | 国产成人亚洲综合无码99 | 无码国产乱人伦偷精品视频 | 久久精品成人一区二区三区蜜臀 | 么公的好大好硬好深好爽视频 | 国产真人无遮挡作爱免费视频 | 四虎国产成人永久精品免费 | 99久久人人爽亚洲精品美女 | 日韩精品一卡二卡 | 高h公妇烈火 | 久久黄色av| 狠狠色噜噜狠狠狠狠97俺也去 | 国产伦精品一区二区三区免费优势 | 黄色毛片黄色毛片 | 狠狠干综合网 | 男人的天堂av网 | 日本女优网址 | 日韩va在线观看 | 亚洲论理 | 女人性做爰100部免费 | 制服丝袜一区二区三区 | 国产特级毛片aaaaaaa高清 | 俄罗斯毛片基地 | 在线视频免费观看你懂的 | 亚洲一区二区三区影院 | 久久久噜噜噜久久久 | 欧美hdse | 奇米7777狠狠狠琪琪视频 | 五月天黄色小说 | 亚洲成人美女xvideos | 97视频人人| 日韩 在线 中文 制服一区 | 欧美日韩国产麻豆 | 成在线人视频免费视频 | 特级黄色一级片 | 日日好av | 亚洲精品久久久久国产 | 青青草精品 | 在线精品国产成人综合 | 国产精品人妻一码二码尿失禁 | www精品美女久久久tv | 亚洲欧洲精品在线 | 欧美成人一区二免费视频软件 | 99精品视频国产 | 国产精品xxxx喷水欧美 | 日日躁夜夜躁狠狠躁aⅴ蜜 日日躁夜夜躁狠狠躁超爽2001 | 综合久久综合久久 | 韩国三级做爰视频 | 亚欧乱色| 超污视频在线观看 | tube中国91xxxxx国产 | 男人午夜天堂 | 欧美成人久久久免费播放 | 白嫩日本少妇做爰 | 亚洲日本中文字幕 | 好男人蜜桃av久久久久久蜜桃 | 东京热久久综合久久88 | 隔壁邻居是巨爆乳寡妇 | 欧美精品日韩在线 | 暖暖在线日本免费高清最新版 | 一本一道久久a久久 | 欧美浮力影院 | 亚洲精品欧美激情 | 亚洲图片 自拍偷拍 | 欧美性受xxxxxx黑人xyx性爽 | 欧美人妻精品一区二区三区 | 精品一区二区久久久久久久网站 | 午夜精品视频在线 | 大乳三级a做爰大乳 | 国产精品色婷婷99久久精品 | 天天弄天天模 | 国精产品一品二品国精品69xx | 国产nv在线观看 | 毛片最新网址 | 久久逼逼 | 国产乱子伦一区二区三区四区五区 | 成人欧美一区二区三区黑人免费 | 天天躁狠狠躁狠狠躁夜夜躁68 | 人妻精品无码一区二区三区 | avhd101高清在线迷片麻豆 | 99久久九九 | 一区二区三区在线免费视频 | 亚洲人吸女人奶水 | 日本少妇丰满大bbb的小乳沟 | 亚洲成成品牛牛 | av在线一区二区三区四区 | 中文字幕视频一区 | 免费大片黄在线观看视频网站 | 一级国产精品一级国产精品片 | 狠狠躁18三区二区一区ai明星 | 91精品国产乱码麻豆白嫩 | 日本大尺度吃奶呻吟视频 | 亚洲国产一线二线三线 | 亚洲精品国产乱码在线看蜜月 | 免费无码一区二区三区a片百度 | 国产成人综合在线观看不卡 | 亚洲一区二区三区中文字幕 | 爽成人777777婷婷 | 国产特黄aaa大片免费观看 | 国产黄色av片 | 欧美性大战久久久久久久 | 殴美性生活 | 一本大道色婷婷在线 | 成人网在线播放 | 中国浓毛少妇毛茸茸 | 亚洲国产成人在线视频 | 午夜视| 欧美日韩一级大片 | 自拍视频一区二区三区 | 国产成人a亚洲精v品无码 | 国产91在线免费观看 | 亚洲一区二区三区免费视频 | 99精品视频在线观看 | 精品国产一区在线观看 | 精品一区视频 | 亚洲精品一区二区三区98年 | 狼人狠狠干| 日韩 欧美 亚洲 精品 少妇 | 九九九视频在线观看 | 日本做爰全过程免费看 | 国内精品久久久久久久久久久 | 中国china露脸自拍性hd | 日韩av中文字幕在线免费观看 | 内射爽无广熟女亚洲 | 日韩人妻无码一区二区三区综合部 | 成年人在线观看av | 国产精品久久二区二区 | 男人天堂伊人 | 麻豆av在线播放张芸熙 | 国产精品久久久久久亚洲 | 久久影视一区 | 夜夜嗨av一区二区三区免费区 | 久久国产夫妻 | 三级a级片| 变态孕妇孕交av免费看 | 久草在线视频在线 | 欧美精品第二页 | 95久久| 久久av免费看 | 亚洲精品乱码久久久久久金桔影视 | 国产成人一区二区三区在线观看 | 寂寞少妇色按摩bd | 成人av影视在线观看 | 视频精品久久 | 噼里啪啦免费观看 | 青青视频网| 九九九久久久 | 一区成人 | 天海翼一区二区三区四区在线观看 | 在线免费观看网站入口在哪 | 看黄网站在线观看 | 日本www在线观看 | youporn免费视频成人软件 | 亚洲春色av | 成人午夜视频精品一区 | 日韩欧美国产片 | 第四色激情 | 国产r级在线观看 | 成年人免费看黄 | 国产午夜亚洲精品不卡下载 | 韩国19禁无遮挡啪啪无码网站 | 天天综合天天操 | 亚洲精品在线免费观看视频 | 亚洲少妇激情 | 亚洲黄v| julia一区二区在线播放 | 夜色福利站www国产在线视频 | 成人激情站 | 91视频黄版| 国产爽爽久久影院hd | 国产精品亚洲一区二区三区在线 | 成人淫片免费视频95视频 | www.在线观看av | 久久久精品波多野结衣av | 吞精囗交系列av | 亚洲妇熟xxxx妇色黄 | 色黄视频在线观看 | 黄色免费一级 | 国产aaa | 91丨九色丨刺激 | 国产资源网 | 一本a道新久花碟 | 亚洲一区二区三区精品动漫 | 日本特黄 | 亚洲一区二区三区高清av | 亚洲a免费 | 久草在线新时代视觉体验 | 久久r精品| 亚洲欧美日韩综合久久久 | 久久久视 | 中文一区二区在线观看 | av福利在线免费观看 | 自拍av在线 | 在线播放免费人成毛片乱码 | 极品美女穴 | 后进极品白嫩翘臀在线视频 | 日本黄色生活片 | 国产精品无码制服丝袜 | 污视频在线播放网站 | 国产黄色片视频 | 免费看一级黄色大全 | 国产精品亚洲五月天高清 | 国産精品久久久久久久 | 亚洲天堂久久 | 红桃视频一区 | 午夜精品久久久久久久91蜜桃 | 国产精品久久久久久久久久免 | 日本免费一区二区三区视频观看 | 天堂男人网 | 在线观看日批视频 | 狠狠躁夜夜躁人人躁婷婷91 | 日本免费黄色小视频 | 五月婷婷丁香激情 | 情趣内衣a∨片在线观看 | 久久美| 超碰综合网 | 毛片在线免费观看网址 | videosex抽搐痉挛高潮 | 欧美日韩免费观看视频 | 日本福利视频一区 | 97人妻天天摸天天爽天天 | 天堂中文在线8最新版精品版软件 | 亚洲精品午夜久久久久久久灵蛇爱 | 亚洲午夜18毛片在线看 | 欧美专区在线播放 | 三级成人在线 | 国语自产精品视频在线看 | 精品对白一区国产伦 | 久久亚洲精| 欧美视频四区 | 欧美裸体性生活 | 你懂的网址在线播放 | 特黄特色大片免费视频观看 | 最近中文字幕在线 | 国产成人av三级在线观看 | 野狼av午夜福利在线 | 亚洲一区二区三区香蕉 | av老司机久久 | 都市激情av | 国产精品网站在线观看免费传媒 | 国产经典三级 | 毛茸茸绝色孕妇孕交 | 成人性做爰av片免费看 | 日本特黄一级片 | 北条麻妃一区二区三区av高清 | 91色在线视频 | 日韩在线视 | 亚洲欧美日韩中文久久 | 狠狠色综合7777久夜色撩人ⅰ | 久久久久99精品成人片三人毛片 | 直接看的av网站 | 久久久久久久久久一毛喷水 | 99热在线免费观看 | 亚洲图片欧美日韩 | 欧美精品久久天天躁 | 一品二品三品中文字幕 | 国产精品污www在线观看17c | 3atv精品不卡视频 | 风韵丰满熟妇啪啪区老熟熟女 | 国产女主播在线一区二区 | 日韩激情毛片 | 99久久国产综合精品麻豆 | 久草在线视频精品 | 色噜噜狠狠色综合中国 | 国产福利91精品 | 91橘梨纱中出体验在线观看 | 色99999| 日本内射精品一区二区视频 | 国产综合久久 | 屁屁影院一区二区三区 | 久久爱网| 97国产在线观看 | 欧美一级性 | 国产片网址 | 一二三区在线 | 第一av| 午夜精品久久久久久毛片 | 欧美激情一区二区三区蜜桃视频 | 精品香蕉99久久久久网站 | 香港三级毛片 | 日韩av无码中文无码不卡电影 | 北京少妇宾馆露脸对白 | 欧美日韩一区二区三区视频 | 日韩美女一区 | 最近最新中文字幕 | 亚洲中文字幕无码一区二区三区 | 日本激情在线 | 成人激情视频 | 91精品综合久久久久m3u8 | 中文字幕亚洲精品无码 | 日本xxxx18高清hd | 免费观看av网址 | 91成人在线观看喷潮蘑菇 | 北条麻妃在线一区二区韩世雅 | 日韩高清av| 中文字幕久久熟女人妻av免费 | 人妻有码av中文字幕久久琪 | 色呦呦在线观看视频 | 尤物视频在线 | 五月天在线观看 | 中国人妻被两个老外三p | 日韩激情一区 | 好吊妞视频在线观看 | 亚洲乱码国产乱码精品精大量 | 日韩色资源 | 女性向av片在线观看免费 | 中文av一区二区三区 | 欧美精品偷拍 | 欧美人xxxx | 欧美一级大片在线观看 | 久久免费大片 | 香港三级韩国三级日本三级 | 东北老女人高潮久久91 | 激情小说一区 | 长腿校花无力呻吟娇喘的视频 | 国产精品国产三级国产普通话99 | 91岛国| 波多野结衣视频在线播放 | 精品久久久久久久久久久国产字幕 | 一本一道波多野结衣av中文 | 日本大人吃奶视频xxxx | 亚洲精品一二三区 | 自拍亚洲国产 | 亚洲视频一二区 |