日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程语言 > asp.net >内容正文

asp.net

.sql文件_面试题:mybatis 中的 DAO 接口和 XML 文件里的 SQL 是如何建立关系的?

發(fā)布時間:2024/9/19 asp.net 38 豆豆
生活随笔 收集整理的這篇文章主要介紹了 .sql文件_面试题:mybatis 中的 DAO 接口和 XML 文件里的 SQL 是如何建立关系的? 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

前言

這是 mybatis 比較常問到的面試題,我自己在以前的面試過程中被問到了2次,2次都是非常重要的面試環(huán)節(jié),因此自己印象很深刻。這個題目我很早就深入學(xué)習(xí)了,但是一直沒有整理出來,剛好最近一段時間由于工作太忙,大概有半年沒有技術(shù)文章產(chǎn)出,因此趁著五一有點時間,整理了下分享給大家。另外,估計不少同學(xué)應(yīng)該也注意到了,DAO 接口的全路徑名和 XML 文件中的 SQL 的 namespace + id 是一樣的。其實,這也是建立關(guān)聯(lián)的根本原因。本文中的源碼使用當前最新的版本,即:mybatis-spring 為 2.0.4,mybatis 為 3.5.4,引入這2個 jar 包即可查看到本文的所有代碼。

正文

當一個項目中使用了 Spring 和 Mybatis 時,通常會有以下配置。當然現(xiàn)在很多項目應(yīng)該都是 SpringBoot 了,可能沒有以下配置,但是究其底層原理都是類似的,無非是將掃描 bean 等一些工作通過注解來實現(xiàn)。<bean class="org.mybatis.spring.mapper.MapperScannerConfigurer"> <property name="basePackage" value="com.joonwhee.open.mapper"/> <property name="sqlSessionFactoryBeanName" value="sqlSessionFactory"/>bean> <bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean"> <property name="dataSource" ref="dataSource"/> <property name="mapperLocations" value="classpath:config/mapper/*.xml"/> <property name="configLocation" value="classpath:config/mybatis/mybatis-config.xml"/> <property name="typeAliasesPackage" value="com.joonwhee.open.po"/>bean> <bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource" init-method="init" destroy-method="close"> <property name="driverClassName" value="${driver}"/> <property name="url" value="${url}"/> <property name="username" value="${username}"/> <property name="password" value="${password}"/>bean>通常我們還會有 DAO 類和 對用的 mapper 文件,如下。package com.joonwhee.open.mapper; import com.joonwhee.open.po.UserPO; public interface UserPOMapper { UserPO queryByPrimaryKey(Integer id);}<?xml version="1.0" encoding="UTF-8" ?><mapper namespace="com.joonwhee.open.mapper.UserPOMapper" > <resultMap id="BaseResultMap" type="com.joonwhee.open.po.UserPO"> <result column="id" property="id" jdbcType="INTEGER" /> <result column="name" property="name" jdbcType="VARCHAR" /> resultMap> <select id="queryByPrimaryKey" resultMap="BaseResultMap" parameterType="java.lang.Integer"> select id, name from user where id = #{id,jdbcType=INTEGER} select>mapper>

1、解析 MapperScannerConfigurer

MapperScannerConfigurer 是一個 BeanDefinitionRegistryPostProcessor,會在 Spring 構(gòu)建 IoC容器的早期被調(diào)用重寫的 postProcessBeanDefinitionRegistry 方法,參考:Spring IoC:invokeBeanFactoryPostProcessors 詳解@Overridepublic void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) { if (this.processPropertyPlaceHolders) { processPropertyPlaceHolders(); } // 1.新建一個ClassPathMapperScanner,并填充相應(yīng)屬性 ClassPathMapperScanner scanner = new ClassPathMapperScanner(registry); scanner.setAddToConfig(this.addToConfig); scanner.setAnnotationClass(this.annotationClass); scanner.setMarkerInterface(this.markerInterface); scanner.setSqlSessionFactory(this.sqlSessionFactory); scanner.setSqlSessionTemplate(this.sqlSessionTemplate); scanner.setSqlSessionFactoryBeanName(this.sqlSessionFactoryBeanName); scanner.setSqlSessionTemplateBeanName(this.sqlSessionTemplateBeanName); scanner.setResourceLoader(this.applicationContext); scanner.setBeanNameGenerator(this.nameGenerator); scanner.setMapperFactoryBeanClass(this.mapperFactoryBeanClass); if (StringUtils.hasText(lazyInitialization)) { // 2.設(shè)置mapper bean是否需要懶加載 scanner.setLazyInitialization(Boolean.valueOf(lazyInitialization)); } // 3.注冊Filter,因為上面構(gòu)造函數(shù)我們沒有使用默認的Filter, // 有兩種Filter,includeFilters:要掃描的;excludeFilters:要排除的 scanner.registerFilters(); // 4.掃描basePackage,basePackage可通過",; \t\n"來填寫多個, // ClassPathMapperScanner重寫了doScan方法 scanner.scan( StringUtils.tokenizeToStringArray(this.basePackage, ConfigurableApplicationContext.CONFIG_LOCATION_DELIMITERS));}3.注冊 Filter,見代碼塊1。4.掃描 basePackage,這邊會走到 ClassPathBeanDefinitionScanner(ClassPathMapperScanner 的父類),然后在執(zhí)行 “doScan(basePackages)” 時回到 ClassPathMapperScanner 重寫的方法,見代碼塊2。

代碼塊1:registerFilters

public void registerFilters() { boolean acceptAllInterfaces = true; // if specified, use the given annotation and / or marker interface // 1.如果指定了注解,則將注解添加到includeFilters if (this.annotationClass != null) { addIncludeFilter(new AnnotationTypeFilter(this.annotationClass)); acceptAllInterfaces = false; } // override AssignableTypeFilter to ignore matches on the actual marker interface // 2.如果指定了標記接口,則將標記接口添加到includeFilters, // 但這邊重寫了matchClassName方法,并返回了false, // 相當于忽略了標記接口上的匹配項,所以該參數(shù)目前相當于沒有任何作用 if (this.markerInterface != null) { addIncludeFilter(new AssignableTypeFilter(this.markerInterface) { @Override protected boolean matchClassName(String className) { return false; } }); acceptAllInterfaces = false; } // 3.如果沒有指定annotationClass和markerInterface,則 // 添加默認的includeFilters,直接返回true,接受所有類 if (acceptAllInterfaces) { // default include filter that accepts all classes addIncludeFilter((metadataReader, metadataReaderFactory) -> true); } // exclude package-info.java // 4.添加默認的excludeFilters,排除以package-info結(jié)尾的類 addExcludeFilter((metadataReader, metadataReaderFactory) -> { String className = metadataReader.getClassMetadata().getClassName(); return className.endsWith("package-info"); });}

通常我們都不會指定 annotationClass 和 markerInterface,也就是會添加默認的 Filter,相當于會接受除了 package-info 結(jié)尾的所有類。因此,basePackage 包下的類不需要使用 @Component 注解或 XML 中配置 bean 定義,也會被添加到 IoC 容器中。

代碼塊2:doScan

@Overridepublic Set doScan(String... basePackages) { // 1.直接使用父類的方法掃描和注冊bean定義, // 之前在spring中已經(jīng)介紹過:https://joonwhee.blog.csdn.net/article/details/87477952 代碼塊5 Set beanDefinitions = super.doScan(basePackages); if (beanDefinitions.isEmpty()) { LOGGER.warn(() -> "No MyBatis mapper was found in '" + Arrays.toString(basePackages) + "' package. Please check your configuration."); } else { // 2.對掃描到的beanDefinitions進行處理,主要4件事: // 1)將bean的真正接口類添加到通用構(gòu)造函數(shù)參數(shù)中 // 2)將beanClass直接設(shè)置為MapperFactoryBean.class, // 結(jié)合1,相當于要使用的構(gòu)造函數(shù)是MapperFactoryBean(java.lang.Class) // 3)添加sqlSessionFactory屬性,sqlSessionFactoryBeanName和 // sqlSessionFactory中,優(yōu)先使用sqlSessionFactoryBeanName // 4)添加sqlSessionTemplate屬性,同樣的,sqlSessionTemplateBeanName // 優(yōu)先于sqlSessionTemplate, processBeanDefinitions(beanDefinitions); } return beanDefinitions;}小結(jié),解析 MapperScannerConfigurer 主要是做了幾件事:1)新建掃描器 ClassPathMapperScanner;2)使用 ClassPathMapperScanner 掃描注冊 basePackage 包下的所有 bean;3)將 basePackage 包下的所有 bean 進行一些特殊處理:beanClass 設(shè)置為 MapperFactoryBean、bean 的真正接口類作為構(gòu)造函數(shù)參數(shù)傳入 MapperFactoryBean、為 MapperFactoryBean 添加 sqlSessionFactory 和 sqlSessionTemplate屬性。2、解析 SqlSessionFactoryBean對于 SqlSessionFactoryBean 來說,實現(xiàn)了2個接口,InitializingBean 和 FactoryBean,看過我之前 Spring 文章的同學(xué)應(yīng)該對這2個接口不會陌生,簡單來說:1)FactoryBean 可以自己定義創(chuàng)建實例對象的方法,只需要實現(xiàn)它的 getObject() 方法;InitializingBean 則是會在 bean 初始化階段被調(diào)用。SqlSessionFactoryBean 重寫這兩個接口的部分方法代碼如下,核心代碼就一個方法—— “buildSqlSessionFactory()”。@Overridepublic SqlSessionFactory getObject() throws Exception { if (this.sqlSessionFactory == null) { // 如果之前沒有構(gòu)建,則這邊也會調(diào)用afterPropertiesSet進行構(gòu)建操作 afterPropertiesSet(); } return this.sqlSessionFactory;} @Overridepublic void afterPropertiesSet() throws Exception { // 省略部分代碼 // 構(gòu)建sqlSessionFactory this.sqlSessionFactory = buildSqlSessionFactory();}buildSqlSessionFactory()主要做了幾件事:1)對我們配置的參數(shù)進行相應(yīng)解析;2)使用配置的參數(shù)構(gòu)建一個 Configuration;3)使用 Configuration 新建一個 DefaultSqlSessionFactory。這邊的核心內(nèi)容是對于 mapperLocations 的解析,如下代碼。protected SqlSessionFactory buildSqlSessionFactory() throws Exception { // 省略部分代碼 // 5.mapper處理(最重要) if (this.mapperLocations != null) { if (this.mapperLocations.length == 0) { LOGGER.warn(() -> "Property 'mapperLocations' was specified but matching resources are not found."); } else { for (Resource mapperLocation : this.mapperLocations) { if (mapperLocation == null) { continue; } try { // 5.1 新建XMLMapperBuilder XMLMapperBuilder xmlMapperBuilder = new XMLMapperBuilder(mapperLocation.getInputStream(), targetConfiguration, mapperLocation.toString(), targetConfiguration.getSqlFragments()); // 5.2 解析mapper文件 xmlMapperBuilder.parse(); } catch (Exception e) { throw new NestedIOException("Failed to parse mapping resource: '" + mapperLocation + "'", e); } finally { ErrorContext.instance().reset(); } LOGGER.debug(() -> "Parsed mapper file: '" + mapperLocation + "'"); } } } else { LOGGER.debug(() -> "Property 'mapperLocations' was not specified."); } // 6.使用targetConfiguration構(gòu)建DefaultSqlSessionFactory return this.sqlSessionFactoryBuilder.build(targetConfiguration);}5.2 解析mapper文件,見代碼塊3。

代碼塊3:parse()

public void parse() { // 1.如果resource沒被加載過才進行加載 if (!configuration.isResourceLoaded(resource)) { // 1.1 解析mapper文件 configurationElement(parser.evalNode("/mapper")); // 1.2 將resource添加到已加載列表 configuration.addLoadedResource(resource); // 1.3 綁定namespace的mapper bindMapperForNamespace(); } parsePendingResultMaps(); parsePendingCacheRefs(); parsePendingStatements();}1.1 解析mapper文件,見代碼4。1.3 綁定namespace的mapper,見代碼塊6。代碼塊4:configurationElementprivate void configurationElement(XNode context) { try { // 1.獲取namespace屬性 String namespace = context.getStringAttribute("namespace"); if (namespace == null || namespace.isEmpty()) { throw new BuilderException("Mapper's namespace cannot be empty"); } // 2.設(shè)置currentNamespace屬性 builderAssistant.setCurrentNamespace(namespace); // 3.解析parameterMap、resultMap、sql等節(jié)點 cacheRefElement(context.evalNode("cache-ref")); cacheElement(context.evalNode("cache")); parameterMapElement(context.evalNodes("/mapper/parameterMap")); resultMapElements(context.evalNodes("/mapper/resultMap")); sqlElement(context.evalNodes("/mapper/sql")); // 4.解析增刪改查節(jié)點,封裝成Statement buildStatementFromContext(context.evalNodes("select|insert|update|delete")); } catch (Exception e) { throw new BuilderException("Error parsing Mapper XML. The XML location is '" + resource + "'. Cause: " + e, e); }} private void buildStatementFromContext(Listlist) { if (configuration.getDatabaseId() != null) { buildStatementFromContext(list, configuration.getDatabaseId()); } // 解析增刪改查節(jié)點,封裝成Statement buildStatementFromContext(list, null);} private void buildStatementFromContext(Listlist, String requiredDatabaseId) { for (XNode context : list) { // 1.構(gòu)建XMLStatementBuilder final XMLStatementBuilder statementParser = new XMLStatementBuilder(configuration, builderAssistant, context, requiredDatabaseId); try { // 2.解析節(jié)點 statementParser.parseStatementNode(); } catch (IncompleteElementException e) { configuration.addIncompleteStatement(statementParser); } }}這邊會一直執(zhí)行到 “statementParser.parseStatementNode();”,見代碼塊5。這邊每個 XNode 都相當于如下的一個 SQL,下面封裝的每個 MappedStatement 可以理解就是每個 SQL。<select id="queryByPrimaryKey" resultMap="BaseResultMap" parameterType="java.lang.Integer"> select id, name, password, age from user where id = #{id,jdbcType=INTEGER}select>代碼塊5:parseStatementNodepublic void parseStatementNode() { // 省略所有的屬性解析 // 將解析出來的所有參數(shù)添加到 mappedStatements 緩存 builderAssistant.addMappedStatement(id, sqlSource, statementType, sqlCommandType, fetchSize, timeout, parameterMap, parameterTypeClass, resultMap, resultTypeClass, resultSetTypeEnum, flushCache, useCache, resultOrdered, keyGenerator, keyProperty, keyColumn, databaseId, langDriver, resultSets);} // MapperBuilderAssistant.javapublic MappedStatement addMappedStatement( String id, SqlSource sqlSource, StatementType statementType, SqlCommandType sqlCommandType, Integer fetchSize, Integer timeout, String parameterMap, Class> parameterType, String resultMap, Class> resultType, ResultSetType resultSetType, boolean flushCache, boolean useCache, boolean resultOrdered, KeyGenerator keyGenerator, String keyProperty, String keyColumn, String databaseId, LanguageDriver lang, String resultSets) { if (unresolvedCacheRef) { throw new IncompleteElementException("Cache-ref not yet resolved"); } // 1.將id填充上namespace,例如:queryByPrimaryKey變成 // com.joonwhee.open.mapper.UserPOMapper.queryByPrimaryKey id = applyCurrentNamespace(id, false); boolean isSelect = sqlCommandType == SqlCommandType.SELECT; // 2.使用參數(shù)構(gòu)建MappedStatement.Builder MappedStatement.Builder statementBuilder = new MappedStatement.Builder(configuration, id, sqlSource, sqlCommandType) .resource(resource) .fetchSize(fetchSize) .timeout(timeout) .statementType(statementType) .keyGenerator(keyGenerator) .keyProperty(keyProperty) .keyColumn(keyColumn) .databaseId(databaseId) .lang(lang) .resultOrdered(resultOrdered) .resultSets(resultSets) .resultMaps(getStatementResultMaps(resultMap, resultType, id)) .resultSetType(resultSetType) .flushCacheRequired(valueOrDefault(flushCache, !isSelect)) .useCache(valueOrDefault(useCache, isSelect)) .cache(currentCache); ParameterMap statementParameterMap = getStatementParameterMap(parameterMap, parameterType, id); if (statementParameterMap != null) { statementBuilder.parameterMap(statementParameterMap); } // 3.使用MappedStatement.Builder構(gòu)建MappedStatement MappedStatement statement = statementBuilder.build(); // 4.將MappedStatement 添加到緩存 configuration.addMappedStatement(statement); return statement;}該方法會將節(jié)點的屬性解析后封裝成 MappedStatement,放到 mappedStatements 緩存中,key 為 id,例如:com.joonwhee.open.mapper.UserPOMapper.queryByPrimaryKey,value 為 MappedStatement。代碼塊6:bindMapperForNamespaceprivate void bindMapperForNamespace() { String namespace = builderAssistant.getCurrentNamespace(); if (namespace != null) { Class> boundType = null; try { // 1.解析namespace對應(yīng)的綁定類型 boundType = Resources.classForName(namespace); } catch (ClassNotFoundException e) { // ignore, bound type is not required } if (boundType != null && !configuration.hasMapper(boundType)) { // Spring may not know the real resource name so we set a flag // to prevent loading again this resource from the mapper interface // look at MapperAnnotationBuilder#loadXmlResource // 2.boundType不為空,并且configuration還沒有添加boundType, // 則將namespace添加到已加載列表,將boundType添加到knownMappers緩存 configuration.addLoadedResource("namespace:" + namespace); configuration.addMapper(boundType); } }} public void addMapper(Classtype) { mapperRegistry.addMapper(type);} public void addMapper(Classtype) { if (type.isInterface()) { if (hasMapper(type)) { throw new BindingException("Type " + type + " is already known to the MapperRegistry."); } boolean loadCompleted = false; try { // 將type和以該type為參數(shù)構(gòu)建的MapperProxyFactory作為鍵值對, // 放到knownMappers緩存中去 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); } } }}主要是將剛剛解析過的 mapper 文件的 namespace 放到 knownMappers 緩存中,key 為 namespace 對應(yīng)的 class,value 為 MapperProxyFactory。小結(jié),解析 SqlSessionFactoryBean 主要做了幾件事:1)解析處理所有屬性參數(shù)構(gòu)建 Configuration ,使用 Configuration 新建 DefaultSqlSessionFactory;2)解析 mapperLocations 屬性的 mapper 文件,將 mapper 文件中的每個 SQL 封裝成 MappedStatement,放到 mappedStatements 緩存中,key 為 id,例如:com.joonwhee.open.mapper.UserPOMapper.queryByPrimaryKey,value 為 MappedStatement。3)將解析過的 mapper 文件的 namespace 放到 knownMappers 緩存中,key 為 namespace 對應(yīng)的 class,value 為 MapperProxyFactory。3、解析 DAO 文件DAO 文件,也就是 basePackage 指定的包下的文件,也就是上文的 interface UserPOMapper 。上文 doScan 中說過,basePackage 包下所有 bean 定義的 beanClass 會被設(shè)置成 MapperFactoryBean.class,而 MapperFactoryBean 也是 FactoryBean,因此直接看 MapperFactoryBean 的 getObject 方法。@Overridepublic T getObject() throws Exception { // 1.從父類中拿到sqlSessionTemplate,這邊的sqlSessionTemplate也是doScan中添加的屬性 // 2.通過mapperInterface獲取mapper return getSqlSession().getMapper(this.mapperInterface);} // SqlSessionTemplate@Overridepublic T getMapper(Class type) { return getConfiguration().getMapper(type, this);} // Configuration.javapublic T getMapper(Class type, SqlSession sqlSession) { return mapperRegistry.getMapper(type, sqlSession);} // MapperRegistry.javapublic T getMapper(Class type, SqlSession sqlSession) { // 1.從knownMappers緩存中獲取 final MapperProxyFactory mapperProxyFactory = (MapperProxyFactory) knownMappers.get(type); if (mapperProxyFactory == null) { throw new BindingException("Type " + type + " is not known to the MapperRegistry."); } try { // 2.新建實例 return mapperProxyFactory.newInstance(sqlSession); } catch (Exception e) { throw new BindingException("Error getting mapper instance. Cause: " + e, e); }} // MapperProxyFactory.javapublic T newInstance(SqlSession sqlSession) { // 1.構(gòu)造一個MapperProxy final MapperProxy mapperProxy = new MapperProxy<>(sqlSession, mapperInterface, methodCache); // 2.使用MapperProxy來構(gòu)建實例對象 return newInstance(mapperProxy);} protected T newInstance(MapperProxy mapperProxy) { // 使用JDK動態(tài)代理來代理要創(chuàng)建的實例對象,InvocationHandler為mapperProxy, // 因此當我們真正調(diào)用時,會走到mapperProxy的invoke方法 return (T) Proxy.newProxyInstance(mapperInterface.getClassLoader(), new Class[] { mapperInterface }, mapperProxy);}這邊代碼用到的 sqlSessionTemplate、mapperInterface 等都是之前添加的屬性。小結(jié),解析 DAO 文件 主要做了幾件事:1)通過 mapperInterface 從 knownMappers 緩存中獲取到 MapperProxyFactory 對象;2)通過 JDK 動態(tài)代理創(chuàng)建 MapperProxyFactory 實例對象,InvocationHandler 為 MapperProxy。4、DAO 接口被調(diào)用當 DAO 中的接口被調(diào)用時,會走到 MapperProxy 的 invoke 方法。@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable { try { if (Object.class.equals(method.getDeclaringClass())) { return method.invoke(this, args); } else { // 1.創(chuàng)建MapperMethodInvoker // 2.將method -> MapperMethodInvoker放到methodCache緩存 // 3.調(diào)用MapperMethodInvoker的invoke方法 return cachedInvoker(method).invoke(proxy, method, args, sqlSession); } } catch (Throwable t) { throw ExceptionUtil.unwrapThrowable(t); }} // MapperProxy.javaprivate MapperMethodInvoker cachedInvoker(Method method) throws Throwable { try { // 1.放到methodCache緩存,key為method,value為MapperMethodInvoker return methodCache.computeIfAbsent(method, m -> { if (m.isDefault()) { // 2.方法為默認方法,Java8之后,接口允許有默認方法 try { if (privateLookupInMethod == null) { return new DefaultMethodInvoker(getMethodHandleJava8(method)); } else { return new DefaultMethodInvoker(getMethodHandleJava9(method)); } } catch (IllegalAccessException | InstantiationException | InvocationTargetException | NoSuchMethodException e) { throw new RuntimeException(e); } } else { // 3.正常接口會走這邊,使用mapperInterface、method、configuration // 構(gòu)建一個MapperMethod,封裝成PlainMethodInvoker return new PlainMethodInvoker(new MapperMethod(mapperInterface, method, sqlSession.getConfiguration())); } }); } catch (RuntimeException re) { Throwable cause = re.getCause(); throw cause == null ? re : cause; }}3.調(diào)用 MapperMethodInvoker 的 invoke 方法,見代碼塊7。代碼塊7:invoke@Overridepublic Object invoke(Object proxy, Method method, Object[] args, SqlSession sqlSession) throws Throwable { return mapperMethod.execute(sqlSession, args);} // MapperMethod.javapublic Object execute(SqlSession sqlSession, Object[] args) { Object result; // 1.根據(jù)命令類型執(zhí)行來進行相應(yīng)操作 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()); } if (result == null && method.getReturnType().isPrimitive() && !method.returnsVoid()) { throw new BindingException("Mapper method '" + command.getName() + " attempted to return null from a method with a primitive return type (" + method.getReturnType() + ")."); } return result;}

這邊就比較簡單,根據(jù)不同的操作類型執(zhí)行相應(yīng)的操作,最終將結(jié)果返回,見代碼塊8。

這邊的 command 是上文 “new MapperMethod(mapperInterface, method, sqlSession.getConfiguration())” 時創(chuàng)建的。

代碼塊8:增刪改查

// 1.insert@Overridepublic int insert(String statement, Object parameter) { return update(statement, parameter);} // 2.update@Overridepublic int update(String statement, Object parameter) { try { dirty = true; // 從mappedStatements緩存拿到對應(yīng)的MappedStatement對象,執(zhí)行更新操作 MappedStatement ms = configuration.getMappedStatement(statement); return executor.update(ms, wrapCollection(parameter)); } catch (Exception e) { throw ExceptionFactory.wrapException("Error updating database. Cause: " + e, e); } finally { ErrorContext.instance().reset(); }} // 3.delete@Overridepublic int delete(String statement, Object parameter) { return update(statement, parameter);} // 4.select,以executeForMany為例private Object executeForMany(SqlSession sqlSession, Object[] args) { List result; // 1.參數(shù)轉(zhuǎn)換成sql命令參數(shù) Object param = method.convertArgsToSqlCommandParam(args); if (method.hasRowBounds()) { RowBounds rowBounds = method.extractRowBounds(args); result = sqlSession.selectList(command.getName(), param, rowBounds); } else { // 2.執(zhí)行查詢操作 result = sqlSession.selectList(command.getName(), param); } // 3.處理返回結(jié)果 // issue #510 Collections & arrays support if (!method.getReturnType().isAssignableFrom(result.getClass())) { if (method.getReturnType().isArray()) { return convertToArray(result); } else { return convertToDeclaredCollection(sqlSession.getConfiguration(), result); } } return result;} @Overridepublic List selectList(String statement, Object parameter) { return this.selectList(statement, parameter, RowBounds.DEFAULT);} @Overridepublic List selectList(String statement, Object parameter, RowBounds rowBounds) { try { //從mappedStatements緩存中拿到對應(yīng)的MappedStatement對象,執(zhí)行查詢操作 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(); }}

可以看出,最終都是從 mappedStatements 緩存中拿到對應(yīng)的 MappedStatement 對象,執(zhí)行相應(yīng)的操作。

這邊的增刪改查不是直接調(diào)用 SqlSession 中的方法,而是調(diào)用 SqlSessionTemplate 中的方法,繼而通過 sqlSessionProxy 來調(diào)用 SqlSession 中的方法。SqlSessionTemplate 中的方法主要是通過 sqlSessionProxy 做了一層動態(tài)代理,基本沒差別。

總結(jié)

整個流程主要是以下幾個核心步驟:

1)掃描注冊 basePackage 包下的所有 bean,將 basePackage 包下的所有 bean 進行一些特殊處理:beanClass 設(shè)置為 MapperFactoryBean、bean 的真正接口類作為構(gòu)造函數(shù)參數(shù)傳入 MapperFactoryBean、為 MapperFactoryBean 添加 sqlSessionFactory 和 sqlSessionTemplate屬性。

2)解析 mapperLocations 屬性的 mapper 文件,將 mapper 文件中的每個 SQL 封裝成 MappedStatement,放到 mappedStatements 緩存中,key 為 id,例如:com.joonwhee.open.mapper.UserPOMapper.queryByPrimaryKey,value 為 MappedStatement。并且將解析過的 mapper 文件的 namespace 放到 knownMappers 緩存中,key 為 namespace 對應(yīng)的 class,value 為 MapperProxyFactory。

3)創(chuàng)建 DAO 的 bean 時,通過 mapperInterface 從 knownMappers 緩存中獲取到 MapperProxyFactory 對象,通過 JDK 動態(tài)代理創(chuàng)建 MapperProxyFactory 實例對象,InvocationHandler 為 MapperProxy。

4)DAO 中的接口被調(diào)用時,通過動態(tài)代理,調(diào)用 MapperProxy 的 invoke 方法,最終通過 mapperInterface 從 mappedStatements 緩存中拿到對應(yīng)的 MappedStatement,執(zhí)行相應(yīng)的操作。

轉(zhuǎn)發(fā)至朋友圈,是對我最大的支持。

朕已閱??

總結(jié)

以上是生活随笔為你收集整理的.sql文件_面试题:mybatis 中的 DAO 接口和 XML 文件里的 SQL 是如何建立关系的?的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。

亚洲成a人片77777潘金莲 | 国产在线播放不卡 | 狠狠色噜噜狠狠狠狠2021天天 | 国产一二区在线观看 | 亚洲人成在线电影 | 亚洲欧洲日韩在线观看 | 一区二区精品在线 | 91精品婷婷国产综合久久蝌蚪 | 91女人18片女毛片60分钟 | 蜜臀久久99精品久久久无需会员 | 亚洲国产mv | 999久久久免费视频 午夜国产在线观看 | 日韩视频免费在线 | 久久久久国产一区二区三区 | 亚洲精品在线视频观看 | 国产精品99蜜臀久久不卡二区 | 不卡精品视频 | 97精品国产97久久久久久免费 | 手机在线黄色网址 | 手机av在线网站 | 久久福利国产 | 国产xvideos免费视频播放 | 久久er99热精品一区二区 | 欧美日韩国产伦理 | 国产v在线 | a天堂最新版中文在线地址 久久99久久精品国产 | 国产精品扒开做爽爽的视频 | 免费观看www7722午夜电影 | 99久久精品免费视频 | 久久精品成人热国产成 | 成人一级黄色片 | 日本在线视频网址 | 婷婷伊人综合 | 国产精品免费观看久久 | 亚洲欧美激情插 | 国产精品久久一卡二卡 | 亚洲国产欧美在线看片xxoo | 天天色天天草天天射 | 欧美视频在线观看免费网址 | 国产一级一片免费播放放 | 日日爱影视 | 亚洲激情p | 国产精品白虎 | 久久精品www人人爽人人 | 韩国av电影网 | 一区二区视 | 狠狠亚洲 | av网站在线观看免费 | 日韩欧美视频免费观看 | 999成人精品 | 午夜精品福利一区二区 | 激情综合亚洲精品 | 日精品在线观看 | 久草97| 亚洲少妇影院 | 久久精品亚洲一区二区三区观看模式 | 欧美色图视频一区 | 成年人在线电影 | 一区二区 不卡 | 精品久久精品 | 免费看片网页 | 成人av一区二区兰花在线播放 | 国产精品电影一区 | 亚洲欧美日韩中文在线 | 午夜18视频在线观看 | av观看在线观看 | 国产玖玖精品视频 | 依人成人综合网 | 国模精品一区二区三区 | 亚洲免费视频观看 | 久草网视频在线观看 | 麻豆成人网 | 最近中文字幕完整视频高清1 | 九九久久免费视频 | 国产精品一区久久久久 | 国产一区二区在线免费观看 | 国产老太婆免费交性大片 | 黄色小视频在线观看免费 | 日本精品一区二区三区在线播放视频 | 天天操天天操天天干 | 色www精品视频在线观看 | 国产精品视频全国免费观看 | 久久精品专区 | 91久久国产自产拍夜夜嗨 | 天天曰夜夜操 | 日韩理论电影在线观看 | 日韩视频一区二区三区 | 天天干天天干天天干天天干天天干天天干 | 成人黄大片视频在线观看 | 人交video另类hd | 欧美老少交| 粉嫩aⅴ一区二区三区 | 最近中文字幕 | 亚洲成av人影片在线观看 | 日韩黄色中文字幕 | 国产麻豆精品久久一二三 | 在线a视频免费观看 | 中文字幕成人一区 | 四虎国产视频 | 中文字幕在线日本 | 天天射天天 | 久久久久久久久久久国产精品 | 国产视频精品免费 | 国产一区在线免费 | 天天操天天操天天操天天操天天操 | 在线播放 一区 | 亚洲男人天堂a | 中文在线a√在线 | 国产精久久久久久妇女av | 免费看黄色大全 | 久久99国产精品免费 | 一区三区视频在线观看 | 色婷婷狠狠 | 亚洲四虎影院 | 亚洲精品乱码久久久久久蜜桃不爽 | 精品亚洲午夜久久久久91 | 丁香婷婷激情国产高清秒播 | 日日干综合 | 91人人在线 | 日韩精品在线观看视频 | 狠狠操操 | 91一区啪爱嗯打偷拍欧美 | 99精品在线免费观看 | 成人免费看片98欧美 | 久久精品aaa | 久久视频网址 | 国产免费观看av | 久久免费视频这里只有精品 | 国产精品成人久久久久久久 | 亚洲精选在线 | 亚洲精品在线观看网站 | 蜜臀av性久久久久av蜜臀妖精 | 国产精品视频999 | 国产亚洲永久域名 | 精品在线观看一区二区 | 99精品在线 | 天天操狠狠操夜夜操 | 婷婷综合导航 | 免费又黄又爽视频 | 偷拍视频一区 | 一区二区三区在线视频观看58 | aaa日本高清在线播放免费观看 | 亚洲va欧洲va国产va不卡 | 91视频91自拍| 亚洲天堂精品 | 亚洲 欧美 国产 va在线影院 | 亚洲欧美视频一区二区三区 | 黄网站色视频免费观看 | 精品一区二区免费 | 韩国精品一区二区三区六区色诱 | 九草视频在线 | 黄色毛片视频 | 欧美a在线看 | 香蕉视频在线免费看 | 日韩精品一区二区在线视频 | 在线观看免费黄视频 | 99久久www| 日韩理论视频 | 在线国产中文 | 九九热在线免费观看 | 中文字幕 国产专区 | 99精品视频在线观看视频 | 欧洲性视频 | 黄色国产高清 | 东方av在| 福利视频导航网址 | 亚洲丁香久久久 | 免费色视频网站 | 人人爱人人射 | 91成人精品一区在线播放69 | 免费亚洲黄色 | 国产精品一区免费在线观看 | 天天av资源 | 中文字幕av最新 | 九九九九热精品免费视频点播观看 | 婷婷激情五月 | 成人中文字幕av | 91.dizhi永久地址最新 | av片免费播放 | 亚洲国产精久久久久久久 | 久久久免费网站 | 免费看污黄网站 | 91成人看片 | 蜜臀av在线一区二区三区 | 操操操av | 蜜臀av夜夜澡人人爽人人桃色 | 香蕉影院在线观看 | 国产三级精品三级在线观看 | 一本一本久久aa综合精品 | 精品国产一区二区三区噜噜噜 | 高清一区二区 | 精品999| 嫩小bbbb摸bbb摸bbb | 久久黄色美女 | 精品久久国产精品 | 日韩专区在线 | 亚洲免费成人av电影 | 精品国产一区二区三区久久 | 在线激情影院一区 | 欧美日韩调教 | 色婷婷在线观看视频 | 婷婷色 亚洲| av黄色免费在线观看 | 国产伦精品一区二区三区免费 | www.久久久| 一区在线观看 | 天天草天天干天天射 | 黄色毛片网站在线观看 | 超碰免费久久 | 女人18毛片a级毛片一区二区 | 九色精品在线 | 日韩色中色 | 在线之家免费在线观看电影 | 狠狠色噜噜狠狠狠狠 | 国产日韩精品视频 | 天天操天天色天天 | 香蕉视频色 | 久久久国产精品人人片99精片欧美一 | 久久久精品视频成人 | 天天干天天拍天天操 | 在线亚洲高清视频 | 中文av一区二区 | www.xxxx欧美 | 天天草综合 | 亚洲日韩欧美一区二区在线 | 成人xxxx | 日韩午夜视频在线观看 | www.亚洲精品视频 | 国产精品一区二区麻豆 | 国产精品久久久久久久久久三级 | 欧美一级看片 | 夜夜爽88888免费视频4848 | 亚洲永久字幕 | 国产一级在线免费观看 | 免费看亚洲毛片 | 在线观看日韩专区 | 久久久91精品国产一区二区精品 | 九九久久国产精品 | 亚洲乱亚洲乱妇 | 国产中文字幕在线观看 | 亚洲精品午夜国产va久久成人 | 亚洲一区二区高潮无套美女 | 中文字幕麻豆 | 深爱开心激情网 | 欧美在线资源 | 久久伊人国产精品 | 成人精品在线 | 手机看片99 | 日韩中文在线字幕 | 国产成人精品一区在线 | 黄网站大全 | 亚洲综合色婷婷 | 五月婷婷开心 | 国产一级久久久 | 午夜美女福利直播 | 色综合网 | 日韩在观看线 | 中文字幕有码在线 | 中文视频在线播放 | 亚洲第一久久久 | 国产亚洲精品久久 | 久久久官网 | av不卡在线看 | 久久精品婷婷 | 一区二区三区电影 | 精品久久片 | 婷婷丁香社区 | 国产在线精品一区二区不卡了 | 一级黄色片毛片 | 人人搞人人干 | 狠狠干美女| 精品uu | 久久国产麻豆 | 久久综合影视 | 精品播放 | 99精品国产在热久久下载 | 国内精品亚洲 | 美女视频又黄又免费 | 91看片淫黄大片91 | 99国产一区二区三精品乱码 | 91中文在线观看 | 99色在线观看 | 久久成人免费视频 | 欧美日韩国产mv | 久草在线资源网 | 天天超碰 | 四虎国产精品免费观看视频优播 | 久久av在线 | 婷婷六月天丁香 | 国产高清专区 | 日韩欧美在线一区二区 | 少妇自拍av | 精品国产电影一区 | 91尤物国产尤物福利在线播放 | 视频在线精品 | 日韩大片在线免费观看 | 日韩午夜在线观看 | 99久久精品国 | 久色婷婷 | 国产一级a毛片视频爆浆 | 久久影院中文字幕 | 欧美福利在线播放 | 日韩性色 | 精品久久精品 | 久久久久国产免费免费 | 久久久久福利视频 | 久久久久北条麻妃免费看 | 超碰免费久久 | 亚洲国内精品在线 | 久久久久国产免费免费 | 国产69精品久久久久9999apgf | 国产一区二区三区高清播放 | 久草综合在线 | 久久久精品国产一区二区 | 人人爽人人搞 | 精品免费一区二区三区 | 美腿丝袜av | 日韩伦理片一区二区三区 | 91麻豆高清视频 | 欧美片网站yy| 亚洲综合色婷婷 | 婷婷伊人五月 | 视频在线一区二区三区 | 黄色免费大全 | 日韩在线免费看 | 一区二区三区免费在线 | 国模精品一区二区三区 | 天天se天天cao天天干 | 手机看片午夜 | 日韩欧美精品在线视频 | 日韩免费福利 | 日韩成人精品在线观看 | 国产精品久久二区 | 中文网丁香综合网 | 精品久久久久久久久久久久久 | 日韩一级片观看 | 91高清免费在线观看 | 成人a在线观看 | 国产精品一区二区在线 | 欧美调教网站 | bayu135国产精品视频 | 国产一区二区成人 | 日日夜夜网站 | 日韩中文字幕第一页 | 在线探花| 免费男女网站 | 日韩乱色精品一区二区 | 一区二区视频免费在线观看 | 欧美一级特黄高清视频 | 麻豆免费在线视频 | 国产精品乱码久久久久久1区2区 | 欧美人zozo | 久草视频99| 摸bbb搡bbb搡bbbb| 国产精品123| 在线日韩av| 久久精品99视频 | 亚洲视频精品在线 | 国产区精品视频 | 西西44人体做爰大胆视频 | 久久网站免费 | 欧美日韩一区二区三区在线观看视频 | 亚洲三级av | 精品久久精品久久 | 黄色大片国产 | 99精品乱码国产在线观看 | 亚洲干视频在线观看 | 人人爽人人爽人人爽 | 亚洲美女视频网 | 日本中文字幕观看 | 精品 激情 | 五月天激情综合 | 国产乱老熟视频网88av | 欧美色综合天天久久综合精品 | 在线观看岛国片 | 一区二区 不卡 | 国产私拍在线 | 精品一区二区视频 | 色综合小说 | 日韩欧美在线观看一区 | 久久综合福利 | 精品久久久久久久久中文字幕 | 日韩乱码中文字幕 | 亚洲一区av | 精品国产三级 | 中文字幕免费高清av | 成人超碰97 | 久久九九国产视频 | 综合久久精品 | 一级黄色大片 | 国产精品免费久久久久影院仙踪林 | 中文字幕在线观看av | 久久伊人精品一区二区三区 | 日韩高清成人在线 | 久草在线高清 | 99热这里只有精品国产首页 | 五月婷婷.com | 一区二区视频电影在线观看 | 中文字幕制服丝袜av久久 | 成人h动漫在线看 | av日韩不卡 | 在线精品视频免费播放 | 精品日韩中文字幕 | 欧美一区,二区 | 2019免费中文字幕 | 国产片免费在线观看视频 | 主播av在线 | 黄色在线看网站 | 日韩高清免费在线观看 | 免费看黄网站在线 | 伊人婷婷激情 | 99在线精品视频 | 亚洲成aⅴ人在线观看 | 97人人模人人爽人人喊中文字 | 91在线看黄 | 国产视频在线播放 | 91手机视频 | 国产亚洲精品久久久久秋 | 国产91学生| 国产免费观看久久黄 | 亚洲丝袜一区二区 | 狠狠夜夜 | 中文字幕在线观看完整 | 69精品在线观看 | 日韩理论视频 | 欧美日韩高清一区二区 国产亚洲免费看 | 国产精品成人av久久 | 中文在线亚洲 | 亚洲乱码中文字幕综合 | 一区二区亚洲精品 | 中文字幕在线免费播放 | 色综合久久88色综合天天免费 | 人人爱人人射 | 欧美一级在线观看视频 | av五月婷婷| 免费看片在线观看 | 亚洲精品五月天 | 91视频 - v11av| 日韩视频免费播放 | 亚洲高清资源 | 国产精品mm | 色噜噜日韩精品一区二区三区视频 | 极品久久久久久久 | 天海冀一区二区三区 | 日韩av免费在线电影 | 久草在线视频在线 | 99热国产精品 | 视频在线观看一区 | 亚洲精品久久久蜜桃直播 | 91精品久久久久久综合乱菊 | 成人午夜av电影 | 亚洲国产中文字幕在线观看 | 伊人婷婷在线 | 日韩成人精品一区二区 | 久久久久一区 | 成年人网站免费在线观看 | 久久69精品久久久久久久电影好 | 中文日韩在线视频 | 亚洲天堂网站视频 | 欧美日韩视频在线观看一区二区 | 99视频这里有精品 | 欧美日韩色婷婷 | 色中射 | 国产精品99视频 | 久草视频在线新免费 | 五月激情在线 | 国产亚洲精品久久久网站好莱 | 国产在线精品二区 | 欧美做受高潮 | 亚洲精品视频免费在线观看 | 精品久久久久一区二区国产 | 99久热在线精品视频成人一区 | 色综合久久天天 | 人人爽人人 | www·22com天天操| 婷婷激情综合五月天 | 亚洲视频观看 | 亚洲激情六月 | 亚洲最新av在线网址 | 亚洲国产精品推荐 | 精品久久久精品 | 西西4444www大胆视频 | 国产成人在线播放 | www黄色av| 奇米影视四色8888 | 亚洲国产精品视频在线观看 | 亚洲天堂香蕉 | 美女视频黄色免费 | 国产69精品久久久久久 | 国产精品普通话 | 国产亚洲欧美精品久久久久久 | 久久久久久高潮国产精品视 | 亚洲最大免费成人网 | 久久精品99国产精品 | 亚洲成人黄色 | 国产中文字幕在线免费观看 | 中文字幕黄色网址 | 九九视频网| 最近字幕在线观看第一季 | 日韩三级久久 | 99免费看片| 亚洲欧洲视频 | 欧美日一级片 | 免费视频99 | 久久久亚洲精华液 | 看国产黄色片 | 在线精品视频免费观看 | 最近能播放的中文字幕 | 亚洲国内在线 | 国产精品午夜av | 欧美日韩亚洲在线观看 | 国产亚洲午夜高清国产拍精品 | 欧美乱码精品一区二区 | 久久精品8| 狠狠干 狠狠操 | 成人午夜电影在线播放 | 日韩精品免费一区二区在线观看 | 天天射天天爱天天干 | 久99久精品视频免费观看 | av中文字幕亚洲 | 人人草人人草 | 午夜精品久久久久久久99婷婷 | 天天操夜夜操天天射 | 成人在线免费小视频 | 亚洲国产经典视频 | 中日韩欧美精彩视频 | 亚洲精品在线一区二区三区 | 亚洲视屏在线播放 | 午夜丰满寂寞少妇精品 | 免费一级毛毛片 | 日韩欧美成人网 | 丁香视频全集免费观看 | 草久中文字幕 | 国产精品视频大全 | 欧洲亚洲女同hd | 91精品在线免费观看 | 欧美 亚洲 另类 激情 另类 | 色综合天天狠天天透天天伊人 | 亚洲成人第一区 | 成人久久18免费网站 | 国产在线黄| 免费色视频在线 | 欧美成人一二区 | 日韩三级视频 | 亚洲精品视频免费看 | 中文字幕精品在线 | 精品国产一区二区久久 | 日本三级久久久 | 欧美极品在线播放 | 中字幕视频在线永久在线观看免费 | 欧美日韩精品网站 | 91网站在线视频 | 午夜精品一区二区三区可下载 | 夜夜骑日日操 | 色欧美成人精品a∨在线观看 | 999成人网| 成 人 a v天堂 | 特级西西444www大胆高清无视频 | 免费在线播放视频 | 亚洲一级电影在线观看 | 99一区二区三区 | 欧美日韩国产成人 | 五月天综合婷婷 | 国产中文视频 | 黄色免费在线看 | 亚洲四虎影院 | 97超视频在线观看 | 99久视频| 91在线九色 | 五月婷婷国产 | 黄色三级久久 | 91视频在线免费 | 日日夜夜天天 | 97视频久久久 | av网站播放 | 99视频在线观看一区三区 | 91精品视频网站 | 成人亚洲精品久久久久 | 丝袜美腿亚洲综合 | 精品在线观看国产 | 免费高清在线视频一区· | 在线观看免费黄色 | 在线观看亚洲专区 | 日本乱码在线 | 少妇bbbb揉bbbb日本 | 国产一区av在线 | 国产人成看黄久久久久久久久 | 激情久久一区二区三区 | 一区二区三区四区不卡 | 中文字幕免费在线看 | 欧美日韩一区二区三区在线观看视频 | 成人黄在线 | 特级西西444www高清大视频 | 91精品老司机久久一区啪 | 久久夜夜夜| 黄色av免费电影 | 日韩高清在线一区二区 | 少妇bbbb搡bbbb桶 | 久久观看免费视频 | 激情久久婷婷 | 91网在线 | 成人免费亚洲 | 国产美女视频黄a视频免费 久久综合九色欧美综合狠狠 | 久草网免费| 婷婷在线精品视频 | 成人av影视在线 | 日韩av网页| 国产视频手机在线 | 欧美大片第1页 | 国产中文字幕在线播放 | 婷婷精品国产欧美精品亚洲人人爽 | 亚洲高清视频在线观看免费 | 亚洲精品午夜久久久久久久 | 欧美激情视频一区二区三区免费 | 狠狠操.com | 天天操夜夜操天天射 | 欧美日韩久久不卡 | 久久国产精品第一页 | 2024av| 日日夜夜天天久久 | 免费在线观看的av网站 | 久久人人爽人人片 | 97看片| 天天久久夜夜 | 三级av黄色 | 日韩一区二区三区视频在线 | 婷婷久久亚洲 | av片中文字幕 | 国产高清在线免费视频 | 一级黄色免费网站 | 国内精品久久久久国产 | 国产精品综合久久久 | 久久久久综合网 | 欧美激情综合五月 | 久久综合久久久 | 久久观看最新视频 | 国产黄a三级三级 | 久久国产精彩视频 | 成人禁用看黄a在线 | 精品一二区 | 国产一区二区在线免费 | 制服丝袜欧美 | 久久日本视频 | 日韩免费在线视频观看 | 欧美日韩精品免费观看 | 在线看毛片网站 | 国产精品第一视频 | 国产美女在线精品免费观看 | 中文字幕av免费在线观看 | 亚洲国产精品传媒在线观看 | 久久国产精品久久w女人spa | 在线 视频 亚洲 | 久久精品在线 | 久久精品999 | 国产中文在线字幕 | 98精品国产自产在线观看 | 国产成人免费高清 | 不卡的av电影在线观看 | 久久精品视频中文字幕 | 91福利国产在线观看 | 午夜在线日韩 | 激情综合久久 | 成人在线观看免费 | 久久免费在线视频 | 欧美成人精品欧美一级乱黄 | 欧美日本中文字幕 | 中文字幕av在线不卡 | 久久精品这里精品 | 国产日韩中文在线 | 日韩av一区在线观看 | 天天爱天天操天天爽 | 成人手机在线视频 | 91最新网址在线观看 | 国产在线第三页 | 国产一二区免费视频 | 91黄视频在线 | 天天干天天在线 | 三级在线视频播放 | 日韩在线视频免费观看 | 五月婷婷丁香色 | 日韩com | 九九久久久久久久久激情 | 精品福利片 | 免费日韩高清 | 91视频黄色 | 日韩电影久久 | 国产精品一区二区久久精品 | 免费手机黄色网址 | 欧美日韩国产高清视频 | 精品久久国产精品 | 国外调教视频网站 | 日韩高清一区 | 国产99久久精品一区二区300 | 亚洲播放一区 | 超碰国产在线观看 | 天天天天干 | 日日日日 | 91视频链接| 国产视频二区三区 | 国产九色视频在线观看 | 国产玖玖精品视频 | 97香蕉超级碰碰久久免费软件 | 丁香五月亚洲综合在线 | 日韩视频在线观看视频 | 操操操av| 亚洲精品免费在线视频 | 日本99热 | 91在线影视 | 国产97av| 欧美va在线观看 | av一区二区三区在线播放 | 日本精品小视频 | 有码视频在线观看 | www在线观看国产 | 在线视频观看你懂的 | www黄在线| 亚洲精品国偷拍自产在线观看 | 97超碰资源 | 人人爽人人av | 亚洲精品免费看 | 午夜久久影视 | 久草在线费播放视频 | 久久精品视频网站 | 欧美精品久久久久久久久免 | 久久免费成人精品视频 | 色婷婷国产精品一区在线观看 | 久久艹艹 | 91精品视频在线观看免费 | 久99久精品视频免费观看 | 91超碰免费在线 | 91视频免费看 | 国产精品大片免费观看 | 日韩高清在线不卡 | 日韩一区二区三区高清在线观看 | 国产精品一区二区视频 | 国产看片 色| 欧美亚洲专区 | 久久99精品久久久久蜜臀 | a在线免费观看视频 | 国产免费激情久久 | 国产字幕在线播放 | 国产成人精品一区二三区 | 天天色天天骑天天射 | 国产欧美日韩视频 | 国产精品久久久久久久久蜜臀 | 婷婷 中文字幕 | 精品亚洲欧美无人区乱码 | 亚洲精色 | 91视频在线观看下载 | 天天干天天操天天搞 | 日韩免费观看av | 色综合 久久精品 | 久草网视频在线观看 | www.国产毛片 | 亚洲一区久久 | 日韩理论在线观看 | 日本精品久久久久中文字幕 | 九九av | 天天艹天天 | 在线免费观看一区二区三区 | 黄污视频网站 | 久久狠狠一本精品综合网 | 天天操天天干天天操天天干 | 绯色av一区 | 337p欧美| 超碰av在线 | 免费在线黄网 | 亚洲五月婷 | 六月婷婷久香在线视频 | 激情五月婷婷综合网 | 在线成人性视频 | 成人观看视频 | 久久久精品一区二区三区 | 国产剧情一区在线 | 亚洲精品国产精品国自产 | 国产精品九九久久久久久久 | 91亚洲精品久久久久图片蜜桃 | 天天综合色天天综合 | 免费国产一区二区 | 麻豆va一区二区三区久久浪 | 天天天天综合 | 国产伦理一区二区三区 | 午夜精品久久久久久久久久久久 | 狠狠操天天射 | 亚洲视频高清 | 激情久久久久久久久久久久久久久久 | 欧美va日韩va | 亚洲免费不卡 | 欧美日本一二三 | 中文字幕在线播放日韩 | 久青草视频 | 欧美激情精品久久久久久免费印度 | 国产专区免费 | 激情五月综合 | 韩国av电影网 | 免费日韩av电影 | 天天人人| 最近乱久中文字幕 | 中文字幕a在线 | 欧美性性网| 成 人 黄 色 视频 免费观看 | 亚洲伊人网在线观看 | av在线网站免费观看 | 欧美色综合久久 | 久久久久国产精品免费免费搜索 | 久久国产精品一区二区三区四区 | 欧美在线视频a | 在线观看视频h | 亚洲精品在线免费观看视频 | 最近中文字幕mv | 成人在线黄色 | 国产香蕉久久精品综合网 | 一区在线播放 | 久久综合免费视频 | 国产精品久久久久久久久久久免费看 | 日韩免费一级电影 | 一级片免费在线 | 黄色软件视频网站 | 黄色成人av网址 | 中文字幕免 | 午夜久久久久久久久久影院 | 国产视频 亚洲视频 | 69国产精品视频免费观看 | 一区二区在线电影 | 97视频在线免费观看 | 国产免费观看久久黄 | 视频一区二区视频 | 亚州国产精品 | 久久成人精品电影 | 国产日本在线观看 | 久久精品国产免费看久久精品 | 免费a视频| 欧美福利视频一区 | av三级在线免费观看 | 国内精品久久久久久久97牛牛 | 狠狠色丁香久久婷婷综 | 日韩中出在线 | 国产福利电影网址 | 麻豆视频www| 涩涩网站在线看 | 欧美va电影 | 草久久影院 | 精品国产欧美一区二区 | 国产69久久精品成人看 | 国产99久久精品一区二区永久免费 | 亚洲精品成人在线 | 色五月激情五月 | 色偷偷网站视频 | 精品在线观看一区二区三区 | 欧美视频日韩 | 日本黄色免费网站 | 福利电影久久 | 国产精品一区二区av日韩在线 | 国产手机av | 日韩xxxbbb| 国产一区二区三区网站 | 91九色精品女同系列 | 7777精品伊人久久久大香线蕉 | 成人国产精品免费观看 | 亚洲精品在线二区 | 日韩在线欧美在线 | 亚洲欧美视频 | 91在线视频免费播放 | 一区中文字幕电影 | 国产精品18久久久久久久久久久久 | 99视频在线看| 久草五月 | 六月久久婷婷 | 国产真实精品久久二三区 | 免费在线色视频 | 亚洲精品视频网站在线观看 | 波多野结衣一区二区 | 免费网址在线播放 | 免费观看一级 | 欧美日韩在线播放一区 | 亚洲精品视频在线 | 在线观看亚洲精品视频 | 精品国产成人av | 亚洲精品视频网站在线观看 | 激情亚洲综合在线 | 91九色在线 | 久久久亚洲国产精品麻豆综合天堂 | 国产美腿白丝袜足在线av | 91精品国产91热久久久做人人 | 久久99久久精品 | 久久天天躁狠狠躁夜夜不卡公司 | 在线视频1卡二卡三卡 | 国产精品久久麻豆 | 久久久综合色 | 色悠悠久久综合 | 欧美日本国产在线观看 | 日日夜夜天天人人 | 国产自产高清不卡 | 91在线视频免费观看 | 国产99久久久国产精品免费看 | 亚洲精品在线电影 | 99久久99久久精品免费 | 天天干天天操天天做 | www黄免费| 91亚洲欧美激情 | 欧美精品一区二区三区一线天视频 | 成人a级免费视频 | 国产中文字幕在线免费观看 | 99精品国产在热久久 | 色婷婷电影网 | 欧美激情片在线观看 | 黄色av观看 | 婷婷成人综合 | 九九视频这里只有精品 | 中文字幕免费 | av在线播放一区二区三区 | av大片网站| 国产亚洲一级高清 | 欧美做受高潮 | 欧美精品久久久久久久久久久 | 久久精品一区二区三 | 国产日产精品一区二区三区四区的观看方式 | 最近中文字幕mv免费高清在线 | 国产精品久久久久9999吃药 | 久久免费黄色 | 在线国产精品视频 | 亚洲精品久 | 婷婷伊人五月天 | 黄色在线免费观看网址 | 日日草夜夜操 | 综合铜03 | 日本少妇高清做爰视频 | 中文字字幕在线 | 国产精品1区2区在线观看 | 亚洲精品国偷拍自产在线观看蜜桃 | 免费黄色在线播放 | 久久伊人国产精品 | 亚洲综合激情网 | av片一区二区 | 一区二区三区在线免费 | 久久99久久99精品免费看小说 | 亚洲欧美激情插 | 婷婷成人亚洲综合国产xv88 | 日韩二区在线 | 日本女人b | 婷婷色av | 亚洲欧洲精品在线 | 亚洲免费一级电影 | 中文字幕黄网 | 久久九精品 | 永久中文字幕 | 国产在线精品区 | 久久精品三级 | 国产一级视频在线免费观看 | 五月激情六月丁香 | 五月婷婷色综合 | 碰超在线观看 | 久久都是精品 | 91av短视频 | 天天躁日日躁狠狠躁av麻豆 | 超碰97国产在线 | 1024手机在线看| 国产精品18久久久久久vr | av免费看在线 | 久草免费在线观看 | 色婷婷一区| 久久精品中文字幕免费mv | 在线国产不卡 | 伊人在线视频 | 亚洲天堂在线观看完整版 | 久草在线中文视频 | 午夜久久久久久久久久久 | 日韩精品视频免费专区在线播放 | 国产精品 中文字幕 亚洲 欧美 | 麻豆免费在线播放 | 国产欧美综合在线观看 | 激情久久伊人 | 97操碰 | av电影中文字幕在线观看 | 国产精品igao视频网入口 | 成人免费视频网址 | 久久久久久免费毛片精品 | 免费日韩 精品中文字幕视频在线 | 97在线看片| 91av视频免费在线观看 | 日韩欧美在线综合网 | 日日日干 | 中国一级片在线观看 | 成人免费在线观看av | 国产精品久久伊人 | 国产精品久99 | 欧美黑人xxxx猛性大交 | 中文网丁香综合网 | 久草在线手机视频 | 亚洲国产日韩精品 | 成人av电影免费在线观看 | 国产码电影 | 午夜资源站 | 亚洲成a人片在线观看网站口工 | 色成人亚洲 | 91桃色在线观看视频 | 日日夜夜操av | 欧美日韩激情视频8区 |