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

您的位置:首頁(yè)技術(shù)文章
文章詳情頁(yè)

Mybatis 實(shí)現(xiàn)動(dòng)態(tài)組裝查詢條件,仿SQL模式

瀏覽:90日期:2023-10-19 10:12:38
目的:

以前比較習(xí)慣使用Hibernate,后來(lái)覺(jué)得mybatis不能按我想要的自動(dòng)組裝為SQL查詢條件,所以提供該工具類;

效果圖:

Mybatis 實(shí)現(xiàn)動(dòng)態(tài)組裝查詢條件,仿SQL模式

Mybatis 實(shí)現(xiàn)動(dòng)態(tài)組裝查詢條件,仿SQL模式

如圖所示,根據(jù)條件自動(dòng)組裝查詢條件,下面來(lái)說(shuō)一下實(shí)現(xiàn)方法:

1. ServiceImpl書(shū)寫注意項(xiàng)

Page<SysLogin> resultPage = null;try { PageHelper.startPage(pager.getCurrentPage(), pager.getPageSize()); // 判斷是否有分頁(yè) if (ObjectHelper.isNotEmpty(pager.getDirection()) && ObjectHelper.isNotEmpty(pager.getProperties())) {specification.addOrderBy(pager.getProperties(),pager.getDirection()); } // 判斷是否存在邏輯刪除篩選 String sqlStr = specification.sql(); if (sqlStr.indexOf('deleted') == -1) {specification.eq('deleted', '0'); } resultPage = this.sysLoginMapper.page(specification.sql());} catch (Exception e) { result = Result.newFailure('數(shù)據(jù)錯(cuò)誤', '在獲取分頁(yè)列表時(shí)發(fā)生異常。'); log.error(SimpleLogFormater.formatException(result.getMessage(), e)); return result;}2. Mapper.java 書(shū)寫

查詢條件非Map對(duì)象,直接就是SQL語(yǔ)句了;

/** * 分頁(yè)查詢數(shù)據(jù) * * @return */ Page<T> page(String sqlStr);3. 關(guān)于XML的配置,會(huì)拼裝SQL語(yǔ)句

Mybatis 實(shí)現(xiàn)動(dòng)態(tài)組裝查詢條件,仿SQL模式

附:SQL拼裝工具類

/** * @Description: (用一句話描述該文件做什么) * @author heliang * @date 2018-7-6 下午6:43:42 * @version V2.1 */package com.onem2.base.common;import com.onem2.base.helper.ObjectHelper; /** * @ClassName: Specification * @Description: (這里用一句話描述這個(gè)類的作用) * @author heliang * @date 2018-7-6 下午6:43:42 * @version V2.1 * Update Logs: * Name: * Date: * Description: 初始化 */ public class Specification { private StringBuilder where = new StringBuilder(); private String groupBy; private String having; private String orderBy; public StringBuilder getWhere() {return where; } public void setWhere(StringBuilder where) {this.where = where; } public String getGroupBy() {return groupBy; } public void setGroupBy(String groupBy) {this.groupBy = groupBy; } public String getHaving() {return having; } public void setHaving(String having) {this.having = having; } public String getOrderBy() {return orderBy; } public void setOrderBy(String orderBy) {this.orderBy = orderBy; } public Specification addOrderBy(String sort, String order) {if (!isEmpty(sort) && !isEmpty(order)) { this.orderBy = ObjectHelper.underscoreName(sort) + ' ' + order;}return this; } public Specification orLike(String value, String columns) {if (!isEmpty(value)) { StringBuffer strBuf = new StringBuffer(''); for (String column : columns.split(',')) {strBuf.append(ObjectHelper.underscoreName(column) + ' like ’%'+ value + '%’ or '); } String orLikeStr = strBuf.substring(0, strBuf.lastIndexOf('or')); where.append(' and (' + orLikeStr + ')');}return this; } public Specification eq(String column, String value) {if (!isEmpty(value)) { where.append(' and ' + ObjectHelper.underscoreName(column) + ' = ’' + sqlParam(value) + '’');}return this; } public Specification ne(String column, String value) {if (!isEmpty(value)) { where.append(' and ' + ObjectHelper.underscoreName(column) + ' != ’' + sqlParam(value) + '’');}return this; } public Specification like(String column, String value) {if (!isEmpty(value)) { where.append(' and ' + ObjectHelper.underscoreName(column) + ' like ’%' + sqlParam(value) + '%’');}return this; } public Specification notLike(String column, String value) {if (!isEmpty(value)) { where.append(' and ' + ObjectHelper.underscoreName(column) + ' not like ’%' + sqlParam(value) + '%’');}return this; } public Specification in(String column, String... values) {if (!isEmpty(values)) { where.append(' and ' + ObjectHelper.underscoreName(column) + ' in (' + inValuesString(values) + ')');}return this; } public Specification notIn(String column, String... values) {if (!isEmpty(values)) { where.append(' and ' + ObjectHelper.underscoreName(column) + ' not in (' + inValuesString(values) + ')');}return this; } public Specification gt(String column, String value) {if (!isEmpty(value)) { where.append(' and ' + ObjectHelper.underscoreName(column) + ' > ’' + sqlParam(value) + '’');}return this; } public Specification gte(String column, String value) {if (!isEmpty(value)) { where.append(' and ' + ObjectHelper.underscoreName(column) + ' >= ’' + sqlParam(value) + '’');}return this; } public Specification lt(String column, String value) {if (!isEmpty(value)) { where.append(' and ' + ObjectHelper.underscoreName(column) + ' < ’' + sqlParam(value) + '’');}return this; } public Specification lte(String column, String value) {if (!isEmpty(value)) { where.append(' and ' + ObjectHelper.underscoreName(column) + ' <= ’' + sqlParam(value) + '’');}return this; } public Specification between(String column, String from, String to) {if (isEmpty(from) && isEmpty(to)) { return this;}if (isEmpty(to)) { where.append(' and ' + ObjectHelper.underscoreName(column) + ' >= ’' + sqlParam(from) + '’');} else if (isEmpty(from)) { where.append(' and ' + ObjectHelper.underscoreName(column) + ' <= ’' + sqlParam(to) + '’');} else { where.append(' and ' + ObjectHelper.underscoreName(column) + ' between ’' + sqlParam(from) + '’ and ’' + sqlParam(to) + '’');}return this; } public String sql() {StringBuilder sql = new StringBuilder('');final int a = 4;final int b = 5;if (where.length() > a) { sql.append(' ' + where.substring(b));}if (!isEmpty(groupBy)) { sql.append(' group by ' + groupBy);}if (!isEmpty(having)) { sql.append(' having ' + having);}if (!isEmpty(orderBy)) { sql.append(' order by ' + orderBy);}return sql.toString(); } public String toString() {return sql(); } private static boolean isEmpty(String value) {return value == null || ''.equals(value) || value.trim().length() == 0; } private static boolean isEmpty(String[] values) {if (values == null || values.length == 0) { return true;}for (String value : values) { if (!isEmpty(value)) {return false; }}return true; } private static String inValuesString(String[] values) {StringBuilder string = new StringBuilder();for (String value : values) { if (isEmpty(value)) {continue; } string.append(’’’); string.append(value); string.append(’’’); string.append(’,’);}if (string.length() > 0) { string.deleteCharAt(string.length() - 1);}return string.toString(); } private static String sqlParam(String sqlParam) {return sqlParam.replaceAll('([’;]+|(--)+)', ''); }}

附:ObjectHelper 工具源碼:

package com.onem2.base.helper;import java.util.ArrayList;import java.util.Collection;import java.util.List;/** * Object幫助類 功能:此類提供處理 <Object>對(duì)象一系列方法 * * @author 賀亮 * */public class ObjectHelper { /** * 將id數(shù)組轉(zhuǎn)換為id集合 * * @param ids * @return */ public static List<Long> initIds(String[] ids) {List<Long> list = new ArrayList<Long>();list.add(-1L);for (int i = 0; i < ids.length; i++) { list.add(Long.valueOf(ids[i]));}return list; } /** * 組裝條件 * * @param str * @return */ public static List<String> strToList(String str) {if (isEmpty(str)) { return null;}String[] strs = str.split(',');List<String> list = new ArrayList<String>();for (int i = 0; i < strs.length; i++) { list.add(strs[i]);}return list; } /** * 判斷這個(gè)Object是否為Null或長(zhǎng)度為0 * * @param obj * @return */ public static boolean isEmpty(Object obj) {if (obj == null) { return true;}if (obj instanceof Collection) { return ((Collection<?>) obj).isEmpty();} if (obj instanceof String) { return ((String) obj).equalsIgnoreCase('null') | ((String) obj).trim().toString().equals('');} if (obj instanceof StringBuffer) { return ((StringBuffer) obj).length() == 0;} if (obj.getClass().isArray()) { try {Object[] a = (Object[]) obj; boolean b = true;for (Object o : a) { b = b & isEmpty(o); if (!b) {break; }} return b; } catch (ClassCastException e) { }}return false; } /** * 判斷這個(gè)Object是否不為Null或長(zhǎng)度不為0 * * @param obj * @return */ public static boolean isNotEmpty(Object obj) {return !isEmpty(obj); } /** * 返回首字母大寫單詞 * * @param str * @return */ public static String lcyFirstLetterToUpper(String str) {return str.replaceFirst(str.substring(0, 1), str.substring(0, 1).toUpperCase()); } /** * 轉(zhuǎn)換為下劃線 * * @param camelCaseName * @return */ public static String underscoreName(String camelCaseName) {StringBuilder result = new StringBuilder();if (camelCaseName != null && camelCaseName.length() > 0) { result.append(camelCaseName.substring(0, 1).toLowerCase()); for (int i = 1; i < camelCaseName.length(); i++) {char ch = camelCaseName.charAt(i);if (Character.isUpperCase(ch)) { result.append('_'); result.append(Character.toLowerCase(ch));} else { result.append(ch);} }}return result.toString(); } /** * 轉(zhuǎn)換為駝峰 * * @param underscoreName * @return */ public static String camelCaseName(String underscoreName) {StringBuilder result = new StringBuilder();if (underscoreName != null && underscoreName.length() > 0) { boolean flag = false; for (int i = 0; i < underscoreName.length(); i++) {char ch = underscoreName.charAt(i);if ('_'.charAt(0) == ch) { flag = true;} else { if (flag) {result.append(Character.toUpperCase(ch));flag = false; } else {result.append(ch); }} }}return result.toString(); } public static void main(String[] args) {System.out.println(underscoreName('loginName')); }}

這樣就可以做到動(dòng)態(tài)生成查詢條件,復(fù)雜的查詢條件也不會(huì)去改動(dòng)XML配置了。

mybatis原理:參數(shù)解析與SQL動(dòng)態(tài)組裝過(guò)程

mybatis執(zhí)行sql之前, 需要經(jīng)過(guò)參數(shù)解析、sql動(dòng)態(tài)組裝等過(guò)程,本文主要聊聊mybatis的:

(1)參數(shù)解析原理及其過(guò)程

(2)sql動(dòng)態(tài)組裝原理及其過(guò)程

一、數(shù)據(jù)準(zhǔn)備

1.實(shí)體類,省略了set、get方法

public class User { private String id; private String username; private String password; private Integer isValid;}

2.mapper接口UserMapper,可以看作是一個(gè)根據(jù)用戶名和密碼的登錄接口

User getUserByUsernameAndPassword(@Param('name') String username, @Param('pwd') String password);

3.mapper映射

<select resultType='com.qxf.pojo.User'>select id,username,password,is_valid as isValid from t_user<where> <if test='name != null and name != ’’'>username = #{name} </if> <if test='pwd != null and pwd != ’’'>and password = #{pwd} </if></where> </select>

4.測(cè)試,mybatis-config.xml配置文件按一般配置即可,這里就不貼代碼了

//讀取配置信息InputStream inputStream = Resources.getResourceAsStream('mybatis-config.xml');//根據(jù)配置信息,創(chuàng)建SqlSession工廠SqlSessionFactory factory = new SqlSessionFactoryBuilder().build(inputStream);//SqlSession工廠創(chuàng)建SqlSessionSqlSession sqlSession = factory.openSession();//獲取接口的代理對(duì)象UserMapper mapper = sqlSession.getMapper(UserMapper.class);//執(zhí)行相應(yīng)的接口方法User user = mapper.getUserByUsernameAndPassword('張三2', null);System.out.println(user);

下面將以這句代碼為入口:

(注意,這里只是為了測(cè)試,給密碼參數(shù)傳遞了null,正常情況不會(huì)這樣傳遞參數(shù)的,不然結(jié)果返回一個(gè)List集合就會(huì)報(bào)錯(cuò)的)

//執(zhí)行相應(yīng)的接口方法User user = mapper.getUserByUsernameAndPassword('張三2', null);二、參數(shù)解析原理及其過(guò)程

首先要明白一點(diǎn),返回的是mapper接口的代理對(duì)象,所以會(huì)來(lái)到MapperProxy的invoke方法

public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {try { // Object對(duì)象的方法,則直接執(zhí)行 if (Object.class.equals(method.getDeclaringClass())) {return method.invoke(this, args); } if (method.isDefault()) {return this.invokeDefaultMethod(proxy, method, args); }} catch (Throwable var5) { throw ExceptionUtil.unwrapThrowable(var5);}// 獲取mapperMethod,這里面就會(huì)進(jìn)行參數(shù)解析MapperMethod mapperMethod = this.cachedMapperMethod(method);// 執(zhí)行方法return mapperMethod.execute(this.sqlSession, args); }

重點(diǎn)關(guān)注這句:

// 獲取mapperMethod,這里面就會(huì)進(jìn)行參數(shù)解析MapperMethod mapperMethod = this.cachedMapperMethod(method);

參數(shù)的解析可以分成兩部:

(1)形參的解析

(2)實(shí)參的封裝

(1)形成的解析

一路跟進(jìn)去,最終會(huì)來(lái)到 ParamNameResolver,暫且叫做參數(shù)名稱解析器吧,首先會(huì)在構(gòu)造器組裝參數(shù)的位置和名稱的對(duì)應(yīng)關(guān)系,如果我們使用了@Param注解,則會(huì)使用我們定義的名稱,否則會(huì)使用arg0、arg1....依次替代,詳細(xì)代碼如下:

public ParamNameResolver(Configuration config, Method method) {// 獲取參數(shù)列表中,每一個(gè)參數(shù)的類型Class<?>[] paramTypes = method.getParameterTypes();// 獲取參數(shù)注解,因?yàn)槊總€(gè)參數(shù)可能有多個(gè)注解,所以是二維數(shù)組Annotation[][] paramAnnotations = method.getParameterAnnotations();// 存放結(jié)果的mapSortedMap<Integer, String> map = new TreeMap();// 參數(shù)個(gè)數(shù)int paramCount = paramAnnotations.length; for(int paramIndex = 0; paramIndex < paramCount; ++paramIndex) { if (!isSpecialParameter(paramTypes[paramIndex])) {// 參數(shù)名稱String name = null;// 參數(shù)的注解數(shù)組Annotation[] var9 = paramAnnotations[paramIndex];// 參數(shù)注解的個(gè)數(shù)int var10 = var9.length;// 遍歷每個(gè)注解,找到Param注解,拿到value作為參數(shù)名稱for(int var11 = 0; var11 < var10; ++var11) { Annotation annotation = var9[var11]; if (annotation instanceof Param) {this.hasParamAnnotation = true;name = ((Param)annotation).value();break; }} if (name == null) { if (config.isUseActualParamName()) {name = this.getActualParamName(method, paramIndex); } if (name == null) {name = String.valueOf(map.size()); }}// 參數(shù)序號(hào)作為key,從0開(kāi)始,參數(shù)名稱作為值map.put(paramIndex, name); }}// 沒(méi)有做什么,再一次封裝而已this.names = Collections.unmodifiableSortedMap(map); }

結(jié)果是這樣的:符合我們的預(yù)期的

Mybatis 實(shí)現(xiàn)動(dòng)態(tài)組裝查詢條件,仿SQL模式

(2)實(shí)參的封裝

然后會(huì)來(lái)到getNamedParams方法對(duì)參數(shù)進(jìn)一步的封裝:

public Object getNamedParams(Object[] args) {// 參數(shù)個(gè)數(shù),這個(gè)names就是上面解析后的map,key是從0開(kāi)始的參數(shù)序號(hào),value是參數(shù)名稱int paramCount = this.names.size();// 這里的args便是實(shí)參列表// 實(shí)參不為空,形參個(gè)數(shù)不為0if (args != null && paramCount != 0) { if (!this.hasParamAnnotation && paramCount == 1) {// 沒(méi)有使用@Param注解,并且只有一個(gè)參數(shù)return args[(Integer)this.names.firstKey()]; } else {// 將參數(shù)封裝成一個(gè)mapMap<String, Object> param = new ParamMap();int i = 0;// 對(duì)形參循環(huán)迭代for(Iterator var5 = this.names.entrySet().iterator(); var5.hasNext(); ++i) { Entry<Integer, String> entry = (Entry)var5.next(); // names中的參數(shù)名稱為key,值為實(shí)參值 param.put((String)entry.getValue(), args[(Integer)entry.getKey()]); // 并添加key為param1、param2之類的通用參數(shù) String genericParamName = 'param' + String.valueOf(i + 1); if (!this.names.containsValue(genericParamName)) {param.put(genericParamName, args[(Integer)entry.getKey()]); }} return param; }} else { return null;} }

通過(guò)源碼可以發(fā)現(xiàn),

(1)如果只有一個(gè)參數(shù),并且沒(méi)有使用@Param注解,就直接返回第一個(gè)參數(shù)

(2)有多個(gè)參數(shù),則封裝成一個(gè)map,key為參數(shù)參數(shù)名稱,使用了@Param注解,名稱就是注解中的值,否則key為arg0、arg1這種類型,同時(shí),一定含有key為param1、param2的參數(shù),值就是傳入的值

封裝后的結(jié)果如下:

Mybatis 實(shí)現(xiàn)動(dòng)態(tài)組裝查詢條件,仿SQL模式

這樣就完成了參數(shù)的解析過(guò)程,總結(jié)一下:

(1)解析形參,判斷是否使用了@Param注解

(2)封裝實(shí)參,如果只有一個(gè),并且沒(méi)有使用@Param注解,就直接返回第一個(gè)參數(shù)值,否則封裝成map

三、動(dòng)態(tài)組裝sql原理及其過(guò)程

來(lái)到CachingExecutor的如下方法,作為入口:

public <E> List<E> query(MappedStatement ms, Object parameterObject, RowBounds rowBounds, ResultHandler resultHandler) throws SQLException {// 獲取組裝完成的sqlBoundSql boundSql = ms.getBoundSql(parameterObject);// 創(chuàng)建緩存keyCacheKey key = this.createCacheKey(ms, parameterObject, rowBounds, boundSql);// 執(zhí)行查詢r(jià)eturn this.query(ms, parameterObject, rowBounds, resultHandler, key, boundSql); }

重點(diǎn)看這句:

// 獲取組裝完成的sqlBoundSql boundSql = ms.getBoundSql(parameterObject);

一路跟進(jìn)去,來(lái)到DynamicSqlSource的getBoundSql方法:

public BoundSql getBoundSql(Object parameterObject) {// 將參數(shù)封裝成動(dòng)態(tài)上下文,DynamicContext中sqlBuilder就是最后組裝的sqlDynamicContext context = new DynamicContext(this.configuration, parameterObject);// 根據(jù)條件,動(dòng)態(tài)組裝sqlthis.rootSqlNode.apply(context);SqlSourceBuilder sqlSourceParser = new SqlSourceBuilder(this.configuration);Class<?> parameterType = parameterObject == null ? Object.class : parameterObject.getClass();// 將#{參數(shù)}替換為?SqlSource sqlSource = sqlSourceParser.parse(context.getSql(), parameterType, context.getBindings());BoundSql boundSql = sqlSource.getBoundSql(parameterObject);Map var10000 = context.getBindings();Objects.requireNonNull(boundSql);var10000.forEach(boundSql::setAdditionalParameter);return boundSql; }

我們先看下這句:

// 根據(jù)條件,動(dòng)態(tài)組裝sqlthis.rootSqlNode.apply(context);

對(duì)于我們的sql:

<select resultType='com.qxf.pojo.User'>select id,username,password,is_valid as isValid from t_user<where> <if test='name != null and name != ’’'>username = #{name} </if> <if test='pwd != null and pwd != ’’'>and password = #{pwd} </if></where> </select>

每個(gè)標(biāo)簽都有對(duì)應(yīng)的SqlNode來(lái)處理,比如if標(biāo)簽,就由IfSqlNode來(lái)處理,where標(biāo)簽,則會(huì)通過(guò)TrimSqlNode來(lái)處理,SqlNode的具體實(shí)現(xiàn)類如下:

Mybatis 實(shí)現(xiàn)動(dòng)態(tài)組裝查詢條件,仿SQL模式

這里以IfSqlNode處理if標(biāo)簽為例:

Mybatis 實(shí)現(xiàn)動(dòng)態(tài)組裝查詢條件,仿SQL模式

這是就是兩步:

(1)判斷表達(dá)式的值是否為真,這里最終使用的是Ognl來(lái)判斷

(2)如果表達(dá)式的為真,就將標(biāo)簽內(nèi)容追加到sql中去

處理結(jié)果如下:

Mybatis 實(shí)現(xiàn)動(dòng)態(tài)組裝查詢條件,仿SQL模式

因?yàn)槊艽a的參數(shù)傳入為null,所以不會(huì)拼接密碼查詢條件,只拼接了用戶名查詢條件

然后是將#{參數(shù)}替換為?進(jìn)行占位:

// 將#{參數(shù)}替換為?SqlSource sqlSource = sqlSourceParser.parse(context.getSql(), parameterType, context.getBindings());

這個(gè)就比較簡(jiǎn)單了,可以自行看源碼,最終是這樣的:

Mybatis 實(shí)現(xiàn)動(dòng)態(tài)組裝查詢條件,仿SQL模式

以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持好吧啦網(wǎng)。

主站蜘蛛池模板: av免费网址在线观看 | 三级黄色毛片视频 | 91精品国产99 | 免费成人福利视频 | 免费看欧美一级特黄a大片 免费看欧美中韩毛片影院 免费看片91 | 狠狠躁日日躁夜夜躁2022麻豆 | 国产精品婷婷久久爽一下 | 日韩精品免费在线观看 | 国产精品对白久久久久粗 | 曰本女人与公拘交酡免费视频 | 欧洲成人午夜免费大片 | 国产黄色录像 | 少妇一级淫免费放 | 成人交性视频免费看 | 午夜av无码福利免费看网站 | 国产精品久久久久久久久久久久午夜 | 一卡二卡三卡在线 | 亚洲 a v无 码免 费 成 人 a v | 天天做天天干 | 国产香蕉视频 | 中字幕视频在线永久在线观看免费 | 欧美aⅴ在线观看 | 国产精品成人无码久久久 | 国产精品伦一区二区三级视频永妇 | 麻豆精品久久久 | 国产95在线 | 亚洲 | 美女啪啪网址 | 浓精h攵女乱爱av | 欧性猛交ⅹxxx乱大交 | 99精品在线播放 | 亚洲第一视频网 | 一区二区三区视频免费看 | 亚洲乱码无码永久不卡在线 | 国产甜淫av片免费观看 | 黄色片网站在线播放 | 欧美激情xxx| 精品1卡二卡三卡四卡老狼 国内大量偷窥精品视频 | 国产精品网站视频 | 久久久久国产精品免费免费搜索 | 亚洲图片欧美视频 | 91看片在线观看 | 激情四射网 | 婷婷丁香综合 | 波多野吉衣在线观看视频 | 亚洲熟妇丰满多毛xxxx | 日本亚洲在线 | 久久久综合激的五月天 | 国产浮力第一页草草影院 | aaaaa级毛片| 国产粉嫩尤物极品99综合精品 | 久久久综合久久久 | 五月天婷婷在线视频 | brazzers欧美一区二区 | 免费午夜激情 | 日本牲交大片无遮挡 | av 黄色| 丰满少妇在线观看资源站 | 暖暖成人免费视频 | 国产精品蜜| 黄色网入口 | 亚洲午夜久久久精品一区二区三剧 | 在线免费观看a视频 | 少妇浴室愉情韩国理论 | 给我免费播放毛片 | 波多野结衣黄色网址 | 久久久久麻豆v国产精华液好用吗 | 毛片一级片 | 国产曰肥老太婆无遮挡 | 污漫在线观看 | 女人18毛片aaa片水真多 | 国产肉体xxxx裸体784大胆 | 国产性猛交╳xxx乱大交 | 99精品国产免费久久 | 直接看的毛片 | 婷婷亚洲综合五月天小说 | 女学生处破外女出血av喊痛 | 久久极品视频 | 欧美日韩资源 | 亚洲欧美偷拍另类a∨色屁股 | 免费99视频 | 久久久久久aaaabbbb | 日本成人动漫在线观看 | 无尽3d精品hentai在线视频 | 日韩人妻无码一区二区三区综合部 | 久久久综合香蕉尹人综合网 | 国产毛片毛片精品天天看软件 | 亚洲人成网站999久久久综合 | 在线免费你懂的 | 色极品影院 | 日本精品网站 | 婷婷久久久 | 久久天堂综合亚洲伊人hd妓女 | 国产毛片毛片毛片毛片毛片毛片 | 超薄肉色丝袜一二三四区 | 亚洲专区第一页 | av动漫网站 | 99精产国品产在线观看 | 另类国产 | 国产最露的三级 | 一级做a爰片久久毛片a | 国产无套内射又大又猛又粗又爽 | 视频免费1区二区三区 | 国产精品一二区 | 成人免费毛片xxx | 亚洲第一大网站 | 国产精品aⅴ | 久久久久久成人毛片免费看 | 99爱精品 | 久色网 | 老司机亚洲精品影院 | 中文字幕一区二区在线观看 | 日日碰狠狠躁久久躁9 | 日韩女优在线观看 | 久久狠狠高潮亚洲精品 | 四虎永久在线精品免费一区二区 | 国语对白超精彩 | 女同重口另类在线观看 | 黄色美女大片 | 91丨porny丨中文 | 性久久久久久 | 韩国三级中文字幕hd浴缸戏 | 已婚少妇美妙人妻系列 | 久久精品亚洲精品国产欧美kt∨ | 日本成人动漫在线观看 | 精品无码一区二区三区在线 | 国产精品久久久久久欧美2021 | 亚洲精品tv| 在线视频你懂得 | 久久综合网av | 一级国产20岁美女毛片 | ww国产内射精品后入国产 | 久久国产精品广西柳州门 | 欧美激情一区二区三区成人 | 欧美精品一区二区三区四区五区 | 色综合视频一区二区三区 | 日本一二三区视频在线 | 永久免费观看av | 国产精品九九九九 | 波多野结衣一区二区三区在线观看 | 久色在线 | 亚洲一区二区三区在线观看网站 | 国产精品人妻系列21p | 日本肉体xxxx裸体137大胆图 | 国产精品第一 | 69久久夜色精品国产69蝌蚪网 | 国产无遮挡又黄又爽免费软件 | 色婷婷婷婷色 | 黄色激情小说网站 | 在线观看日本www | 一区二区三区在线 | 网站 | xxxxxl19成人免费视频 | 嫩草影院在线观看视频 | 久久国产精品广西柳州门 | 久久这里只有精品视频9 | 国产精品欧美精品 | 免费在线h| 国产99热 | 外国黄色网址 | 亚洲香蕉网站 | 色综合天 | 极品少妇xxxx | 午夜无码一区二区三区在线观看 | 日本一卡二卡四卡无卡乱码视频免费 | 欧美视频网站www色 精品无码久久久久久久动漫 | 狠狠色噜噜狠狠狠狠av不卡 | 国产精品久久久久久亚洲影视内衣 | 国产无套精品 | 日鲁鲁| 国产精品一区在线免费观看 | 亚洲v无码一区二区三区四区观看 | 男女啪啪免费 | 狠狠干狠狠撸 | 青青草五月天 | 国产亚洲真人做受在线观看 | 午夜免费啪视频在线观看 | 奇米影视第四色首页 | 在线精品亚洲 | 亚洲精品乱码久久久久久蜜桃不卡 | 黑人性生活视频 | 日韩一级性| 7777久久久国产精品 | 成人av播放 | 久久嗨 | 中文字幕精品一区二区三区在线 | 中文字幕人妻互换av久久 | 99视频观看 | 欧美日韩网址 | 爱爱网站视频 | 亚洲视频在线免费观看 | 久久久久亚洲ai毛片换脸星大全 | 成人午夜免费福利 | 一级国产航空美女毛片内谢 | 18性夜影院午夜寂寞影院免费 | 一本大道久久久久精品嫩草 | 欧美午夜性春猛交xxxx按摩师 | 日韩欧美理论 | 高清国产精品人妻一区二区 | 国产奶水涨喷在线播放 | 夜夜操夜夜 | av片在线看 | 欧美两根一起进3p做受视频 | 日本大片黄 | 亚洲精品一区三区三区在线观看 | 日本全棵写真视频在线观看 | 国产激情视频在线播放 | 亚洲国产日本 | 成人在线视屏 | 亚洲高清欧美 | 久久免费av | 精品一区在线播放 | 日韩av资源| 国产最新进精品视频 | jizz免费在线观看 | 国产午夜亚洲精品午夜鲁丝片 | 欧美无砖专区免费 | 国产精品久久久久久久久久久久人四虎 | 91麻豆精品91久久久久同性 | 五月色婷婷俺来也在线观看 | 午夜理论片yy6080私人影院 | 精品免费久久久 | 在线aⅴ亚洲中文字幕 | 全部av―极品视觉盛宴 | 精品国产一区二区三区四区 | 蜜乳av一区二区三区 | 成人免费视频观看 | 久久精品丝袜 | 99热99re6国产在线播放 | 男人午夜免费视频 | 欧美日韩激情在线一区二区三区 | 天天干天天色天天射 | 亚洲色大成网站www国产 | 天天操天天添 | 国产丰满天美videossex | 亚洲自拍偷拍网 | 久无码久无码av无码 | 在线免费一区二区 | 日韩精品视频在线免费观看 | 成人拍拍视频 | 在线视频日韩欧美 | 黑人巨大精品欧美一区二区小视频 | 亚洲精品国产精品乱码在线观看 | 免费在线观看a级片 | 欧美日韩一区二区免费视频 | 国产中文字幕网 | 一级二级av | 在线无码va中文字幕无码 | 久久精品波多野结衣 | 免费国产一区 | 青青操免费在线视频 | 97色伦影院| 91精品国产综合久久久久久久久 | 黄色精品网站 | 亚洲国产一区二区a毛片 | 四虎永久在线精品免费网站 | 欧美日韩一区二区三区在线 | 日韩在线视频观看 | 国产精品综合av一区二区国产馆 | 免费黄网站在线 | 777精品出轨人妻国产 | 国产区欧美区日韩区 | 91pornyⅰ九色 | 中国丰满少妇人妻xxx性董鑫洁 | 久久午夜无码鲁丝片秋霞 | 欧美成人h | 自拍偷拍99 | 久久久久久免费毛片精品 | 国产亚洲一区二区在线观看 | 日本不卡视频在线播放 | 99re最新 | 国产成人精品一区二区三区免费 | 懂色一区二区三区久久久 | 五月激情婷婷在线 | 亚洲欧洲精品视频 | 日本人与禽zozzo小小的几孑 | 午夜国产精品国产自线拍免费人妖 | 少妇淫片aaaaa毛片叫床爽 | 视频在线观看免费完整高清中文 | 337p粉嫩大胆色噜噜噜噜 | 免费黄色片子 | 欧美不卡一区二区三区 | 一级毛片黄 | 天天做天天看 | 秋霞二区 | 亚洲人成色7777在线观看 | 东京一本一道一二三区 | 五月天中文字幕在线 | 久久久综合激的五月天 | 把插八插露脸对白内射 | 欧美激情偷拍 | 一本一道av中文字幕无码 | 翔田千里一区二区 | 日韩欧美中文字幕在线观看 | 精品人妻无码一区二区三区换脸 | 8av国产精品爽爽ⅴa在线观看 | 日韩一区二区三区不卡 | 成年人在线观看视频网站 | 亚洲天堂伊人网 | 久久久久北条麻妃免费看 | 伦人伦xxx国产对白 亚洲国产精品一区二区成人片国内 | 污片免费看 | 成人精品一区二区三区中文字幕 | 日本a天堂 | 免费观看av的网站 | 成人热舞视频一区 | 神马午夜91 | 日本无遮挡边做边爱边摸 | 性猛交富婆xxxx乱大 | 欧美日韩精品区 | 一二三四观看视频社区在线 | 日中文字幕 | 国语自产拍91在线a拍拍 | 最新国产黄色网址 | 捆绑紧缚一区二区三区在线观看 | 国产成年人 | 一级少妇淫片免费观看 | 国产91 在线播放 | 丁香色欲久久久久久综合网 | 日本在线免费看 | 一本一道av无码中文字幕麻豆 | 欧美性aaa| 色综合久久88色综合天天 | 人人玩人人添人人澡欧美 | 国产精品高潮呻吟久久久久久 | 三级av| 久久成人麻豆午夜电影 | 国产一区二区三区视频在线播放 | 久久久一 | 97人人模人人爽人人少妇 | 欧美午夜精品久久久久免费视 | 91日日日 | 国产奶头好大揉着好爽视频 | 天天综合网7799精品 | 五月激情丁香婷婷 | 中文无码热在线视频 | 男男成人高潮片免费网站 | 白嫩少妇喷水正在播放 | 久久激情网 | 我和房东少妇激情 | 日本黄漫动漫在线观看视频 | 国产l精品国产亚洲区在线观看 | 日本一区二区视频 | 国产免费破外女真实出血视频 | 免费人妻无码不卡中文字幕18禁 | 狠狠色噜噜 | 欧美v日韩 | 永久免费网站看黄yyy45视频 | 国产精品视频999 | 国产欧美视频一区 | 香蕉网站在线观看 | 成人爱爱免费视频 | 免费观看日本污污ww网站 | 亚洲欧美日韩一区在线观看 | 成人久久18免费网站图片 | 亚洲日产韩国一二三四区 | 夜天干天干啦天干天天爽 | 美一女一无一伦一性一交 | 国产一区亚洲 | 一本加勒比hezyo无码专区 | 欧美亚洲亚洲日韩在线影院 | 啪啪精品| 日本99热| 美女自卫慰黄网站 | 91视在线国内在线播放酒店 | 六月丁香婷婷色狠狠久久 | 久久论理| 爱啪啪av网 | 爱爱短视频 | 九九热精品在线视频 | 91不戴套国语对白在线观看 | 欧美35页视频在线观看 | 国产精品欧美一区二区三区奶水 | 欧美精品在线看 | 秋霞影院一区二区三区 | 久久久性色精品国产免费观看 | 亚洲精品无码专区在线在线播放 | 亚洲精品一区二区三区在线观看 | 免费国产玉足脚交视频 | 一级大片在线观看 | 国产边打电话边做对白刺激 | 动漫卡通精品一区二区三区介绍 | 调教大乳女仆喷奶水 | 福利一区二区 | 亚洲国产精品无码久久久 | 成人午夜高潮a∨猛片 | 免费日韩视频 | 女性毛片 | 一级性爱视频 | 成人国产一区二区 | 国产玉足榨精视频在线观看 | 盗摄中年夫妇啪啪免费观看 | 国产精久久久久久 | 日本欧美另类 | 99视频在线精品免费观看2 | 国产精品亚洲专区无码牛牛 | 美女131爽爽爽做爰视频 | 老司机久久精品视频 | 国产又粗又硬又大爽黄老大爷视 | 97影院 | 色午夜| 亚洲综合另类小说色区 | 性欧美18—19sex性高清 | 成年人视频免费在线观看 | 720lu国产刺激无码 | 国产日韩精品在线观看 | 亚洲女同在线观看 | 2018自拍偷拍 | 人妻互换免费中文字幕 | 精品久久久噜噜噜久久久 | 91九色丨porny丨丰满6 | 伊人情人成综合 | 开心激情av | 国产人伦精品一区二区三区 | 九九热精品视频在线 | 2018天天拍拍天天爽视频 | 深夜视频免费在线观看 | 日韩女人性猛交 | 国产成人在线网站 | 国内精品视频一区二区三区 | 亚洲成a人片在线观看无码专区 | 双性受惨叫扩张调教虐宫h 爽插 | 一性一交一摸一黄按摩精油视频 | 一本大道久久a久久综合婷婷 | 欧美激情在线播放 | 欧美成人a视频 | 精品国产乱码久久久久久1区二区 | 成人日韩在线观看 | 亚洲国产女人aaa毛片在线 | 日本少妇做爰全过程毛片 | 中文字幕热久久久久久久 | 69av在线| 亚洲午夜一区二区 | 777奇米888色狠狠俺也去 | 米奇777四色精品人人爽 | 色窝窝无码一区二区三区色欲 | 91动态图 | 九色在线播放 | 亚洲欧美成人在线 | 人人爽人人爽人人爽人人片av | av男人天堂av | 国产精品久久久一区二区三区 | 亚洲欧美精品久久 | 日韩高清在线播放 | 麻豆国产一区二区三区 | 日韩激情视频网站 | 日韩成人av网址 | 欧美性激情 | 两人做人爱费视频午夜 | 天天干天天射综合网 | 91av在线视频播放 | 三级黄色小视频 | 一个色综合久久 | 91精品国产高清91久久久久久 | 9l视频自拍九色9l视频最新 | 一 级 黄 色 片免费网站 | 在线看片中文字幕 | 蜜桃视频日韩 | 麻豆国产人妻欲求不满谁演的 | 67194午夜| 日韩一二三四五区 | 亚洲a色 | 日韩国产欧美视频 | jizz在亚洲 | 精品久久人妻av中文字幕 | 在线观看毛片网站 | 舌吻激情大尺度做爰视频 | 国产911| 99精品国产兔费观看久久99 | 在线观看日韩 | 九月激情网 | 日韩av成人网 | 亚洲精品无码一区二区 | 国产91在线播放九色 | 国产一区调教91鞭打 | 亚洲午夜剧场 | 无码粉嫩虎白一线天在线观看 | av不卡中文字幕 | 中文字幕五区 | 嫩模周妍希视频一区二区 | 色视频网站免费 | 午夜精品久久久久久久久久久久久蜜桃 | 国产成人精品一区二区三区免费 | 免费国产一区 | 男人的天堂一级片 | 久久精品免费一区二区三区 | 黄色成人av网站 | 亚洲精品久久久久午夜福禁果tⅴ | 亚洲色大网站www永久网站 | 在线观看国产福利 | 日干夜操| 无码人妻av一区二区三区波多野 | 国产欧亚州美日韩综合区 | 精品免费国产 | 亚洲精品久久区二区三区蜜桃臀 | 一级做a爰片久久毛片潮喷动漫 | 91官网在线 | 欧美mv日韩mv国产网站 | 国产精品igao视频网 | 99久久国产露脸国语对白 | 国产第6页 | 999视频在线观看 | 欧美日韩高清在线播放 | 国产麻豆精品久久一二三 | 国产a级黄色 | 在线欧美国产 | 爆乳熟妇一区二区三区 | 一级免费在线 | 毛片视频网 | 久久99国产精品 | 天堂资源地址在线 | 无码人妻视频一区二区三区 | 公妇乱淫3 | 午夜精品福利一区二区三区蜜桃 | 特黄特色大片免费播放叫疼 | www久久婷婷| 日韩一区二区在线观看视频 | 中文字幕在线观看 | 精品视频一区二区三区四区 | 国人天堂va在线观看免费 | 青青草在线视频网站 | 国产99久久精品一区二区 | 久久久www成人免费精品 | 欧美xxxx黑人xyx性爽 | 特一级黄色 | 熟妇人妻系列aⅴ无码专区友真希 | 最近在线更新8中文字幕免费 | 草草影院国产第一页 | 国产女主播高潮在线播放 | 国产麻豆9l精品三级站 | 在火车千女人毛片看看 | 天堂网免费视频 | 91麻豆精品久久久久蜜臀 | 色婷婷综合久久久久中文一区二区 | 久久久久久国产精品免费免费男同 | 女性向h片在线观看 | 免费黄色a级片 | 国产精品久久亚洲7777 | 国产超碰人人做人人爽av牛牛 | 精品乱码一区二区三四区视频 | 天天躁日日躁狠狠躁伊人 | 肥臀浪妇太爽了快点再快点 | 天堂免费在线视频 | 国产白丝精品91爽爽久久 | 成人欧美一区二区三区黑人冫 | a视频免费看 | 国产99页| 天天躁人人躁人人躁狂躁 | 妞干网这里只有精品 | 亚洲最大成人在线 | 亚洲夜夜性无码 | 国产成a人亚洲精v品在线观看 | 国产香蕉尹人综合在线观看 | 成人美女黄网站色大免费的88 | 在线不卡aⅴ片免费观看 | 超碰国产天天做天天爽 | 黄色激情视频网站 | 日本道之久久综合久久爱 | 自拍偷拍色 | 国产人妖乱国产精品人妖 | 亚洲视频黄 | 中文字幕有码无码av | 欧美裸体xxxx极品少妇 | 亚洲性久久久影院 | 免费无码又爽又刺激软件下载直播 | 日本免费黄视频 | 亚洲日韩av无码 | 暖暖 在线 日本 免费 中文 | 欧美精品成人久久 | √最新版天堂资源网在线 | 亚洲tv久久爽久久爽 | 国产在线高清 | 中国丰满少妇人妻xxx性董鑫洁 | 可以免费观看的毛片 | 午夜天堂精品久久久久 | a在线播放 | zzijzzij亚洲丰满少妇 | 黄色免费视频网站 | 亚洲欧美婷婷六月色综合 | 亚洲一级av毛片 | 国产日产欧产精品精品首页 | 亚洲人成图片小说网站 | 综合五月激情二区视频 | 天海翼精品久久中文字幕 | av国産精品毛片一区二区三区 | 日本一本在线观看 | 超碰在线最新 | 日日碰| 动漫av纯肉无码av在线播放 | 久久w5ww成w人免费 | 裸体欧美bbbb极品bbbb | 91九色丨porny丨朋友 | 无码人妻精品一区二区三18禁 | 亚洲黄色在线免费观看 | 老妇裸体性激交老太视频 | 7777奇米四色成人眼影 | 黄色免费网站在线 | 精品福利av导航 | 艳妇乳肉豪妇荡乳在线观看 | 日本一级大片 | 在线观看欧美一区二区 | 精品一区视频 | 91精品国产高潮对白 | 成人羞羞视频 | 日韩免费久久 | 一a级毛片 | 午夜人成免费视频 | 午夜福利1000集在线观看 | 男人天堂网在线 | 中文字幕中出 | 久久99国产综合精品免费 | 亚洲欧美黄色片 | 欧美1区2区3区 | 国产成人艳妇aa视频在线 | 日韩在线aⅴ免费视频 | 天天草夜夜 | 日韩精品亚洲一区 |