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

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

【spring源码分析】IOC容器初始化(二)

發布時間:2024/4/17 编程问答 41 豆豆
生活随笔 收集整理的這篇文章主要介紹了 【spring源码分析】IOC容器初始化(二) 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

前言:在【spring源碼分析】IOC容器初始化(一)文末中已經提出loadBeanDefinitions(DefaultListableBeanFactory)的重要性,本文將以此為切入點繼續分析。


AbstractXmlApplicationContext#loadBeanDefinitions(DefaultListableBeanFactory)

1 protected void loadBeanDefinitions(DefaultListableBeanFactory beanFactory) throws BeansException, IOException { 2 // Create a new XmlBeanDefinitionReader for the given BeanFactory. 3 // 創建XmlBeanDefinitionReader對象 4 XmlBeanDefinitionReader beanDefinitionReader = new XmlBeanDefinitionReader(beanFactory); 5 6 // Configure the bean definition reader with this context's 7 // resource loading environment. 8 // 對XmlBeanDefinitionReader進行環境變量的設置 9 beanDefinitionReader.setEnvironment(this.getEnvironment()); 10 beanDefinitionReader.setResourceLoader(this); 11 beanDefinitionReader.setEntityResolver(new ResourceEntityResolver(this)); 12 13 // Allow a subclass to provide custom initialization of the reader, 14 // then proceed with actually loading the bean definitions. 15 // 對XmlBeanDefinitionReader進行設置,可以進行覆蓋 16 initBeanDefinitionReader(beanDefinitionReader); 17 // 從Resource中加載BeanDefinition 18 loadBeanDefinitions(beanDefinitionReader); 19 }

分析:

  • 首先創建一個XmlBeanDefinitionReader對象,因為我們需要解析xml文件,然后將其封裝成BeanDefinition。
  • 設置XmlBeanDefinitionReader對象的相關屬性,這里著重關注ResourceLoader,這里引申出Resource/ResourceLoader體系。
  • 從Resource中加載BeanDefinition。

Resource體系

Resource繼承InputStreamSource,為spring框架所有資源的訪問提供抽象接口,子類AbstractResource提供Resource接口的默認實現。

ResourceLoader體系

ResourceLoader為spring資源加載的統一抽象,主要應用于根據給定的資源文件地址返回相應的Resource對象,其具體的實現由相應子類去負責。

這里列出筆者認為的幾個比較重要ResourceLoader的實現類。

  • DefaultResourceLoader是ResourceLoader的默認實現
  • PathMatchingResourcePatternResolver,該類比較常用,除了支持"classpath*:"格式,還支持Ant風格的路徑匹配模式

接下來進入AbstractXmlApplicationContext#loadBeanDefinitions方法

1 protected void loadBeanDefinitions(XmlBeanDefinitionReader reader) throws BeansException, IOException { 2 // 從配置文件Resource中,加載BeanDefinition 3 Resource[] configResources = getConfigResources(); 4 if (configResources != null) { 5 reader.loadBeanDefinitions(configResources); 6 } 7 // 從配置文件地址中,加載BeanDefinition 8 String[] configLocations = getConfigLocations(); 9 if (configLocations != null) { 10 reader.loadBeanDefinitions(configLocations); 11 } 12 }

分析:

看到這里是否很熟悉因為我們在【spring源碼分析】IOC容器初始化(一)中已經設置了資源文件的路徑(setConfigLocations)方法,因此這里會直接走到第9行處,然后調用AbstractBeanDefinitionReader#loadBeanDefinitions方法:

1 public int loadBeanDefinitions(String... locations) throws BeanDefinitionStoreException { 2 Assert.notNull(locations, "Location array must not be null"); 3 int count = 0; 4 for (String location : locations) { 5 count += loadBeanDefinitions(location); 6 } 7 return count; 8 }

分析:

這里會遍歷locations,并返回最終加載bean的個數,函數最終切入點:AbstractBeanDefinitionReader#loadBeanDefinitions(String location, @Nullable Set<Resource> actualResources):

1 public int loadBeanDefinitions(String location, @Nullable Set<Resource> actualResources) throws BeanDefinitionStoreException { 2 // 獲取ResourceLoader對象 3 ResourceLoader resourceLoader = getResourceLoader(); 4 // 資源加載器為null,拋出異常 5 if (resourceLoader == null) { 6 throw new BeanDefinitionStoreException( 7 "Cannot load bean definitions from location [" + location + "]: no ResourceLoader available"); 8 } 9 10 // 如果當前ResourceLoader為匹配模式形式的[支持一個location返回Resource[]數組形式] 11 if (resourceLoader instanceof ResourcePatternResolver) { 12 // Resource pattern matching available. 13 try { 14 // 通過location返回Resource[]數組,通過匹配模式形式,可能存在多個Resource 15 Resource[] resources = ((ResourcePatternResolver) resourceLoader).getResources(location); 16 // 加載BeanDefinition,返回BeanDefinition加載的個數 17 int count = loadBeanDefinitions(resources); 18 // 將Resource[] 添加到actualResources中 19 if (actualResources != null) { 20 Collections.addAll(actualResources, resources); 21 } 22 if (logger.isTraceEnabled()) { 23 logger.trace("Loaded " + count + " bean definitions from location pattern [" + location + "]"); 24 } 25 // 返回BeanDefinition加載的個數 26 return count; 27 } catch (IOException ex) { 28 throw new BeanDefinitionStoreException( 29 "Could not resolve bean definition resource pattern [" + location + "]", ex); 30 } 31 // ResourceLoader為默認資源加載器,一個location返回一個Resource 32 } else { 33 // Can only load single resources by absolute URL. 34 Resource resource = resourceLoader.getResource(location); 35 // 加載BeanDefinition,并返回加載BeanDefinition的個數 36 int count = loadBeanDefinitions(resource); 37 // 將Resource添加到actualResources中 38 if (actualResources != null) { 39 actualResources.add(resource); 40 } 41 if (logger.isTraceEnabled()) { 42 logger.trace("Loaded " + count + " bean definitions from location [" + location + "]"); 43 } 44 // 返回BeanDefinition加載的個數 45 return count; 46 } 47 }

分析:

  • 首先獲取ResourceLoader,ResourceLoader的賦值在創建XmlBeanDefinitionReader的過程中,如果未指定則會創建一個PathMatchingResourcePatternResolver對象。
  • 然后根據對應的ResourceLoader返回的Resource對象。

關注第15行代碼,getResources(String)方法,這里會直接委托給PathMatchingResourcePatternResolver#getResources(String)進行處理:

1 public Resource[] getResources(String locationPattern) throws IOException { 2 Assert.notNull(locationPattern, "Location pattern must not be null"); 3 4 // 以"classpath*:"開頭的location 5 if (locationPattern.startsWith(CLASSPATH_ALL_URL_PREFIX)) { 6 // a class path resource (multiple resources for same name possible) 7 8 // #1.isPattern函數的入參為路徑 9 // #2.所以這里判斷路徑是否包含通配符 如com.develop.resource.* 10 if (getPathMatcher().isPattern(locationPattern.substring(CLASSPATH_ALL_URL_PREFIX.length()))) { 11 // a class path resource pattern 12 // 這里通過通配符返回Resource[] 13 return findPathMatchingResources(locationPattern); 14 // 路徑不包含通配符 15 } else { 16 // all class path resources with the given name 17 // 通過給定的路徑,找到所有匹配的資源 18 return findAllClassPathResources(locationPattern.substring(CLASSPATH_ALL_URL_PREFIX.length())); 19 } 20 // 不以"classpath*:" 21 } else { 22 // Generally only look for a pattern after a prefix here, 23 // and on Tomcat only after the "*/" separator for its "war:" protocol. 24 // 通常在這里只是通過前綴后面進行查找,并且在tomcat中只有在"*/"分隔符之后才是其"war:"協議 25 // #1.如果是以"war:"開頭,定位其前綴位置 26 // #2.如果不是以"war:"開頭,則prefixEnd=0 27 int prefixEnd = (locationPattern.startsWith("war:") ? locationPattern.indexOf("*/") + 1 : 28 locationPattern.indexOf(':') + 1); 29 // 判斷路徑中是否含有通配符否含有通配符 30 if (getPathMatcher().isPattern(locationPattern.substring(prefixEnd))) { 31 // a file pattern 32 // 通過通配符返回返回Resource[] 33 return findPathMatchingResources(locationPattern); 34 // 路徑不包含通配符 35 } else { 36 // a single resource with the given name 37 // 通過給定的location返回一個Resource,封裝成數組形式 38 // 獲取Resource的過程都是通過委托給相應的ResourceLoader實現 39 return new Resource[]{getResourceLoader().getResource(locationPattern)}; 40 } 41 } 42 }

分析:

首先兩大分支:根據資源路徑是否包含"classpath*:"進行處理。

#1."classpath*:"分支:

  • 首先判斷路徑中是否含有通配符"*"或"?",然后執行findPathMatchingResources函數。
  • 如果不包含通配符,則根據路徑找到所有匹配的資源,執行findAllClassPathResources函數。

#2.路徑中不含"classpath*:"分支,與上述過程一樣,同樣按分支含有通配符與不含通配符進行處理。

PathMatchingResourcePatternResolver#findPathMatchingResources(String)

1 protected Resource[] findPathMatchingResources(String locationPattern) throws IOException { 2 // 確定根路徑與子路徑 3 String rootDirPath = determineRootDir(locationPattern); 4 String subPattern = locationPattern.substring(rootDirPath.length()); 5 // 得到根路徑下的資源 6 Resource[] rootDirResources = getResources(rootDirPath); 7 Set<Resource> result = new LinkedHashSet<>(16); 8 // 遍歷獲取資源 9 for (Resource rootDirResource : rootDirResources) { 10 // 解析根路徑資源 11 rootDirResource = resolveRootDirResource(rootDirResource); 12 URL rootDirUrl = rootDirResource.getURL(); 13 // bundle類型資源 14 if (equinoxResolveMethod != null && rootDirUrl.getProtocol().startsWith("bundle")) { 15 URL resolvedUrl = (URL) ReflectionUtils.invokeMethod(equinoxResolveMethod, null, rootDirUrl); 16 if (resolvedUrl != null) { 17 rootDirUrl = resolvedUrl; 18 } 19 rootDirResource = new UrlResource(rootDirUrl); 20 } 21 // vfs類型資源 22 if (rootDirUrl.getProtocol().startsWith(ResourceUtils.URL_PROTOCOL_VFS)) { 23 result.addAll(VfsResourceMatchingDelegate.findMatchingResources(rootDirUrl, subPattern, getPathMatcher())); 24 // jar類型資源 25 } else if (ResourceUtils.isJarURL(rootDirUrl) || isJarResource(rootDirResource)) { 26 result.addAll(doFindPathMatchingJarResources(rootDirResource, rootDirUrl, subPattern)); 27 // 其他類型資源 28 } else { 29 result.addAll(doFindPathMatchingFileResources(rootDirResource, subPattern)); 30 } 31 } 32 if (logger.isDebugEnabled()) { 33 logger.debug("Resolved location pattern [" + locationPattern + "] to resources " + result); 34 } 35 // 將結果封裝成數組形式 注意該轉換形式 36 return result.toArray(new Resource[0]); 37 }

分析:

函數的整體處理邏輯比較簡單,根據不同的資源類型,將資源最終轉換為Resource數組。

特別分析:

determineRootDir(String)

1 protected String determineRootDir(String location) { 2 // 確定":"的后一位,如果":"不存在,則prefixEnd=0 3 int prefixEnd = location.indexOf(':') + 1; 4 // location的長度 5 int rootDirEnd = location.length(); 6 // 從location的":"開始(可能不存在)一直到location結束,判斷是否包含通配符,如果存在,則截取最后一個"/"分割的部分 7 /** 8 * 截取過程: 9 * classpath*:com/dev/config/* 10 * prefixEnd=11 11 * subString(prefixEnd,rootDirEnd)=com/dev/config/* 12 * 第一次循環rootDirEnd=26,也就是最后一個"/" 13 * subString(prefixEnd,rootDirEnd)=com/dev/config/ 14 * 第二次循環已經不包含通配符了,跳出循環 15 * 所以根路徑為classpath*:com/dev/config/ 16 */ 17 while (rootDirEnd > prefixEnd && getPathMatcher().isPattern(location.substring(prefixEnd, rootDirEnd))) { 18 // 確定最后一個"/"位置的后一位,注意這里rootDirEnd-2是為了縮小搜索范圍,提升速度 19 rootDirEnd = location.lastIndexOf('/', rootDirEnd - 2) + 1; 20 } 21 // 如果查找完后rootDirEnd=0,則將prefixEnd賦值給rootDirEnd,也就是冒號的后一位 22 if (rootDirEnd == 0) { 23 rootDirEnd = prefixEnd; 24 } 25 // 截取根目錄 26 return location.substring(0, rootDirEnd); 27 }

分析:

該函數有點繞,整體思想就是決定出給定資源路徑的根路徑,代碼中已經給出了詳細注釋,處理效果如下實例:

PathMatchingResourcePatternResolver#findAllClassPathResources(String)

1 protected Resource[] findAllClassPathResources(String location) throws IOException { 2 String path = location; 3 // location是否已"/"開頭 4 if (path.startsWith("/")) { 5 path = path.substring(1); 6 } 7 // 真正加載location下所有classpath下的資源 8 Set<Resource> result = doFindAllClassPathResources(path); 9 if (logger.isDebugEnabled()) { 10 logger.debug("Resolved classpath location [" + location + "] to resources " + result); 11 } 12 return result.toArray(new Resource[0]); 13 }

分析:

該函數會查找路徑下的所有資源,核心函數doFindAllClassPathResources(String):

1 protected Set<Resource> doFindAllClassPathResources(String path) throws IOException { 2 Set<Resource> result = new LinkedHashSet<>(16); 3 ClassLoader cl = getClassLoader(); 4 // 根據ClassLoader來加載資源 5 // 如果PathMatchingResourcePatternResolver在初始化時,設置了ClassLoader,就用該ClassLoader的getResouce方法 6 // 否則調用ClassLoader的getSystemResource方法 7 Enumeration<URL> resourceUrls = (cl != null ? cl.getResources(path) : ClassLoader.getSystemResources(path)); 8 // 遍歷集合將集合轉換成UrlResource形式 9 // 如果path為空,這里就會返回項目中classes的路徑,通過addAllClassLoaderJarRoots方法進行加載 10 while (resourceUrls.hasMoreElements()) { 11 URL url = resourceUrls.nextElement(); 12 result.add(convertClassLoaderURL(url)); 13 } 14 // 如果path為空,則加載路徑下的所有jar 15 if ("".equals(path)) { 16 // The above result is likely to be incomplete, i.e. only containing file system references. 17 // We need to have pointers to each of the jar files on the classpath as well... 18 // 加載所有jar 19 addAllClassLoaderJarRoots(cl, result); 20 } 21 return result; 22 }

分析:該函數的主要功能就是將搜索配置文件路徑下的所有資源,然后封裝成Resource集合返回,供加載BeanDefinition使用。

不含"classpath*:"分支的邏輯與上述分析差不多,這里不再做過多贅述。

Resource資源準備就緒后,再次回到loadBeanDefinitions(String location, @Nullable Set<Resource> actualResources)函數中,在第17行代碼處進入正式加載BeanDefinition過程。

AbstractBeanDefinitionReader#loadBeanDefinitions(Resource... resources)

1 public int loadBeanDefinitions(Resource... resources) throws BeanDefinitionStoreException { 2 Assert.notNull(resources, "Resource array must not be null"); 3 int count = 0; 4 // 通過循環的形式單個加載BeanDefinition 5 for (Resource resource : resources) { 6 count += loadBeanDefinitions(resource); 7 } 8 return count; 9 }

在循環過程中會落入XmlBeanDefinitionReader#loadBeanDefinitions(Resource resource)

1 public int loadBeanDefinitions(Resource resource) throws BeanDefinitionStoreException { 2 // 這里會將Resource封裝成EncodeResource,主要主要為了內容讀取的正確性 3 return loadBeanDefinitions(new EncodedResource(resource)); 4 }

該函數將Resource封裝成EncodeResource,主要是為了內容讀取的正確性,然后進入加載BeanDefinition的核心函數XmlBeanDefinitionReader#loadBeanDefinitions(EncodedResource encodedResource)

1 public int loadBeanDefinitions(EncodedResource encodedResource) throws BeanDefinitionStoreException { 2 Assert.notNull(encodedResource, "EncodedResource must not be null"); 3 if (logger.isInfoEnabled()) { 4 logger.info("Loading XML bean definitions from " + encodedResource.getResource()); 5 } 6 7 // 獲取已經加載過的資源 8 Set<EncodedResource> currentResources = this.resourcesCurrentlyBeingLoaded.get(); 9 // 表示當前沒有資源加載 10 if (currentResources == null) { 11 currentResources = new HashSet<>(4); 12 this.resourcesCurrentlyBeingLoaded.set(currentResources); 13 } 14 // 將當前資源加入記錄中,如果已經存在,則拋出異常,因為currentResource為Set集合 15 // 這里主要為了避免一個EncodeResource還沒加載完成時,又加載本身,造成死循環(Detected cyclic loading of) 16 if (!currentResources.add(encodedResource)) { 17 throw new BeanDefinitionStoreException( 18 "Detected cyclic loading of " + encodedResource + " - check your import definitions!"); 19 } 20 try { 21 // 從封裝的encodeResource中獲取resource,并取得其輸入流,通過流對資源進行操作 22 InputStream inputStream = encodedResource.getResource().getInputStream(); 23 try { 24 // 將流封裝成InputSource 25 InputSource inputSource = new InputSource(inputStream); 26 // 設置InputSource的編碼 27 if (encodedResource.getEncoding() != null) { 28 inputSource.setEncoding(encodedResource.getEncoding()); 29 } 30 // 核心邏輯,實現BeanDefinition的加載 31 return doLoadBeanDefinitions(inputSource, encodedResource.getResource()); 32 } finally { 33 inputStream.close(); 34 } 35 } catch (IOException ex) { 36 throw new BeanDefinitionStoreException( 37 "IOException parsing XML document from " + encodedResource.getResource(), ex); 38 } finally { 39 // 最后從緩存中清除資源 40 currentResources.remove(encodedResource); 41 // 如果當前資源集合為空,則從EncodeResource集合中移除當前資源的集合 42 if (currentResources.isEmpty()) { 43 this.resourcesCurrentlyBeingLoaded.remove(); 44 } 45 } 46 }

分析:

  • 首先判斷緩存中是否已經存在當前資源,如果存在則拋出異常,這里是為了避免循環加載。
  • 然后取出文件流封裝成InputSource,進入加載BeanDefinition的核心函數doLoadBeanDefinitions。
1 protected int doLoadBeanDefinitions(InputSource inputSource, Resource resource) 2 throws BeanDefinitionStoreException { 3 try { 4 // #1.獲取XML的Document實例 5 Document doc = doLoadDocument(inputSource, resource); 6 // #2.根據Document注冊bean,并返回注冊的bean的個數 7 return registerBeanDefinitions(doc, resource); 8 } catch (BeanDefinitionStoreException ex) { 9 throw ex; 10 } catch (SAXParseException ex) { 11 throw new XmlBeanDefinitionStoreException(resource.getDescription(), 12 "Line " + ex.getLineNumber() + " in XML document from " + resource + " is invalid", ex); 13 } catch (SAXException ex) { 14 throw new XmlBeanDefinitionStoreException(resource.getDescription(), 15 "XML document from " + resource + " is invalid", ex); 16 } catch (ParserConfigurationException ex) { 17 throw new BeanDefinitionStoreException(resource.getDescription(), 18 "Parser configuration exception parsing XML from " + resource, ex); 19 } catch (IOException ex) { 20 throw new BeanDefinitionStoreException(resource.getDescription(), 21 "IOException parsing XML document from " + resource, ex); 22 } catch (Throwable ex) { 23 throw new BeanDefinitionStoreException(resource.getDescription(), 24 "Unexpected exception parsing XML document from " + resource, ex); 25 } 26 }

分析:

  • 首先獲取XML配置文件的Document實例。
  • 根據Document注冊Bean,并返回注冊Bean的個數。

XmlBeanDefinitionReader#doLoadDocument(InputSource inputSource, Resource resource)

protected Document doLoadDocument(InputSource inputSource, Resource resource) throws Exception {return this.documentLoader.loadDocument(inputSource, getEntityResolver(), this.errorHandler,getValidationModeForResource(resource), isNamespaceAware());}

分析:

這里是委派給DefaultDocumentLoader#loadDocument函數來實現。

這里有一個驗證模式的入參,從getValidationModeForResource函數而來:

1 protected int getValidationModeForResource(Resource resource) { 2 // 獲取指定的驗證模式,默認為自動模式 3 int validationModeToUse = getValidationMode(); 4 // #1.如果驗證模式不為自動驗證模式,則表示進行了設置,則直接返回驗證模式即可 5 if (validationModeToUse != VALIDATION_AUTO) { 6 return validationModeToUse; 7 } 8 // #2.到這里表示使用了自動驗證模式,再次檢測Resource使用的驗證模式 9 int detectedMode = detectValidationMode(resource); 10 if (detectedMode != VALIDATION_AUTO) { 11 return detectedMode; 12 } 13 // 最后使用默認的VALIDATION_XSD驗證模式 14 // Hmm, we didn't get a clear indication... Let's assume XSD, 15 // since apparently no DTD declaration has been found up until 16 // detection stopped (before finding the document's root tag). 17 return VALIDATION_XSD; 18 }

分析:

  • 首先獲取當前的驗證模式,默認為自動驗證模式。
  • 如果當前驗證模式不為自動驗證模式,則表示進行了設置,則直接返回當前驗證模式即可。
  • 如果使用了自動驗證模式,則需再次檢測Resource使用的驗證模式
  • 最后,如果還是自動驗證模式,則返回XSD驗證模式。

這里要科普一下DTD與XSD

DTD(Document Type Definition),即文檔類型定義,為 XML 文件的驗證機制,屬于 XML 文件中組成的一部分。DTD 是一種保證 XML 文檔格式正確的有效驗證方式,它定義了相關 XML 文檔的元素、屬性、排列方式、元素的內容類型以及元素的層次結構。其實 DTD 就相當于 XML 中的 “詞匯”和“語法”,我們可以通過比較 XML 文件和 DTD 文件 來看文檔是否符合規范,元素和標簽使用是否正確。

但是DTD存在著一些缺陷:

  • 它沒有使用 XML 格式,而是自己定義了一套格式,相對解析器的重用性較差;而且 DTD 的構建和訪問沒有標準的編程接口,因而解析器很難簡單的解析 DTD 文檔。
  • DTD 對元素的類型限制較少;同時其他的約束力也叫弱。
  • DTD 擴展能力較差。
  • 基于正則表達式的 DTD 文檔的描述能力有限。

針對 DTD 的缺陷,W3C 在 2001 年推出 XSD。XSD(XML Schemas Definition)即 XML Schema 語言。XML Schema 本身就是一個 XML文檔,使用的是 XML 語法,因此可以很方便的解析 XSD 文檔。相對于 DTD,XSD 具有如下優勢:

  • XML Schema 基于 XML ,沒有專門的語法。
  • XML Schema 可以象其他 XML 文件一樣解析和處理。
  • XML Schema 比 DTD 提供了更豐富的數據類型。
  • XML Schema 提供可擴充的數據模型。
  • XML Schema 支持綜合命名空間。
  • XML Schema 支持屬性組。

spring中定義了一些驗證模式:

/*** Indicates that the validation should be disabled. 禁用驗證模式*/public static final int VALIDATION_NONE = XmlValidationModeDetector.VALIDATION_NONE;/*** Indicates that the validation mode should be detected automatically. 自動獲取驗證模式*/public static final int VALIDATION_AUTO = XmlValidationModeDetector.VALIDATION_AUTO;/*** Indicates that DTD validation should be used. DTD驗證模式*/public static final int VALIDATION_DTD = XmlValidationModeDetector.VALIDATION_DTD;/*** Indicates that XSD validation should be used. XSD驗證模式*/public static final int VALIDATION_XSD = XmlValidationModeDetector.VALIDATION_XSD;

XmlBeanDefinitionReader#detectValidationMode(Resource resource)函數是檢測資源文件的驗證模式的:

1 protected int detectValidationMode(Resource resource) { 2 // 如果資源已經被打開,則直接拋出異常 3 if (resource.isOpen()) { 4 throw new BeanDefinitionStoreException( 5 "Passed-in Resource [" + resource + "] contains an open stream: " + 6 "cannot determine validation mode automatically. Either pass in a Resource " + 7 "that is able to create fresh streams, or explicitly specify the validationMode " + 8 "on your XmlBeanDefinitionReader instance."); 9 } 10 11 // 打開InputStream流 12 InputStream inputStream; 13 try { 14 inputStream = resource.getInputStream(); 15 } catch (IOException ex) { 16 throw new BeanDefinitionStoreException( 17 "Unable to determine validation mode for [" + resource + "]: cannot open InputStream. " + 18 "Did you attempt to load directly from a SAX InputSource without specifying the " + 19 "validationMode on your XmlBeanDefinitionReader instance?", ex); 20 } 21 22 try { 23 // 檢測InputStream到底使用哪一種驗證模式 24 // 核心邏輯 25 return this.validationModeDetector.detectValidationMode(inputStream); 26 } catch (IOException ex) { 27 throw new BeanDefinitionStoreException("Unable to determine validation mode for [" + 28 resource + "]: an error occurred whilst reading from the InputStream.", ex); 29 } 30 }

其核心功能:檢測資源文件的驗證模式是委托給XmlValidationModeDetector#detectValidationMode(InputStream inputStream)

1 public int detectValidationMode(InputStream inputStream) throws IOException { 2 // 將InputStream進行包裝,便于讀取 3 // Peek into the file to look for DOCTYPE. 4 BufferedReader reader = new BufferedReader(new InputStreamReader(inputStream)); 5 try { 6 // 是否為DTD驗證模式,默認為false,即不是DTD驗證模式,那就是XSD驗證模式 7 boolean isDtdValidated = false; 8 String content; 9 // 循環讀取xml資源的內容 10 while ((content = reader.readLine()) != null) { 11 // 消費注釋內容,返回有用信息 12 content = consumeCommentTokens(content); 13 // 如果為注釋,或者為空,則繼續循環 14 if (this.inComment || !StringUtils.hasText(content)) { 15 continue; 16 } 17 // #1.如果包含"DOCTYPE",則為DTD驗證模式 18 if (hasDoctype(content)) { 19 isDtdValidated = true; 20 break; 21 } 22 // #2.該方法會校驗,內容中是否有"<",并且"<"后面還跟著字母,如果是則返回true 23 // 如果為true,最終就是XSD模式 24 if (hasOpeningTag(content)) { 25 // End of meaningful data... 26 break; 27 } 28 } 29 // 返回DTD模式或XSD模式 30 return (isDtdValidated ? VALIDATION_DTD : VALIDATION_XSD); 31 } catch (CharConversionException ex) { 32 // Choked on some character encoding... 33 // Leave the decision up to the caller. 34 // 如果發生異常,則返回自動驗證模式 35 return VALIDATION_AUTO; 36 } finally { 37 reader.close(); 38 } 39 }

分析:

這里會遍歷資源的內容進行文件驗證模式的判斷

  • consumeCommentTokens(String line)
1 /** 2 * 注釋開始標志 <br/> 3 * The token that indicates the start of an XML comment. 4 */ 5 private static final String START_COMMENT = "<!--"; 6 7 /** 8 * 注釋結束標志"-->" <br/> 9 * The token that indicates the end of an XML comment. 10 */ 11 private static final String END_COMMENT = "-->"; 12 13 private String consumeCommentTokens(String line) { 14 // 非注釋,即為有用信息 15 if (!line.contains(START_COMMENT) && !line.contains(END_COMMENT)) { 16 return line; 17 } 18 String currLine = line; 19 // 消耗注釋內容,使循環跳向下一行 20 while ((currLine = consume(currLine)) != null) { 21 // 當inComment標志位更新,并且返回信息不是以注釋開始標志開始就返回currLine 22 if (!this.inComment && !currLine.trim().startsWith(START_COMMENT)) { 23 return currLine; 24 } 25 } 26 // 如果沒有有用信息,則返回null 27 return null; 28 }

分析:

  • 如果當前行不是注釋,則直接返回。
  • consume函數的主要作用是消耗注釋內容,繼續循環下一行的內容。
1 private String consume(String line) { 2 // 如果inComment:true,則走endComent函數;false,則走startComment函數,初始時為false 3 // 因此這里會走startComment,返回注釋位置的index[注釋位置+1的index] 4 int index = (this.inComment ? endComment(line) : startComment(line)); 5 // 如果index=-1,則表示沒有注釋信息,否則返回注釋信息 6 return (index == -1 ? null : line.substring(index)); 7 } 8 9 private int startComment(String line) { 10 // 返回注釋開始標志的位置信息 11 return commentToken(line, START_COMMENT, true); 12 } 13 14 private int endComment(String line) { 15 return commentToken(line, END_COMMENT, false); 16 } 17 18 private int commentToken(String line, String token, boolean inCommentIfPresent) { 19 // 查找注釋標志的開始位置[<!--或-->] 20 int index = line.indexOf(token); 21 // index>-1表示存在注釋開始標志,并將inComment更新為inCommentIfPresent 22 // [默認在startComment為true,endComment為false] 23 if (index > -1) { 24 this.inComment = inCommentIfPresent; 25 } 26 // 如果index=-1,則返回注釋標志的后一個位置信息index+token.length() 27 return (index == -1 ? index : index + token.length()); 28 }

分析:

  • consume函數意在消費注釋信息,繼續循環下一行的內容。
  • inComment用來標記當前內容是否為注釋,初始時為false,所以剛開始碰到一個注釋語句,會執行startComment(line),將inComment置為true,然后返回"<!--"后面的內容,此時inComment為true,則會繼續循環,此時會執行endComment(line),將inComment置為false,然后會返回"",在detectValidationMode函數中由于content="",此時會繼續循環,從而跳過注釋內容。

消費注釋信息這里稍微有點繞,通過下面流程圖可更好的理解:

文件驗證模式代碼分析完成,這里回到DefaultDocumentLoader#loadDocument:

1 public Document loadDocument(InputSource inputSource, 2 EntityResolver entityResolver, 3 ErrorHandler errorHandler, 4 int validationMode, 5 boolean namespaceAware) throws Exception { 6 // 創建DocumentBuilderFactory 7 DocumentBuilderFactory factory = createDocumentBuilderFactory(validationMode, namespaceAware); 8 if (logger.isTraceEnabled()) { 9 logger.trace("Using JAXP provider [" + factory.getClass().getName() + "]"); 10 } 11 // 創建DocumentBuilder對象 12 DocumentBuilder builder = createDocumentBuilder(factory, entityResolver, errorHandler); 13 // 通過DocumentBuilder解析InputSource,返回Document對象 14 // 解析xml文件的具體過程都是通過jdk內置的類進行解析的--DOMParser為其入口 15 return builder.parse(inputSource); 16 }

分析:

  • 首先根據驗證模式和是否支持命名空間創建DocumentBuilderFactory。
  • 然后創建DocumentBuilder對象。
  • 最后進行XML文件的解析,具體解析過程是利用jdk內置的DOMParser解析器進行解析。

DefaultDocumentLoader#createDocumentBuilderFactory:

1 protected DocumentBuilderFactory createDocumentBuilderFactory(int validationMode, boolean namespaceAware) 2 throws ParserConfigurationException { 3 // 創建DocumentBuilderFactory實例 4 DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance(); 5 // 設置是否支持命名空間 6 factory.setNamespaceAware(namespaceAware); 7 // 是否有校驗模式 8 if (validationMode != XmlValidationModeDetector.VALIDATION_NONE) { 9 // 開啟校驗模式 10 factory.setValidating(true); 11 // XSD模式下設置factory的屬性 12 if (validationMode == XmlValidationModeDetector.VALIDATION_XSD) { 13 // Enforce namespace aware for XSD... 14 // 如果為XSD模式,強制開啟命名空間支持 15 factory.setNamespaceAware(true); 16 try { 17 // 設置SCHEMA_LANGUAGE_ATTRIBUTE屬性為XSD 18 factory.setAttribute(SCHEMA_LANGUAGE_ATTRIBUTE, XSD_SCHEMA_LANGUAGE); 19 } catch (IllegalArgumentException ex) { 20 ParserConfigurationException pcex = new ParserConfigurationException( 21 "Unable to validate using XSD: Your JAXP provider [" + factory + 22 "] does not support XML Schema. Are you running on Java 1.4 with Apache Crimson? " + 23 "Upgrade to Apache Xerces (or Java 1.5) for full XSD support."); 24 pcex.initCause(ex); 25 throw pcex; 26 } 27 } 28 } 29 30 return factory; 31 }

分析:這里邏輯就非常簡單了,主要創建DocumentBuilderFactory對象,然后設置校驗模式等相關屬性。

DefaultDocumentLoader#createDocumentBuilder:

1 protected DocumentBuilder createDocumentBuilder(DocumentBuilderFactory factory, 2 @Nullable EntityResolver entityResolver, @Nullable ErrorHandler errorHandler) 3 throws ParserConfigurationException { 4 // 創建DocumentBuilder對象 5 DocumentBuilder docBuilder = factory.newDocumentBuilder(); 6 // 設置實體解析器 7 if (entityResolver != null) { 8 docBuilder.setEntityResolver(entityResolver); 9 } 10 // 設置錯誤處理器 11 if (errorHandler != null) { 12 docBuilder.setErrorHandler(errorHandler); 13 } 14 return docBuilder; 15 }

分析:根據DocumentBuilderFactory工廠創建DocumentBuilder對象,并設置實體解析器與錯誤處理器。

XML文件的具體解析利用了jdk內置的DOMParser類進行,這里就不在深入了。

到這里就得到了XML配置文件的Document實例,介于篇幅原因,注冊bean的過程將后面進行分析。

總結

這里總結下本文的重點:

  • Resource體系與ResourceLoader體系,加載資源這里比較重要,因為有了資源才能進行后面的BeanDefinition加載。
  • 檢測配置文件是如何確定文件的驗證模式,確定驗證模式這里做的比較巧妙,著重如何消費注釋信息繼續下一次循環。

by Shawn Chen,2018.12.5日,下午。

轉載于:https://www.cnblogs.com/developer_chan/p/10013846.html

總結

以上是生活随笔為你收集整理的【spring源码分析】IOC容器初始化(二)的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

91中文字幕 | 国产精品专区在线 | 免费观看成人网 | 亚洲1级片| 亚洲综合欧美激情 | 在线国产视频一区 | 国产精品久久久久久久妇 | 国产一区免费视频 | 久久久久亚洲精品中文字幕 | 黄色av一区二区三区 | 成年人免费在线 | 亚洲综合射 | 日韩精品一区二区三区高清免费 | 狠狠操狠狠 | 久草视频免费观 | 国产精品久久久久四虎 | 97精产国品一二三产区在线 | 四虎最新入口 | 97狠狠操| 亚洲黄色影院 | 亚洲六月丁香色婷婷综合久久 | 国产精品字幕 | 国产在线精品区 | 免费aa大片 | 久久艹国产| 色婷婷97| 久久6精品 | 天天干天天干天天干 | 亚洲精品乱码久久久久v最新版 | 最近日本韩国中文字幕 | 午夜精品视频一区 | 成人av片免费观看app下载 | 美女网站免费福利视频 | 最新av网址在线观看 | 在线一区观看 | 日批视频在线播放 | 一级黄色在线免费观看 | 国产一区电影在线观看 | 久操伊人| 亚洲精品乱码 | 久久久亚洲国产精品麻豆综合天堂 | 99热手机在线观看 | 超碰97免费在线 | 亚洲精品久久久久久久不卡四虎 | 精品久久久久亚洲 | 中文字幕高清视频 | 久草视频在线播放 | 狠狠操天天干 | 免费av电影网站 | 在线观看视频亚洲 | 香蕉在线观看 | 激情网站免费观看 | www.色午夜| 欧美一级特黄aaaaaa大片在线观看 | av中文在线 | 欧美黑人xxxx猛性大交 | 久久66热这里只有精品 | 日本不卡视频 | 国产精品毛片完整版 | 久久99国产精品自在自在app | 中文字幕在线看人 | 亚洲欧洲视频 | 午夜久久视频 | 亚洲伦理一区 | 激情视频免费观看 | 五月天最新网址 | 91综合在线| 18av在线视频 | 日韩毛片在线播放 | 欧美久久久久久久久久久 | 99精品国产亚洲 | 天天躁日日躁狠狠躁av中文 | 最近日本韩国中文字幕 | 黄色在线小网站 | 91高清免费 | 国产裸体无遮挡 | 视频一区二区免费 | 色综合色综合久久综合频道88 | 激情久久伊人 | 亚洲欧美在线综合 | 国产黄色精品视频 | 91在线视频在线 | 国产九色91 | 国产成人一区二区啪在线观看 | 免费的黄色av | 精品一区 在线 | 欧美激情精品 | 又粗又长又大又爽又黄少妇毛片 | 亚洲va欧美 | 黄在线免费看 | 成人免费观看av | 免费看特级毛片 | 99久久精品国产毛片 | 在线视频app | 狠狠色伊人亚洲综合网站野外 | 九九99| 黄色av一区二区 | 丁香久久 | 91成人午夜 | 久久精品一 | 国产成人一区二区三区在线观看 | 午夜精品区 | 欧美日韩精品在线播放 | 国产精品久久久久高潮 | 91视频电影 | 2021av在线 | 日本三级不卡 | 97久久久免费福利网址 | 久久国产手机看片 | 天天插伊人 | 免费黄色a网站 | 四虎小视频 | 成人午夜黄色 | 久草观看 | 欧美一区二区三区免费观看 | 免费又黄又爽视频 | 亚洲精品国偷拍自产在线观看 | 亚洲波多野结衣 | 激情电影在线观看 | 精品国产精品久久 | 在线免费黄色毛片 | www.99久久.com | 国产精品一区二区av影院萌芽 | 日韩在线字幕 | 日韩欧美在线免费观看 | 五月综合久久 | 国产美女精彩久久 | 在线观看色网站 | 久久久久99精品成人片三人毛片 | 精品亚洲免费视频 | 一本一道久久a久久精品 | 欧美在线资源 | 精品久久网站 | 亚洲国产剧情av | 国产精品精品国产婷婷这里av | 国色天香永久免费 | 日韩毛片在线一区二区毛片 | 天天爽夜夜爽精品视频婷婷 | 人人玩人人添人人澡97 | 天天操天天操 | 99热在线国产 | 亚洲精品视频在线 | 亚洲精品大片www | 国产成人一区二区三区免费看 | 五月开心六月婷婷 | 天天色成人网 | 久久免费大片 | 国内成人综合 | 有码视频在线观看 | 国产精品久久电影网 | 久久午夜免费观看 | 美女久久一区 | 成人h在线 | 国产中文欧美日韩在线 | 69精品在线| 国产a视频免费观看 | 国产精品精品久久久久久 | 久草视频精品 | av免费片 | 色综合天天天天做夜夜夜夜做 | 国产一区在线播放 | 久久免费视频精品 | 中文字幕在线人 | 九九九热 | 在线激情影院一区 | 国产成在线观看免费视频 | 91精品无人成人www | 91精品啪在线观看国产 | 色资源在线观看 | 综合久久久久久 | 亚洲涩综合| 亚洲欧美视频在线观看 | 亚洲免费av网站 | 中文字幕在线播放第一页 | 久草在线99 | 99r国产精品 | 成人免费精品 | 黄色一二级片 | 日韩一级精品 | 五月视频 | 久久国产精品视频观看 | 日韩,精品电影 | 91最新在线 | 在线观看黄色国产 | 日韩午夜高清 | 成人免费网站视频 | 欧美日一级片 | 久久欧美精品 | 黄色免费观看视频 | 欧美一级视频免费 | 一级黄色av | 黄色大片日本免费大片 | 中文字幕一区三区 | 欧美久久久久久久久中文字幕 | 黄色片网站av | 7777xxxx | 日批在线看 | 一区二区三区在线视频观看58 | 黄色一级大片在线免费看国产一 | 丁香视频免费观看 | 免费视频97| 美女免费黄视频网站 | www91在线 | 亚洲va欧美 | 欧美色图亚洲图片 | 91久草视频| 一级黄网| 久久理论电影网 | 亚洲精品视频网址 | 久久久久久久电影 | 00av视频| 91在线免费播放 | 精品久久久久久久久久久院品网 | 国产黄色视| 日本一区二区三区视频在线播放 | 欧美久久久 | 日韩中文幕 | 波多野结衣精品视频 | 国产中的精品av小宝探花 | 五月婷婷精品 | 2021国产精品 | 国产精品黑丝在线观看 | 久久婷婷久久 | 免费视频在线观看网站 | 中文字幕不卡在线88 | 2023av在线| 亚洲性xxxx| 精品国自产在线观看 | 免费在线黄色av | 国产精品成久久久久三级 | 91探花国产综合在线精品 | 嫩草av影院 | 国产专区在线 | 亚洲一片黄 | 一区精品在线 | 欧美日韩一区二区三区在线观看视频 | 久久久国产毛片 | 天天射天天爱天天干 | 在线国产视频 | 嫩草av影院 | 国产精品成人av电影 | 麻豆视频免费观看 | 91最新视频在线观看 | 久艹视频在线观看 | 日韩成人看片 | 久久久久网址 | 日日爽天天操 | 日韩一级黄色av | 国产精品自在线 | 日日干精品 | 亚洲三级毛片 | 福利一区二区三区四区 | 天堂久久电影网 | 国产高清在线视频 | 天天综合网天天 | 深夜免费小视频 | 99人久久精品视频最新地址 | 国产自制av | 成人免费一区二区三区在线观看 | 国产高清在线不卡 | 色婷婷激情综合 | 亚洲一级片在线看 | 天天色播 | 日韩精品免费一线在线观看 | 国产涩图| 久久99亚洲精品久久 | 久久久精品视频网站 | 中国一级片在线观看 | 日韩免费播放 | 超碰999| av片在线观看免费 | 亚洲精品在线观 | 特级毛片在线 | 色丁香色婷婷 | 999热视频| 亚洲第五色综合网 | 久久午夜精品视频 | 久久精品精品 | 亚洲精品国精品久久99热 | 久久久久久久99精品免费观看 | av黄在线播放 | 免费福利视频网站 | 精品国产乱码久久久久久天美 | 四虎影视成人永久免费观看视频 | 韩国精品福利一区二区三区 | 黄色av在 | 一区二区伦理 | 一区二区不卡高清 | 久久综合色8888 | 精品国产91亚洲一区二区三区www | 久久精品9 | 欧美成年人在线观看 | 精品国产_亚洲人成在线 | 在线精品观看 | 精品亚洲成a人在线观看 | 玖草影院 | 色综合久久久久 | 国产一区在线视频播放 | 亚洲欧美日韩精品久久久 | 伊人电影在线观看 | 色综合天天做天天爱 | 国产精品美女久久久 | 久久大片网站 | av黄色大片 | 精品亚洲午夜久久久久91 | 中文在线字幕免费观看 | 久久精品综合一区 | 99免费在线视频 | 99久久久| 日本 在线 视频 中文 有码 | 精品一区精品二区 | 国产一级大片免费看 | 91精品在线免费 | 成人一级电影在线观看 | 亚洲欧美日韩中文在线 | 一区二区精品在线视频 | 色噜噜在线观看 | 综合久久久久久久 | 久久国产电影 | 91精品视频一区 | 国产午夜精品一区二区三区嫩草 | 日日干天天操 | 国产精品久久久久一区 | 日本一区二区三区免费看 | 欧美福利网址 | 黄网站www | 91视频国产高清 | 亚洲成人av一区二区 | 久久天天躁夜夜躁狠狠85麻豆 | 天天干人人干 | 九九视频这里只有精品 | av视屏在线播放 | 欧美视频18| 国产日本在线播放 | 97碰在线| 在线成人免费 | 伊人五月天.com | 日本丰满少妇免费一区 | 麻豆视频免费在线播放 | 亚洲精品国内 | 精品国产一区二区三区四 | 日本精品视频在线观看 | 激情喷水 | 欧美色图一区 | 视频在线观看99 | 伊人网综合在线观看 | 欧美 亚洲 另类 激情 另类 | 日韩av视屏在线观看 | 欧美粗又大 | 免费福利片 | 激情综合婷婷 | 亚洲自拍自偷 | 在线观看网站你懂的 | 91精品国产欧美一区二区成人 | 精品国产精品久久一区免费式 | 久久精品999 | 久久久久国产视频 | 久久99热这里只有精品国产 | 性色av一区二区三区在线观看 | 在线国产不卡 | 亚洲一级理论片 | 久久99国产综合精品免费 | 亚洲伊人av | 91精品日韩 | 久久伊人国产精品 | 国产欧美中文字幕 | 不卡的av电影 | 中文字幕在线观看资源 | 国产精品国产精品 | 久久午夜影院 | 国产日韩欧美视频在线观看 | 蜜臀av夜夜澡人人爽人人 | 91免费在线播放 | 91看片网址 | 精品国产激情 | 麻豆网站免费观看 | 国内一级片在线观看 | 国产日韩一区在线 | 欧美日韩免费网站 | 日韩久久视频 | 91麻豆精品国产91久久久无限制版 | 天天摸天天操天天爽 | 亚洲精品久久久久中文字幕二区 | 精品免费一区二区三区 | 久久精品99视频 | 中文字幕精品视频 | 人人插人人插 | av福利电影 | 国产麻豆精品95视频 | 免费a网址| h动漫中文字幕 | 最新色视频 | 国产一级一片免费播放放 | 成 人 黄 色 片 在线播放 | 亚洲一区久久 | av噜噜噜在线播放 | 国产精品9999久久久久仙踪林 | av一二三区 | 日韩成人精品一区二区 | 色wwwww| 久久色网站| 精品国产精品一区二区夜夜嗨 | av三区在线 | 99免费在线观看视频 | 国产96在线| 麻花豆传媒mv在线观看 | 四虎影视8848aamm | 狠狠干夜夜爽 | 夜夜操网| 久久中文字幕视频 | 日韩中文字幕免费 | 日韩精品一区在线观看 | 国产福利在线免费 | 免费在线激情电影 | 中文字幕亚洲欧美日韩2019 | 97超碰站| 二区三区在线 | 日韩欧美视频免费在线观看 | 国产成人一区二区三区在线观看 | 国产毛片久久久 | 玖玖精品在线 | 国产你懂的在线 | 久久久av免费 | 色婷婷激婷婷情综天天 | 狠狠色丁香九九婷婷综合五月 | 色 免费观看 | 一区二区三区精品在线视频 | 国产一区二区视频在线播放 | 精品国产精品一区二区夜夜嗨 | 亚洲一级国产 | 成 人 黄 色 片 在线播放 | 久久午夜视频 | 在线观看91视频 | 99九九热只有国产精品 | 97精品在线 | 中文字幕在线观看国产 | 色在线免费 | 日韩网站免费观看 | 在线观看国产 | 国产高清免费视频 | 国产一级精品在线观看 | av短片在线 | 国产精品国产三级在线专区 | 一区二区国产精品 | 久久精国产 | 日日夜夜综合网 | 国产精品女同一区二区三区久久夜 | 久久看毛片 | 免费在线观看成人av | 综合久久婷婷 | 精品视频在线免费 | 色小说av| 最新日韩视频 | 91av在| 久久久久久久久久久网站 | 国产在线资源 | 国产精品成久久久久 | 成 人 黄 色 免费播放 | 国产午夜三级一区二区三 | 日韩有码第一页 | 久热爱 | 日韩色一区二区三区 | 国产精品视频地址 | 日本亚洲国产 | www.天天色 | 91成版人在线观看入口 | 婷婷综合五月天 | а中文在线天堂 | 国产剧情在线一区 | 91网在线观看 | www.xxxx欧美 | 中文字幕 在线看 | 国产91电影在线观看 | 欧美乱大交 | 国产又粗又猛又爽又黄的视频先 | 97视频久久久 | 福利精品在线 | 91精品国产99久久久久久久 | 国产成人无码AⅤ片在线观 日韩av不卡在线 | 久草免费新视频 | av大片免费在线观看 | 丁香六月综合网 | 国产视频一级 | 国产网站av| 黄色软件在线看 | www天天干 | 久久久久亚洲精品 | 中文字幕二区三区 | 99久久精品午夜一区二区小说 | 黄在线| 婷婷丁香社区 | 国产精品爽爽久久久久久蜜臀 | 午夜国产一区二区三区四区 | 亚洲日韩欧美一区二区在线 | 在线国产99 | 麻豆91精品 | 园产精品久久久久久久7电影 | 亚洲日本激情 | 久久久久综合精品福利啪啪 | 免费在线观看国产精品 | 9999毛片| 久久久久久久国产精品 | 久久人91精品久久久久久不卡 | 91精品一区二区三区久久久久久 | 在线观看日韩中文字幕 | 亚洲成人午夜在线 | 天天色综合三 | 蜜臀av免费一区二区三区 | 亚洲在线精品视频 | 九色精品免费永久在线 | 国产成人综 | 9999毛片 | 日韩免费在线看 | av三级av | 97超碰伊人| 免费在线日韩 | 婷婷久久久 | 婷婷激情小说网 | 欧美极品裸体 | 精品一区二区电影 | 插插插色综合 | 五月天堂网 | 中文字幕一区二区在线播放 | 日韩欧美视频免费在线观看 | 中文字幕第一页av | 亚洲精品乱码久久久久久写真 | 久久99国产精品免费 | 久久久久久久久久久免费视频 | 超碰人人在线观看 | 最新日韩在线观看视频 | 国产精品欧美 | 97超碰人人模人人人爽人人爱 | 亚洲 欧美变态 另类 综合 | 精品国产精品久久 | 中文在线字幕免 | 国内精品久久久久久久久久久久 | 99热在线观看免费 | 国产精品久久伊人 | 国产在线观看国语版免费 | 国产香蕉在线 | 亚洲性xxxx| 久久精品视频日本 | 91男人影院 | 亚洲理论在线观看电影 | 91亚瑟视频 | 激情开心 | 九九在线国产视频 | 99精品国产aⅴ | 国产精品国产亚洲精品看不卡 | 国产日韩在线看 | 日韩欧美国产激情在线播放 | 99免费在线 | 日韩精品视频免费在线观看 | 天天干天天玩天天操 | 午夜精品一二区 | 亚洲激情在线 | 欧美天堂视频在线 | 亚洲精品裸体 | 欧美va天堂va视频va在线 | 亚洲国产免费网站 | 人人草人人做 | 国产亚洲成av片在线观看 | 日韩黄色在线观看 | av片在线观看免费 | 国产精品一级视频 | 日韩成人看片 | 国产在线观看你懂的 | 在线欧美小视频 | 亚洲天堂网在线观看视频 | 狠狠干成人综合网 | 四虎国产精品成人免费影视 | 六月激情网| 天天操 夜夜操 | 91麻豆精品91久久久久同性 | 国产成人精品一区二区在线 | 久久综合之合合综合久久 | 嫩草伊人久久精品少妇av | 天天摸日日摸人人看 | 国产综合91 | 五月婷婷欧美视频 | 三级黄色在线 | 成人动漫精品一区二区 | 日韩av片无码一区二区不卡电影 | 97精品国产91久久久久久 | 亚洲aⅴ久久精品 | 亚洲精品国精品久久99热 | 久久情网 | 天天爽夜夜爽人人爽一区二区 | 福利视频第一页 | 青春草免费视频 | 成人午夜精品福利免费 | www.xxxx欧美 | 亚洲精品在| 日本精品久久久久影院 | 激情综合国产 | 日日夜夜操操 | 免费色网| 日韩电影一区二区在线 | 天堂va欧美va亚洲va老司机 | 国产亚洲欧美一区 | 久久毛片网站 | 免费a级毛片在线看 | 国产精品乱码在线 | 免费观看一级 | 日韩av成人免费看 | 国产一级片免费播放 | 天天操夜夜操天天射 | 九九九热视频 | 在线观看岛国av | 久久99久久99精品中文字幕 | 中文字幕日本在线观看 | 99re8这里有精品热视频免费 | 国产在线更新 | 91丨porny丨九色 | 日本久久久影视 | 亚洲国产精久久久久久久 | 亚洲国产美女精品久久久久∴ | 久久99精品国产91久久来源 | 四虎在线观看视频 | 91精品国产欧美一区二区 | 97超碰精品 | 国产精品自产拍在线观看网站 | 中文字幕一区二区三区在线播放 | 欧美日韩国产一区二区三区在线观看 | 国产色综合 | 超碰在97| 超碰97成人 | 精选久久 | 在线观看视频97 | 国产精品久久久一区二区三区网站 | 亚洲国产av精品毛片鲁大师 | 日韩在线观看一区二区三区 | 99热精品国产一区二区在线观看 | 97超视频 | caobi视频| 久久视频精品在线观看 | 国产91免费观看 | 五月婷婷久草 | 九九九九九九精品任你躁 | 91亚洲成人| 亚洲精品免费在线 | www.天天射.com| 久久人人插 | 国产999精品久久久久久 | 国产午夜剧场 | 日本h视频在线观看 | 国产精品久久久久久超碰 | 日日夜夜亚洲 | 黄色小视频在线观看免费 | 国内精品久久久久久久久久清纯 | 日韩在线观看中文 | 欧美一区二区在线免费看 | 碰天天操天天 | 在线播放亚洲 | 免费看日韩 | 国产在线欧美在线 | 日韩二区精品 | 久久久国产电影 | 国产一级一片免费播放放a 一区二区三区国产欧美 | 天堂av在线网 | 福利片视频区 | 日本中文乱码卡一卡二新区 | 6080yy精品一区二区三区 | 国产原创在线 | 91chinese在线| 日韩精品一区二区三区第95 | 97国产大学生情侣白嫩酒店 | 性色av免费在线观看 | 久久一级电影 | 免费麻豆视频 | 国产精品一区二 | 久久久一本精品99久久精品66 | 成片免费观看视频大全 | 91精品办公室少妇高潮对白 | 亚洲精品成人网 | 亚洲精品xxx | 91麻豆精品国产自产在线 | 日日弄天天弄美女bbbb | 视频在线一区 | 欧美午夜精品久久久久 | 国产精品久久久久高潮 | 欧美精品一区在线 | 成人免费共享视频 | 国产91精品久久久久久 | 婷婷精品国产欧美精品亚洲人人爽 | 亚洲成人免费在线 | 国产福利91精品一区 | 久久精品一二三区 | 激情五月六月婷婷 | 亚洲精品乱码白浆高清久久久久久 | 午夜国产福利在线 | 国产精品久久久久久久久婷婷 | 午夜影视一区 | 96精品视频 | 最近中文字幕免费av | 色综合久久久网 | 国产精品久久二区 | 国产亚洲视频在线 | 亚洲美女精品视频 | 色国产精品 | 久久99偷拍视频 | 色婷婷在线播放 | 久久久久久免费 | 韩国三级av在线 | 国产999视频在线观看 | 一级黄色片在线观看 | 97麻豆视频 | 国产在线91在线电影 | 高清不卡一区二区三区 | 国产精品久久久影视 | 国产精品一区二区三区在线看 | 久久国产精品99久久久久 | 97超碰超碰久久福利超碰 | av网站播放| 亚洲精品毛片一级91精品 | 五月综合网站 | 天天干天天操天天 | 欧美男同网站 | 日韩av午夜| 国产精品久久网 | 欧美极品在线播放 | 视频在线一区 | 国产在线播放一区二区三区 | 超碰在线网 | 日韩午夜剧场 | 久久久国产一区二区三区 | 欧美一区二视频在线免费观看 | 蜜臀91丨九色丨蝌蚪老版 | 免费在线观看a v | 欧美一级在线观看视频 | 四虎免费在线观看 | 久草在线综合网 | 国产精品视频久久久 | 久久久91精品国产一区二区三区 | 天天天天天天干 | 久久久久久久影院 | 国产91在线免费视频 | 国产一级二级在线 | 亚洲国产精品推荐 | 久久免费国产精品 | 亚洲精品视频在线免费播放 | 美女在线免费视频 | 黄色在线网站噜噜噜 | 最近更新好看的中文字幕 | 亚洲天堂精品视频 | 久草在线精品观看 | 麻豆精品在线视频 | 久久国产精品视频观看 | 91av短视频| 西西www444 | 国产在线中文字幕 | 亚洲日本韩国一区二区 | 午夜123| 国产成人在线观看免费 | 人人干,人人爽 | 国产精品久久久久久久久久久久 | 九九热在线精品视频 | 粉嫩av一区二区三区四区 | 狠狠干夜夜爱 | 久久在线视频在线 | 四虎精品成人免费网站 | 欧美巨乳网 | 久久久久国产精品一区二区 | 亚洲97在线| 免费色黄 | 国产日本在线 | 少妇搡bbbb搡bbb搡aa | 深爱五月网 | 男女啪啪视屏 | 国产96在线视频 | 永久中文字幕 | 日韩免费视频一区二区 | 欧美日韩二三区 | 91视频链接 | 美女性爽视频国产免费app | 人人爱天天操 | 欧美一级黄大片 | 日本免费久久高清视频 | 午夜久久 | 在线精品在线 | 最新国产视频 | 日韩av免费观看网站 | 国产一级久久 | 爱爱av在线 | av超碰在线 | 久久久久久高潮国产精品视 | 天天操夜夜操夜夜操 | 久久久久久蜜av免费网站 | av午夜电影 | 有没有在线观看av | 精品久久久久久亚洲综合网 | 久久久91精品国产 | 亚洲最新合集 | av电影免费在线播放 | 久久久久久网址 | 日本韩国精品在线 | 人九九精品| 成人小视频免费在线观看 | 日韩一区二区在线免费观看 | 国产精品久久久久久久久久免费看 | 91精品啪啪 | 亚洲高清在线精品 | 免费高清在线观看电视网站 | 中文字幕91视频 | 狠狠网站| 国产专区精品视频 | 久久精品日韩 | 亚洲一区二区三区毛片 | 亚洲欧美日韩国产一区二区三区 | 国产精品99久久久久 | 亚洲每日更新 | 国产精品自产拍在线观看桃花 | 国产成人三级三级三级97 | 激情影音 | 久久久91精品国产一区二区精品 | 国产精品久久一区二区无卡 | 国产美女免费观看 | 国产成人av在线 | 国产精品国产三级国产aⅴ9色 | 91在线视频免费播放 | 黄色福利| 99在线视频观看 | 久草在线视频在线观看 | 伊人丁香| 日日夜夜噜 | 国产日韩中文在线 | 日韩欧美高清视频在线观看 | 国产精品久久久久国产a级 激情综合中文娱乐网 | 亚洲婷婷伊人 | 特及黄色片 | 国产成人在线观看免费 | 国产精品嫩草在线 | 久久五月婷婷综合 | 绯色av一区 | 99视频免费观看 | 97自拍超碰 | 日韩精品免费在线观看视频 | 国产在线无 | 久久免费视频6 | 青青河边草观看完整版高清 | 日韩激情在线 | 久久久久久久久久久久久国产精品 | 96久久精品| 国偷自产中文字幕亚洲手机在线 | 久久国产一区二区三区 | 欧美日韩视频免费 | 久久 一区 | 久久成人久久 | 久久综合久色欧美综合狠狠 | 日韩一区二区三区高清免费看看 | 色综合久久天天 | 国产精成人品免费观看 | 人人射网站 | 国产男女免费完整视频 | 日韩乱码在线 | 久久久精品国产一区二区电影四季 | 在线综合色| 婷婷四房综合激情五月 | 婷婷丁香七月 | 亚洲成年人在线播放 | 国产精品福利在线观看 | 久久一二三四 | 91在线视频导航 | 亚洲专区欧美 | 成片免费 | 久久小视频| 天天狠狠操 | 涩涩网站在线观看 | 中文字幕视频一区 | 天天色影院 | 亚洲视频99| 五月激情在线 | 天天射综合 | 天海冀一区二区三区 | 久久久久久久久久久高潮一区二区 | 婷婷中文字幕综合 | 激情五月五月婷婷 | 天天综合色天天综合 | 国产精品久久一区二区三区, | 96亚洲精品久久久蜜桃 | 亚洲国产福利视频 | 九九热精品在线 | 久久成人免费视频 | 大胆欧美gogo免费视频一二区 | 色5月婷婷| 国产精品色在线 | 久久公开免费视频 | 手机av在线网站 | 免费在线国产 | 国产亚洲精品综合一区91 | 91精选在线 | 亚洲2019精品 | 欧美在线观看视频免费 | 又色又爽又黄高潮的免费视频 | 人人干狠狠操 | 欧美成人亚洲 | 精品国产一区二区三区久久久蜜臀 | 91人人澡人人爽人人精品 | 99热最新地址| 色婷婷六月| 91麻豆国产福利在线观看 | 久久手机免费视频 | 国产一级免费播放 | 日韩视频免费 | 黄在线免费观看 | 久久黄色免费视频 | 亚洲黄色在线观看 | www.天天操 | www.天天色.com | 中文字幕精品一区二区精品 | 色多多视频在线观看 | 开心激情久久 | 亚洲欧洲精品久久 | 欧美国产视频在线 | 日韩av在线免费播放 | 欧美一区二区视频97 | 国产 色| 久久成人麻豆午夜电影 | 日韩免费一区二区 | 日韩精品欧美精品 | 免费看一级特黄a大片 | a色网站 | 玖玖在线资源 | 国产在线资源 | 中文在线免费一区三区 | 日韩av在线看 | 三级av在线播放 | 精品国产诱惑 | 精品国产自在精品国产精野外直播 | 日韩精品一区二区三区水蜜桃 | 国产区欧美 | 久久国产色 | 丁香六月五月婷婷 | 91视视频在线直接观看在线看网页在线看 | 一本大道久久精品懂色aⅴ 五月婷社区 | 久草热视频 | 99久久日韩精品视频免费在线观看 | 九九九九九九精品任你躁 | 天天想夜夜操 | 日韩久久午夜一级啪啪 | 国产大尺度视频 | 6080yy精品一区二区三区 | 精品女同一区二区三区在线观看 | 久久久久久久国产精品视频 | 一区久久久 | 91精品婷婷国产综合久久蝌蚪 | 日日干狠狠操 | 成人a级网站 | 深夜免费福利网站 | 欧美日韩精品在线观看视频 | 操久| 久久五月婷婷丁香社区 | 美女视频永久黄网站免费观看国产 | 97国产一区二区 | 日韩高清不卡一区二区三区 | 国产破处精品 | 国产精品久久久久久久久久久久久久 | 久久99久国产精品黄毛片入口 | 中文字幕在线观看日本 | 免费视频久久久 | 亚洲精品人人 | 五月婷婷开心 | 成人动态视频 | 亚洲精品综合一二三区在线观看 | 一区二区理论片 | mm1313亚洲精品国产 | av不卡中文 | 日韩1级片 | 青青草在久久免费久久免费 | 欧美日韩在线网站 | 亚洲精品黄色 | 97涩涩视频 | 国产99久久九九精品免费 | 久久69av | 五月婷婷精品 | 天天做天天爱天天综合网 | 91久久一区二区 | 国产精品国产毛片 | 日韩精品视频第一页 | 99久久精品免费看 | 黄色影院在线免费观看 | 亚洲影院色 | 久草在线播放视频 | 麻豆国产在线播放 | 免费的国产精品 | 99国产成+人+综合+亚洲 欧美 | 黄色亚洲大片免费在线观看 | 国产视频一区二区在线播放 | 久草在线精品观看 | 91麻豆精品一区二区三区 | 天堂av免费观看 | 天天摸天天舔天天操 | 最新黄色av网址 | 深夜免费福利在线 | 又爽又黄又刺激的视频 | 久久的色| 亚洲妇女av | 黄色小说在线免费观看 |