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

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 >

component是什么接口_逐行解读Spring(二)什么,自定义标签没听说过?

發布時間:2024/10/12 90 豆豆
生活随笔 收集整理的這篇文章主要介紹了 component是什么接口_逐行解读Spring(二)什么,自定义标签没听说过? 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

一、自定義標簽是什么?

上一篇我們講了默認標簽-bean標簽的解析,今天我們講一下自定義標簽的解析。

1. 自定義標簽的定義

這個問題其實上一篇有講過,這邊再復述一遍,在spring的xml配置文件中,我們可以把所有的標簽分為兩類:自定義標簽和默認標簽,區別如下

復制代碼

需要注意的是,自定義標簽的概念,并不完全只指我們開發時自己定義的標簽,而是spring的開發者為之后拓展預留的拓展點,這個拓展點我們可以用,spring的開發人員在為spring添加新功能時,也可以使用。

2. 關于spring內置的自定義標簽context:component-scan

我們現在的開發中,更多的情況下,其實是使用@Configuration、@Component、@Service 等注解來進行bean的聲明而不是使用xml的bean 標簽了。

那么為什么一個類加上了這些注解之后,就能被spring管理了呢?

實際上這些拓展功能是spring通過自己預留的自定義標簽的拓展點進行拓展的,對于上述的功能,具體是使用的context:component-scan標簽。

我們今天就通過對自定義標簽context:component-scan的解析來跟蹤一下相應的源碼,理解spring自定義標簽解析的流程,同時也對context:component-scan實現的功能做一個講解,看一下@Component等標簽的實現原理。

二、源碼解析

1. 自定義標簽解析過程

由于上一篇對xml的源碼跟過了,這期我們之間定位到相應代碼org.springframework.beans.factory.xml.DefaultBeanDefinitionDocumentReader#parseBeanDefinitions

protected void parseBeanDefinitions(Element root, BeanDefinitionParserDelegate delegate) { // 判斷是否是默認的命名空間 if (delegate.isDefaultNamespace(root)) { NodeList nl = root.getChildNodes(); for (int i = 0; i < nl.getLength(); i++) { Node node = nl.item(i); if (node instanceof Element) { Element ele = (Element) node; if (delegate.isDefaultNamespace(ele)) { // 解析默認標簽 parseDefaultElement(ele, delegate); } else { // 可以看到代理主要進行自定義標簽的解析 delegate.parseCustomElement(ele); } } } } else { // 可以看到代理主要進行自定義標簽的解析 delegate.parseCustomElement(root); }}@Nullablepublic BeanDefinition parseCustomElement(Element ele, @Nullable BeanDefinition containingBd) { // 獲取標簽對于的namespaceUrl, 即配置文件頭部beans標簽里面那些xmlns:xxx=www.xxx.com String namespaceUri = getNamespaceURI(ele); if (namespaceUri == null) { return null; } // 獲取自定義標簽對應的NamespaceHandler,從這里我們可以看到,對于每一個namespaceUri應該都有唯一一個對應的NamespaceHandler NamespaceHandler handler = this.readerContext.getNamespaceHandlerResolver().resolve(namespaceUri); if (handler == null) { error("Unable to locate Spring NamespaceHandler for XML schema namespace [" + namespaceUri + "]", ele); return null; } // 把自定義標簽委托給對應的NamespaceHandler解析 return handler.parse(ele, new ParserContext(this.readerContext, this, containingBd));}復制代碼

我們先看一下NamespaceHandler這個自定義標簽的解析接口的結構:

public interface NamespaceHandler { // 初始化,我們可以合理猜測,這個方法將會在NamespaceHandler實例化之后,使用之前調用 void init(); // xml解析入口 @Nullable BeanDefinition parse(Element element, ParserContext parserContext); // 裝飾接口,其實用的比較少,上一篇有稍微帶到過一下,默認bean標簽解析完之后,可以有一個機會對解析出來的beanDefinition進行裝飾,實際開發中很少使用 // 有興趣的同學可以自行看下源碼,源碼在 org.springframework.beans.factory.xml.DefaultBeanDefinitionDocumentReader#processBeanDefinition @Nullable BeanDefinitionHolder decorate(Node source, BeanDefinitionHolder definition, ParserContext parserContext);}復制代碼

接下來當然是需要看一下獲取NamespaceHandler的流程:

public NamespaceHandler resolve(String namespaceUri) { // 獲取到了一個handlerMapping,具體邏輯我們之后再看 Map handlerMappings = getHandlerMappings(); // 通過namespaceUri獲取到一個對象 Object handlerOrClassName = handlerMappings.get(namespaceUri); if (handlerOrClassName == null) { return null; } // 如果handlerOrClassName是一個NamespaceHandler對象,則直接返回 - 拿到對應的handler了 else if (handlerOrClassName instanceof NamespaceHandler) { return (NamespaceHandler) handlerOrClassName; } else { // 如果handlerOrClassName不是NamespaceHandler對象,則是String對象 String className = (String) handlerOrClassName; // 通過String獲取到一個Class對象,那么這個String對象肯定是一個類的全限定名啦 Class> handlerClass = ClassUtils.forName(className, this.classLoader); // handlerClass必須繼承自NamespaceHandler,很好理解,畢竟是spring提供的拓展點,自然需要符合它定義的規則 if (!NamespaceHandler.class.isAssignableFrom(handlerClass)) { throw new FatalBeanException("..."); } // 直接通過反射構造一個實例,點進去看會發現是調用的無參構造器,我們就不看了 NamespaceHandler namespaceHandler = (NamespaceHandler) BeanUtils.instantiateClass(handlerClass); // !!! 調用了init()方法,和我們之前的推測一致 namespaceHandler.init(); // !!! 把handler對象塞回了handlerMappings,所以我們下次再通過namespaceUri獲取時,會直接拿到一個NamespaceHandler對象 // 也即每個namespaceUri對應的NamespaceHandler對象是單例的,而init()方法也只會調用一次 handlerMappings.put(namespaceUri, namespaceHandler); return namespaceHandler; // 去除掉了異常處理 }}復制代碼

由上述源碼其實我們已經得知了NamespaceHandler的一個初始化過程,但其實還有一個疑問,就是這個handlerMappings中最初的那些namespaceUri對應的handler的類名是哪來的呢?這個時候我們就需要去看一下getHandlerMappings()的過程啦

private Map getHandlerMappings() { Map handlerMappings = this.handlerMappings; if (handlerMappings == null) { synchronized (this) { handlerMappings = this.handlerMappings; // 雙重檢查加鎖,看來我們的handlerMappings之后加載一次 if (handlerMappings == null) { // 可以看到這邊是去加載了文件 // 文件加載的過程我們就不去跟了,跟主流程關系不大,我們主要看一下這個文件位置 // this.handlerMappingsLocation是哪里 Properties mappings = PropertiesLoaderUtils.loadAllProperties(this.handlerMappingsLocation, this.classLoader); handlerMappings = new ConcurrentHashMap<>(mappings.size()); // 然后把文件中的kev-value屬性都合并到了一個map里 CollectionUtils.mergePropertiesIntoMap(mappings, handlerMappings); this.handlerMappings = handlerMappings; // 干掉了異常處理代碼 } } } return handlerMappings;}// 字段的定義, 需要說一下當前類是DefaultNamespaceHandlerResolver,喜歡自己探索的同學可以直接空降/** Resource location to search for. */private final String handlerMappingsLocation;// 可以看到這個值是Resolver的構造器中設值的public DefaultNamespaceHandlerResolver(@Nullable ClassLoader classLoader, String handlerMappingsLocation) { this.classLoader = (classLoader != null ? classLoader : ClassUtils.getDefaultClassLoader()); this.handlerMappingsLocation = handlerMappingsLocation;}// 默認是取的DEFAULT_HANDLER_MAPPINGS_LOCATION這個常量public DefaultNamespaceHandlerResolver() { this(null, DEFAULT_HANDLER_MAPPINGS_LOCATION);}// 我們看一下這個常量的值public static final String DEFAULT_HANDLER_MAPPINGS_LOCATION = "META-INF/spring.handlers";復制代碼

如果對SPI比較熟悉的同學,應該已經知道這是個什么套路了,并且對META-INF這個目錄也比較熟悉,那么現在,我們看一下這個META-INF/spring.handlers文件中到底寫了一些什么東西,以context:component-scan標簽為例,我們知道這個標簽是spring-context包里面提供的,直接去找這個jar包的對應文件,看一下里面的內容:

## 我們可以很明顯的看到一個key=value結構http://www.springframework.org/schema/context=org.springframework.context.config.ContextNamespaceHandlerhttp://www.springframework.org/schema/jee=org.springframework.ejb.config.JeeNamespaceHandlerhttp://www.springframework.org/schema/lang=org.springframework.scripting.config.LangNamespaceHandlerhttp://www.springframework.org/schema/task=org.springframework.scheduling.config.TaskNamespaceHandlerhttp://www.springframework.org/schema/cache=org.springframework.cache.config.CacheNamespaceHandler復制代碼

我們在回憶一下自定義標簽的定義:

復制代碼

可以看到我們的META-INF/spring.handlers文件中key就是自定義標簽的namespaceUri,value則是對應的NamespaceHandler的全限定名。

那么簡單總結一下,我們的自定義標簽解析的流程就是:

  • 加載所有jar中META-INF/spring.handlers文件中的namespaceUri和NamespaceHandler的全限定名的映射關系到handlerMappings
  • 根據namespaceUri從handlerMappings獲取對象 如果從handlerMappings獲取到的對象為空,直接返回 如果獲取到的是NamespaceHandler對象,直接使用 如果獲取到的對象是string類型,則實例化這個string對應的全限定名的NamespaceHandler對象,并調用init()方法,然后將 namespaceUri-NamespaceHandler對象關系放回handlerMappings
  • 將自定義標簽委托給2獲取到的NamespaceHandler對象解析-調用parse方法(如果2未獲取到對應的NamespaceHandler對象,則此自定義標簽無法解析,直接跳過)
  • 2. context:component-scan標簽工作原理

    接下來我們來看一下 context:component-scan標簽的工作原理,從spring-context包的META-INF/spring.handlers文件我們可以找到該標簽對應的處理器:

    http://www.springframework.org/schema/context=org.springframework.context.config.ContextNamespaceHandler復制代碼

    直接找到這個類:

    public class ContextNamespaceHandler extends NamespaceHandlerSupport { @Override public void init() { // 刪掉了一些我們不關注的標簽的Parser的注入代碼... // 我們可以看到這里注冊了一個BeanDefinitionParser,而且這個注冊方法的第一個參數明顯是 // `context:component-scan` 標簽中刪掉前綴的部分,我們先記下來 registerBeanDefinitionParser("component-scan", new ComponentScanBeanDefinitionParser()); // 刪掉了一些我們不關注的標簽的Parser的注入代碼... }}復制代碼

    可以看到ContextNamespaceHandler繼承自NamespaceHandlerSupport,這是一個典型的模本方法設計模式。這里不做拓展,我們直接看一下NamespaceHandlerSupport:

    // 這里我們只保存了與解析器相關的代碼,并且調整了一下源碼順序// 裝飾相關的代碼我去除掉了,并不是NamespaceHandlerSupport中沒有,不過它的邏輯和解析基本是一致的// 如果同學們還記得哪里對beanDefinition進行了裝飾,并且感興趣的話,可以自行了解一下 (* ̄︶ ̄)public abstract class NamespaceHandlerSupport implements NamespaceHandler { // 保存標簽名-解析器對應關系的容器 private final Map parsers = new HashMap<>(); // 保存標簽名-裝飾器對應關系的容器 private final Map decorators = new HashMap<>(); // 保存屬性名-裝飾器對應關系的容器 private final Map attributeDecorators = new HashMap<>(); // 可以看到我們init()方法中的register其實就只是把對應elementName-Parser放入map而已 protected final void registerBeanDefinitionParser(String elementName, BeanDefinitionParser parser) { this.parsers.put(elementName, parser); } public BeanDefinition parse(Element element, ParserContext parserContext) { // 獲取 Parser BeanDefinitionParser parser = findParserForElement(element, parserContext); // 委托給 Parser解析 return (parser != null ? parser.parse(element, parserContext) : null); } private BeanDefinitionParser findParserForElement(Element element, ParserContext parserContext) { // 這里是獲取了去掉標簽中去掉前綴后的名稱 context:component-scan --> component-scan String localName = parserContext.getDelegate().getLocalName(element); // 從map中獲取到對應的Parser return this.parsers.get(localName); }}復制代碼

    到此為止其實還是蠻簡單的嘛,我們又把標簽委托給了對應的Parser來處理,那么我們現在來看一下component-scan對應的ComponentScanBeanDefinitionParser的邏輯,我們先看parse方法,也是我們的入口方法:

    public BeanDefinition parse(Element element, ParserContext parserContext) { // 獲取標簽上配置并處理的base-package屬性 String basePackage = element.getAttribute(BASE_PACKAGE_ATTRIBUTE); // 處理占位符 basePackage = parserContext.getReaderContext().getEnvironment().resolvePlaceholders(basePackage); // 最終獲取到的是一個數組 - 因為我們配置的時候是可以配置多個的 String[] basePackages = StringUtils.tokenizeToStringArray(basePackage, ConfigurableApplicationContext.CONFIG_LOCATION_DELIMITERS); // 獲取一個掃描器 - 這個東西很重要,我們以后還會看到 ClassPathBeanDefinitionScanner scanner = configureScanner(parserContext, element); // 嗯,掃描器進行掃描,看來就是這個方法會掃描那些注解了 Set beanDefinitions = scanner.doScan(basePackages); // 注冊一些組件 registerComponents(parserContext.getReaderContext(), beanDefinitions, element); return null;}復制代碼

    我們先看一下掃描器是怎么創建出來的:

    // 把一些異常處理都干掉了protected ClassPathBeanDefinitionScanner configureScanner(ParserContext parserContext, Element element) { // 解析一下是否用默認的過濾器 --> 這里解釋一下,其實這個過濾器就是指我們那些注解@Service等。 // 其實這里就是定義那些注解是我們掃描到了之后會把它納入IOC管理的,具體代碼之后解析的時候會看到 boolean useDefaultFilters = true; if (element.hasAttribute(USE_DEFAULT_FILTERS_ATTRIBUTE)) { useDefaultFilters = Boolean.parseBoolean(element.getAttribute(USE_DEFAULT_FILTERS_ATTRIBUTE)); } // 直接創建一個掃描器 ClassPathBeanDefinitionScanner scanner = createScanner(parserContext.getReaderContext(), useDefaultFilters); // 從parserContext獲取到的默認的beanDefinition的配置,即之后解析的beanDefinition的缺省配置 scanner.setBeanDefinitionDefaults(parserContext.getDelegate().getBeanDefinitionDefaults()); // 從parserContext獲取到的默認的自動裝配的模式,byType、byName那些 scanner.setAutowireCandidatePatterns(parserContext.getDelegate().getAutowireCandidatePatterns()); // 掃描的資源路徑,一般我們也不配置 if (element.hasAttribute(RESOURCE_PATTERN_ATTRIBUTE)) { scanner.setResourcePattern(element.getAttribute(RESOURCE_PATTERN_ATTRIBUTE)); } // 沒什么用的...一般也不會去自定義,即使用注解時,生成bean的name的策略也可以自定義 parseBeanNameGenerator(element, scanner); // 基本也不用,scope相關的,大概意思就是這個bean會存在于哪些scope,一般不用 parseScope(element, scanner); // 解析類型過濾器-這個算相對重要,其實就是我們可以自定義需要掃描哪些注解 parseTypeFilters(element, scanner, parserContext); return scanner;}復制代碼

    我們先看一下如果useDefaultFilters=true會注冊哪些過濾器,createScanner中其實就是直接調用了構造器,那我們直接看一下構造器邏輯:

    public ClassPathBeanDefinitionScanner(BeanDefinitionRegistry registry, boolean useDefaultFilters, Environment environment, @Nullable ResourceLoader resourceLoader) { Assert.notNull(registry, "BeanDefinitionRegistry must not be null"); this.registry = registry; if (useDefaultFilters) { // 注冊默認的過濾器 registerDefaultFilters(); } setEnvironment(environment); setResourceLoader(resourceLoader);}protected void registerDefaultFilters() { // includeFilters添加了一個AnnotationTypeFilter,過濾器構造器傳入了Component的Class對象 this.includeFilters.add(new AnnotationTypeFilter(Component.class)); // 省略了兩個JSR規范注解的注冊代碼,我們一般用不到,@javax.annotation.ManagedBean和@javax.inject.Named}復制代碼

    由于Filter的匹配過程不是主流程,不在這里多寫,但是我會寫一段源碼解析到這一節的末尾,感興趣的同學也可以看一下。

    我們解析來看一下類型過濾器標簽的解析:

    protected void parseTypeFilters(Element element, ClassPathBeanDefinitionScanner scanner, ParserContext parserContext) { // ... NodeList nodeList = element.getChildNodes(); for (int i = 0; i < nodeList.getLength(); i++) { Node node = nodeList.item(i); // 找到每一個子節點 if (node.getNodeType() == Node.ELEMENT_NODE) { String localName = parserContext.getDelegate().getLocalName(node); if (INCLUDE_FILTER_ELEMENT.equals(localName)) { // 如果是標簽則創建一個Filter并加入includeFilters TypeFilter typeFilter = createTypeFilter((Element) node, classLoader, parserContext); scanner.addIncludeFilter(typeFilter); } else if (EXCLUDE_FILTER_ELEMENT.equals(localName)) { // 如果是標簽則創建一個Filter并加入includeFilters TypeFilter typeFilter = createTypeFilter((Element) node, classLoader, parserContext); scanner.addExcludeFilter(typeFilter); } } }}復制代碼

    那么我們看一下createTypeFilter究竟做了一些什么:

    // 邏輯還是比較直觀的protected TypeFilter createTypeFilter(Element element, @Nullable ClassLoader classLoader, ParserContext parserContext) { String filterType = element.getAttribute(FILTER_TYPE_ATTRIBUTE); String expression = element.getAttribute(FILTER_EXPRESSION_ATTRIBUTE); expression = parserContext.getReaderContext().getEnvironment().resolvePlaceholders(expression); if ("annotation".equals(filterType)) { // 如果我們想掃描自定義的注解,那可以使用這個annotation類型,expression填注解全限定名就好了 return new AnnotationTypeFilter((Class) ClassUtils.forName(expression, classLoader)); } else if ("assignable".equals(filterType)) { // 掃描配置的類及其子類,expression填類的全限定名就好了,這個也偶爾用到,主要用來指定掃描一些二方庫的bean return new AssignableTypeFilter(ClassUtils.forName(expression, classLoader)); } else if ("aspectj".equals(filterType)) { // 掃描切面表達式所匹配的類 return new AspectJTypeFilter(expression, classLoader); } else if ("regex".equals(filterType)) { // 掃描正則表達式所匹配的類 return new RegexPatternTypeFilter(Pattern.compile(expression)); } else if ("custom".equals(filterType)) { // 自定義的過濾器,對應的類需要實現TypeFilter接口 Class> filterClass = ClassUtils.forName(expression, classLoader); if (!TypeFilter.class.isAssignableFrom(filterClass)) { throw new IllegalArgumentException( "Class is not assignable to [" + TypeFilter.class.getName() + "]: " + expression); } return (TypeFilter) BeanUtils.instantiateClass(filterClass); } else { throw new IllegalArgumentException("Unsupported filter type: " + filterType); }}復制代碼

    好了,context:component-scan標簽的屬性解析就告一段落了,我們主要記住base-package和Filter相關的就好了,其余的其實也用不太到,畢竟這個掃描的功能主要只需要確定需要掃描哪些包以及需要關注哪些類就好了。那么我們接下來再往回看一下,掃描器的掃描邏輯是怎么樣的,同學們可以空降ComponentScanBeanDefinitionParser#parse,然后我們來看一下獲取到scanner之后,scanner.doScan(basePackages)的邏輯:

    protected Set doScan(String... basePackages) { for (String basePackage : basePackages) { // 找到所以掃描到的beanDefinition Set candidates = findCandidateComponents(basePackage); for (BeanDefinition candidate : candidates) { // 獲取beanName,要知道,我們使用注解的時候,其實是沒有一個像xml標簽屬性那樣的東西來獲取name的 // 這里通過beanNameGenerator來獲取了beanName,默認就是通過注解內的對應屬性或者類名。感興趣的同學可以看下 AnnotationBeanNameGenerator String beanName = this.beanNameGenerator.generateBeanName(candidate, this.registry); // 刪掉了一下不重要的屬性的賦值 if (candidate instanceof AnnotatedBeanDefinition) { // 里是處理類上的一些公共注解的地方,比如@Primary,@Lazy等 AnnotationConfigUtils.processCommonDefinitionAnnotations((AnnotatedBeanDefinition) candidate); } // 這個判斷的大概意思就是,看一下我們掃描出來的beanDifinition是不是第一次注冊 // 如果不是第一次注冊就不會再注冊了,是通過beanName來從IOC容器中找有沒有一樣的 if (checkCandidate(beanName, candidate)) { // ... // 注冊bean,這個邏輯我們第一篇看過了,就不在看了,實際上就是把beanDefinition放入 // beanDefinitionMap和beanDefinitionNames這兩個容器里面 registerBeanDefinition(definitionHolder, this.registry); } } } return beanDefinitions;}復制代碼

    我們先看一下是怎么AnnotationConfigUtils.processCommonDefinitionAnnotations()中是怎么處理類上的注解的:

    static void processCommonDefinitionAnnotations(AnnotatedBeanDefinition abd, AnnotatedTypeMetadata metadata) { AnnotationAttributes lazy = attributesFor(metadata, Lazy.class); if (lazy != null) { abd.setLazyInit(lazy.getBoolean("value")); } else if (abd.getMetadata() != metadata) { lazy = attributesFor(abd.getMetadata(), Lazy.class); if (lazy != null) { abd.setLazyInit(lazy.getBoolean("value")); } } if (metadata.isAnnotated(Primary.class.getName())) { abd.setPrimary(true); } AnnotationAttributes dependsOn = attributesFor(metadata, DependsOn.class); if (dependsOn != null) { abd.setDependsOn(dependsOn.getStringArray("value")); } AnnotationAttributes role = attributesFor(metadata, Role.class); if (role != null) { abd.setRole(role.getNumber("value").intValue()); } AnnotationAttributes description = attributesFor(metadata, Description.class); if (description != null) { abd.setDescription(description.getString("value")); }}復制代碼

    大聰明們肯定已經發現了,其實就是看一下類上面有沒有對應的注解,然后把對應的屬性塞入beanDefinition對象嘛。這豈不是可以說是跟XML解析獲取beanDefinition時的流程一模一樣的?

    是的,其實不管是基于注解還是基于xml,都是把一些描述bean的信息,收集匯總到相應的beanDefinition中而已。而beanDefinition的屬性決定了這個bean會怎么實例化,需要注入哪些屬性等等等等。

    收集信息來注解beanDefinition的途徑可以有多種--甚至你自己寫一個解析json格式文件的組件也不是不行,但是結果都是殊途同歸的。

    從這也可以看出spring設計的強大,這種模塊化的設計思想和對單一職責原則(比較直觀的是各種委托模式,專業的事給專業的類做)和開閉原則(到我們現在講到的地方:自定義標簽的設計,在不觸動原有核心邏輯的情況下,我們可以很簡單的對spring做一些自定義的拓展)的實踐,我們日常開發中是否也可以借鑒借鑒呢?

    好了,感慨完了,我們繼續回到源碼,接下來我們具體看下掃描器是怎么掃描到那些被注解標記的類的(其實就是對之前注冊的過濾器的應用),findCandidateComponents()中調用了scanCandidateComponents(),我們之間看scanCandidateComponents():

    // 去除掉了異常處理和日志打印private Set scanCandidateComponents(String basePackage) { Set candidates = new LinkedHashSet<>(); String packageSearchPath = ResourcePatternResolver.CLASSPATH_ALL_URL_PREFIX + resolveBasePackage(basePackage) + '/' + this.resourcePattern; // 這一段邏輯極其復雜切對我們理解主流程沒太大幫助,我們就不看了(主要是涉及到模糊匹配的文件尋找) // 大概就是把所有符合的類文件找出來了 Resource[] resources = getResourcePatternResolver().getResources(packageSearchPath); for (Resource resource : resources) { // 解析文件信息,加載到內存,這里看下去也賊復雜,都是一些字節碼解析技術了 // 我們只需要知道這樣操作一番后,這個MetadataReader能拿到我們這個類的所有信息就好了 MetadataReader metadataReader = getMetadataReaderFactory().getMetadataReader(resource); // 這里就是我們過濾器發揮作用的地方了,符合條件的類才會生成beanDefinition if (isCandidateComponent(metadataReader)) { ScannedGenericBeanDefinition sbd = new ScannedGenericBeanDefinition(metadataReader); sbd.setResource(resource); sbd.setSource(resource); // 這里主要判斷一下,我們匹配到的類是不是一個符合條件的bean // 比如說如果我們注解打在接口上,這里就不會把這個beanDefinition加入返回的容器了 if (isCandidateComponent(sbd)) { candidates.add(sbd); } } } return candidates;}// 過濾器判斷是否是我們關注的類,邏輯很直觀protected boolean isCandidateComponent(MetadataReader metadataReader) throws IOException { // 先判斷的excludeFilters for (TypeFilter tf : this.excludeFilters) { if (tf.match(metadataReader, getMetadataReaderFactory())) { return false; } } // 再判斷的includeFilters for (TypeFilter tf : this.includeFilters) { if (tf.match(metadataReader, getMetadataReaderFactory())) { // 如果是我們關注的類,還需要處理類上面的@Conditional注解 // 這里不繼續往下拓展了,我簡單講一下邏輯: // 1.找到類上面所有的@Conditional簇的注解 // 2.實例化所有對應的Conditional類,并排序 // 3.依次調用所有condition.matches(),所有條件全部滿足才返回true // 具體細節同學們感興趣可以自己看下 return isConditionMatch(metadataReader); } } return false;}// 判斷是否不是接口那些protected boolean isCandidateComponent(AnnotatedBeanDefinition beanDefinition) { return (metadata.isIndependent() // 不是實例內部類 并且 && (metadata.isConcrete() // 不是接口或者抽象類 或者 || (metadata.isAbstract() && metadata.hasAnnotatedMethods(Lookup.class.getName())))); // 是抽象類但是有些方法被@Lookup注解標記,這個之前有稍微提過,xml標簽里那個lookup-method標簽跟這個是一個意思,相當于把這個方法委托/代理給另一個bean了,所以即使是抽象類也是可以變成一個bean的 -> spring動態代理生成一個子類}復制代碼

    3. Filter匹配流程

    本來不想寫Filter的匹配流程的,因為其實不是主流程,不過想想還是寫一下吧,不然有些同學可能會糾結。

    主要講一下我們用的比較多的AnnotationTypeFilter,先看一下AnnotationTypeFilter的構造器:

    // 我們簡單看一下AnnotationTypeFilter的構造器public AnnotationTypeFilter(Class extends Annotation> annotationType) { this(annotationType, true, false);}// 可以看到,我們掃描@Component注解時,是考慮源注解,且不考慮接口上的注解的public AnnotationTypeFilter( // 注解類型 Class extends Annotation> annotationType, // 是否考慮源注解 boolean considerMetaAnnotations, // 是否考慮接口 boolean considerInterfaces) { // 第一個參數是是否考慮繼承的注解 super(annotationType.isAnnotationPresent(Inherited.class), considerInterfaces); this.annotationType = annotationType; this.considerMetaAnnotations = considerMetaAnnotations;}復制代碼

    再看一下核心的match方法,這里也是一個模板方法模式:

    // 先看頂層類public abstract class AbstractTypeHierarchyTraversingFilter implements TypeFilter { @Override public boolean match(MetadataReader metadataReader, MetadataReaderFactory metadataReaderFactory) throws IOException { // 直接看當前類是否匹配 - 模板方法,由子類實現,默認返回了false if (matchSelf(metadataReader)) { return true; } // 提供一個通過className判斷是否匹配的鉤子 ClassMetadata metadata = metadataReader.getClassMetadata(); if (matchClassName(metadata.getClassName())) { return true; } if (this.considerInherited) { // 如果考慮繼承的注解,則找到對應的父類 String superClassName = metadata.getSuperClassName(); if (superClassName != null) { // 先看下子類有沒有 單獨判斷父類是否匹配 的邏輯 Boolean superClassMatch = matchSuperClass(superClassName); if (superClassMatch != null) { // 有寫這個邏輯則直接用這個返回結果了 if (superClassMatch.booleanValue()) { return true; } } else { // 沒有 單獨判斷父類是否匹配 的邏輯 則直接走當前這個匹配邏輯 if (match(metadata.getSuperClassName(), metadataReaderFactory)) { return true; } } } } if (this.considerInterfaces) { // 如果考慮接口的注解,則找到對應的接口,因為接口是多個,所以要循環 // 邏輯和父類那里類似,不多講了 for (String ifc : metadata.getInterfaceNames()) { Boolean interfaceMatch = matchInterface(ifc); if (interfaceMatch != null) { if (interfaceMatch.booleanValue()) { return true; } } else { if (match(ifc, metadataReaderFactory)) { return true; } } } } return false; }}復制代碼

    再看一下AnnotationTypeFilter的幾個核心方法:

    public class AnnotationTypeFilter extends AbstractTypeHierarchyTraversingFilter { protected boolean matchSelf(MetadataReader metadataReader) { AnnotationMetadata metadata = metadataReader.getAnnotationMetadata(); // 類上有目標注解 return metadata.hasAnnotation(this.annotationType.getName()) || // 如果可以從源注解拿,則找一下類上面有沒有源注解是和目標注解一樣的 (this.considerMetaAnnotations && metadata.hasMetaAnnotation(this.annotationType.getName())); } protected Boolean matchSuperClass(String superClassName) { return hasAnnotation(superClassName); } protected Boolean matchInterface(String interfaceName) { return hasAnnotation(interfaceName); } @Nullable protected Boolean hasAnnotation(String typeName) { if (Object.class.getName().equals(typeName)) { return false; } // 這個父類和接口的匹配邏輯居然只能匹配到jdk內置(java開頭)的類 // 看來默認的實現應該是用來支持JSR標準的那些注解的 else if (typeName.startsWith("java")) { // ... 不關注 } return null; }}復制代碼

    我們可以看到,我們默認的AnnotationTypeFilter是考慮源注解的,那么這個源注解到底是個什么東西呢?

    public @interface Controller { @AliasFor(annotation = Component.class) String value() default "";}public @interface Service { @AliasFor(annotation = Component.class) String value() default "";}public @interface Repository { @AliasFor(annotation = Component.class) String value() default "";}復制代碼

    常見的就是這個東西啦@AliasFor(annotation = Component.class),這也是為什么我們默認的includeFilters明明只注冊了一個@Component類型的AnnotationTypeFilter,但是我們@Service等也能被掃描到的原因啦!我們構造的AnnotationTypeFilter是考慮源注解的!

    4. 注冊公共組件

    看到這里,我們已經明白了context:component-scan標簽是怎么掃描,怎么支持@Component注解的了,但是細心的同學們可能已經發現了,現在我們確實能掃描@Component注解了,但是我們bean中那些屬性是怎么注入的呢?@Autowrite、@Resource這些注解是怎么支持的呢?以及@Configuration、@Bean又是如何支持的呢?

    這些功能其實是在相應的BeanPostProcessor中完成的,而這些BeanPostProcessor的注冊,也是在我們context:component-scan標簽的解析過程中注入的。如果同學們還有印象的話,應該還記得ComponentScanBeanDefinitionParser的parse方法中,我們再創建了掃描器并且進行掃描之后,還做了一些公共組件注冊的工作:

    public BeanDefinition parse(Element element, ParserContext parserContext) { // ... // 獲取一個掃描器 - 這個東西很重要,我們以后還會看到 ClassPathBeanDefinitionScanner scanner = configureScanner(parserContext, element); // 嗯,掃描器進行掃描,看來就是這個方法會掃描那些注解了 Set beanDefinitions = scanner.doScan(basePackages); // 注冊一些組件 registerComponents(parserContext.getReaderContext(), beanDefinitions, element); return null;}復制代碼

    我們看一下registerComponents方法:

    protected void registerComponents( XmlReaderContext readerContext, Set beanDefinitions, Element element) { // ... // Register annotation config processors, if necessary. boolean annotationConfig = true; if (element.hasAttribute(ANNOTATION_CONFIG_ATTRIBUTE)) { annotationConfig = Boolean.parseBoolean(element.getAttribute(ANNOTATION_CONFIG_ATTRIBUTE)); } // 看下annotation-config配置,默認是為true的 if (annotationConfig) { Set processorDefinitions = // 注冊一些支撐注解功能的Processors AnnotationConfigUtils.registerAnnotationConfigProcessors(readerContext.getRegistry(), source); // ... } // ...}復制代碼

    那么到底注冊了哪些Processor呢?

    public static Set registerAnnotationConfigProcessors( BeanDefinitionRegistry registry, @Nullable Object source) { // ... Set beanDefs = new LinkedHashSet<>(8); // 這里注冊了一個ConfigurationClassPostProcessor,顧名思義,這個應該是支撐@Configuration相關的注解的 if (!registry.containsBeanDefinition(CONFIGURATION_ANNOTATION_PROCESSOR_BEAN_NAME)) { RootBeanDefinition def = new RootBeanDefinition(ConfigurationClassPostProcessor.class); def.setSource(source); // 注冊邏輯registerPostProcessor beanDefs.add(registerPostProcessor(registry, def, CONFIGURATION_ANNOTATION_PROCESSOR_BEAN_NAME)); } // 注冊了一個AutowiredAnnotationBeanPostProcessor,用來處理@Autowire,@Value注解的 if (!registry.containsBeanDefinition(AUTOWIRED_ANNOTATION_PROCESSOR_BEAN_NAME)) { RootBeanDefinition def = new RootBeanDefinition(AutowiredAnnotationBeanPostProcessor.class); def.setSource(source); beanDefs.add(registerPostProcessor(registry, def, AUTOWIRED_ANNOTATION_PROCESSOR_BEAN_NAME)); } // Check for JSR-250 support, and if present add the CommonAnnotationBeanPostProcessor. // 這里是支撐JSR-250規范的@Resource、@PostConstruct、@PreDestroy注解的 if (jsr250Present && !registry.containsBeanDefinition(COMMON_ANNOTATION_PROCESSOR_BEAN_NAME)) { RootBeanDefinition def = new RootBeanDefinition(CommonAnnotationBeanPostProcessor.class); def.setSource(source); beanDefs.add(registerPostProcessor(registry, def, COMMON_ANNOTATION_PROCESSOR_BEAN_NAME)); } // Check for JPA support, and if present add the PersistenceAnnotationBeanPostProcessor. // 這里是支持注解形式的jpa的BeanPostProcessor if (jpaPresent && !registry.containsBeanDefinition(PERSISTENCE_ANNOTATION_PROCESSOR_BEAN_NAME)) { RootBeanDefinition def = new RootBeanDefinition(); try { def.setBeanClass(ClassUtils.forName(PERSISTENCE_ANNOTATION_PROCESSOR_CLASS_NAME, AnnotationConfigUtils.class.getClassLoader())); } catch (ClassNotFoundException ex) { throw new IllegalStateException( "Cannot load optional framework class: " + PERSISTENCE_ANNOTATION_PROCESSOR_CLASS_NAME, ex); } def.setSource(source); beanDefs.add(registerPostProcessor(registry, def, PERSISTENCE_ANNOTATION_PROCESSOR_BEAN_NAME)); } // 支撐spring-event相關注解的processor,對@EventListener的支撐 if (!registry.containsBeanDefinition(EVENT_LISTENER_PROCESSOR_BEAN_NAME)) { RootBeanDefinition def = new RootBeanDefinition(EventListenerMethodProcessor.class); def.setSource(source); beanDefs.add(registerPostProcessor(registry, def, EVENT_LISTENER_PROCESSOR_BEAN_NAME)); } // 支撐spring-event相關注解的processor,對@EventListener的支撐 if (!registry.containsBeanDefinition(EVENT_LISTENER_FACTORY_BEAN_NAME)) { RootBeanDefinition def = new RootBeanDefinition(DefaultEventListenerFactory.class); def.setSource(source); beanDefs.add(registerPostProcessor(registry, def, EVENT_LISTENER_FACTORY_BEAN_NAME)); } return beanDefs;}復制代碼

    到此,context:component-scan標簽所做的所有事情都做完了。它主要就是創建了一個掃描器來掃描我們基本的需要注冊的bean,之后注冊了一些支撐相應注解功能的Processor,對于這些Processor,我這邊不會去單獨講解每個Processor是什么時候被調用,怎么實現它的功能的,感興趣的同學可以自行找到對應的類去看實現邏輯。

    而之后講bean初始化邏輯和生命周期的時候,我會在特定的拓展點,講到一些Processor的調用以及內部的邏輯,希望到時候同學們還能記起來這些Processor是在哪里注冊的。

    三、實踐

    都說實踐出真知,我們跟著源碼分析了這么一大波,但是事實是不是如我們分析的那樣呢?為了證實一下,這邊我簡單使用一下spring預留的拓展點。

    1. 使用context:component-scan掃描自定義注解

    我們首先需要自定義一個注解:

    @Target({ElementType.TYPE})@Retention(RetentionPolicy.RUNTIME)public @interface MyService {}復制代碼

    然后配置一下context:component-scan標簽:

    復制代碼

    為我們的業務類打上注解:

    @Data@MyServicepublic class MyAnnoClass { public String username = "xiaoxizi";}復制代碼

    運行:

    public void test1() { applicationContext = new ClassPathXmlApplicationContext("spring.xml"); MyAnnoClass myAnnoClass = applicationContext.getBean(MyAnnoClass.class); System.out.println(myAnnoClass);}復制代碼

    輸出結果:

    MyAnnoClass(username=xiaoxizi)復制代碼

    說明我們的自定義注解掃描到了,并且成功生成了beanDefinition并實例化了bean!

    2. 自定義標簽

    先創建一個具體標簽的解析類,我們這邊簡單點,直接繼承了spring內部的一個類:

    public class SimpleBeanDefinitionParse extends AbstractSingleBeanDefinitionParser { @Override protected String getBeanClassName(final Element element) { System.out.println("SimpleBeanDefinitionParse ... getBeanClassName()"); return element.getAttribute("className"); }}復制代碼

    然后創建一個SimpleNamespaceHandler:

    public class SimpleNamespaceHandler extends NamespaceHandlerSupport { @Override public void init() { System.out.println("SimpleNamespaceHandler ... init()"); this.registerBeanDefinitionParser("simpleBean", new SimpleBeanDefinitionParse()); }}復制代碼

    配置寫入META-INF/spring.handlers文件:

    http://www.xiaoxize.com/schema/simple=com.xiaoxizi.spring.tag.SimpleNamespaceHandler復制代碼

    xml配置中使用:

    復制代碼

    目標類:

    @Data// @MyServicepublic class MyAnnoClass { public String username = "xiaoxizi";}復制代碼

    運行:

    public void test1() { applicationContext = new ClassPathXmlApplicationContext("spring.xml"); MyAnnoClass myAnnoClass = applicationContext.getBean(MyAnnoClass.class); System.out.println(myAnnoClass);}復制代碼

    輸出結果-各種報錯,哈哈哈:

    Caused by: org.xml.sax.SAXParseException; lineNumber: 21; columnNumber: 91; cvc-complex-type.2.4.c: 通配符的匹配很全面, 但無法找到元素 'xiaoxizi:simple' 的聲明。復制代碼

    emmmm,翻車車啦~這里還是卡了一會的,主要是對xml規范的不熟悉導致的,原來我們在聲明命名空間的時候,還要聲明并定義對應的XSD文件,(這里我自己寫了一個xsd文件,并通過idea的配置引入了工作空間)像這樣:

    復制代碼

    然后發現還是不行:

    java.net.UnknownHostException: www.xiaoxize.comorg.xml.sax.SAXParseException: schema_reference.4: 無法讀取方案文檔 'http://www.xiaoxize.com/schema/simple.xsd', 原因為 1) 無法找到文檔; 2) 無法讀取文檔; 3) 文檔的根元素不是 。Caused by: org.xml.sax.SAXParseException; lineNumber: 21; columnNumber: 91; cvc-complex-type.2.4.c: 通配符的匹配很全面, 但無法找到元素 'xiaoxizi:simple' 的聲明。復制代碼

    啊,原來spring解析這個xml的時候,是不歸idea管的,他還是會去對應的域名下找這個xsd文件(而我根本沒有xiaoxizi這個域名...),最后我把xsd文件丟到自己服務器上,并且調整了域名那些,終于可以了:

    SimpleNamespaceHandler ... init()SimpleBeanDefinitionParse ... getBeanClassName()MyAnnoClass(username=xiaoxizi)復制代碼

    大功告成,所以自定義標簽還是蠻簡單的嘛(認真臉!

    四、總結

    1. 自定義標簽解析過程

  • 第一個自定義標簽開始解析時,將會從所有jar包的META-INF/spring.handlers文件加載 自定義標簽命名空間-對應NamespaceHandler全限定名到內存中的DefaultNamespaceHandlerResolver.handlerMappings
  • 同樣前綴的自定義標簽第一次解析時,將會實例化對應的NamespaceHandler,并調用其init()方法,然后把自定義標簽命名空間-對應NamespaceHandler實例放入handlerMappings,下次再有同樣的標簽過來解析,就直接能拿到對應的NamespaceHandler實例了
  • 使用找到的NamespaceHandler實例的parse方法解析自定義標簽
  • spring貼心的為我們準備了NamespaceHandler相關的模版類NamespaceHandlerSupport,如果我們自定義的處理器繼承了這個模版,那只需要在init方法中為具體的標簽注入相應的BeanDefinitionParser或者BeanDefinitionDecorator就可以實現功能了
  • 2. context:component-scan做了什么

  • 自定義標簽context:component-scan對應的解析器是ComponentScanBeanDefinitionParser(找的過程我們不贅述了)。
  • 解析器的parse方法中,我們通過標簽配置的屬性創建了一個掃描器ClassPathBeanDefinitionScanner
  • 默認情況下, 我們會注冊一個@Component注解的AnnotationTypeFilter,并且注冊到掃描器的includeFilters中
  • 然后掃描器開始掃描basePackage下所有的java類,并且找到所有不需要排除(excludeFilters)的候選類(includeFilters),然后為其生成一個beanDefinition,如果該類是一個合法的beanDefinition(非接口那些判斷),那么就會將這些beanDefinition收集起來并返回
  • 對于所有的候選beanDefinition,掃描器還會進一步掃描類上的@Lazy、@Primary、@DependsOn等屬性,然后設值到beanDefinition的對應屬性中
  • 最后一步,把我們所有掃描到的所有合法的beanDefinition注冊到IOC容器
  • 由于@Component是@Service、@Controller等注解的源注解,所以@Service這些注解標記的類也會被includeFilters掃描到
  • 注冊一系列對@Configuration、@Autowired、@Resource等注解進行支撐的Processor
  • 五、其他

    實踐中使用的的簡單的XSD文件

    <?xml version="1.0" encoding="UTF-8"?>復制代碼

    多多支持,即可免費獲取以上所有資料--轉發+評論,關注我,私信口令 “Java” ,(承諾100%免費)

    希望大家將此篇文章分享,轉載,讓更多需要的朋友看到,這樣不僅幫助了自己,也幫助了他人,謝謝!!

    總結

    以上是生活随笔為你收集整理的component是什么接口_逐行解读Spring(二)什么,自定义标签没听说过?的全部內容,希望文章能夠幫你解決所遇到的問題。

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

    国产高清综合 | 美女久久久久久久久久 | 国产.精品.日韩.另类.中文.在线.播放 | 久久精品韩国 | 狠狠地日 | 久久官网 | 很黄很色很污的网站 | 久久女同性恋中文字幕 | 国产成人久久 | 国产r级在线观看 | 00av视频| 精品不卡视频 | 五月婷网站 | 在线中文字幕视频 | 99久久精品一区二区成人 | 国产一级视频在线观看 | 中文字幕首页 | 欧美激情精品久久久 | 人人干天天干 | 久久96国产精品久久99软件 | 久久综合色天天久久综合图片 | 国产精品九九久久99视频 | 中文字幕日韩电影 | 一区二区三区免费播放 | 99在线观看视频网站 | 91九色国产蝌蚪 | 国产日韩精品在线 | 国产免费观看久久黄 | 日日碰狠狠躁久久躁综合网 | 欧美日韩中文字幕视频 | 色吧av色av | 一本一本久久a久久精品综合妖精 | 中文字幕日韩电影 | 99久久精品费精品 | 精品视频免费 | 国产精品久久久久久久免费观看 | 日韩中文字幕免费在线观看 | 天天操人人要 | 亚洲乱码精品久久久久 | 成人av一区二区三区 | 夜夜躁天天躁很躁波 | 国产视频一区二区三区在线 | 日韩电影在线观看一区二区 | 亚洲理论在线观看 | 成人黄色大片在线观看 | 日韩午夜网站 | 成人av资源网| 婷婷在线视频观看 | 天天色天天干天天色 | 夜色资源网 | 久久精品视频网 | 日韩理论在线播放 | 成 人 免费 黄 色 视频 | 欧美精品一区在线发布 | 久艹视频在线观看 | 欧美精品一区二区三区四区在线 | 在线免费观看成人 | 99在线高清视频在线播放 | 日韩丝袜在线观看 | 99电影| 91久久精品日日躁夜夜躁国产 | 久久无码精品一区二区三区 | 欧美一级性视频 | 久久久久久欧美二区电影网 | 九九在线国产视频 | 麻豆视频在线播放 | 狠狠网站 | 久久 国产一区 | 免费黄色小网站 | 97人人澡人人添人人爽超碰 | 一区二区三区四区在线免费观看 | 操操操人人 | 在线观看视频福利 | 在线观看亚洲a | 日韩欧美网址 | 久章草在线 | 精品国产乱码久久久久久1区二区 | 精品女同一区二区三区在线观看 | 免费看搞黄视频网站 | 欧美亚洲免费在线一区 | 国产精品久久一区二区三区不卡 | 亚洲精品大片www | 久久婷婷五月综合色丁香 | 午夜999 | 国产伦精品一区二区三区高清 | 天天操天天操天天操天天 | a级国产乱理伦片在线播放 久久久久国产精品一区 | 久久国产精品久久精品国产演员表 | 亚洲精品h | 四虎成人精品在永久免费 | 国产日韩精品在线 | 在线看日韩| 久久久久久久久久亚洲精品 | 免费看麻豆 | 日本中文字幕免费观看 | 日本aaaa级毛片在线看 | 久久免费看av | 久久精品—区二区三区 | 国产黄色精品视频 | 久久人网| 久久激情视频 久久 | 日日夜夜狠狠操 | 美女免费视频观看网站 | 欧美激情操 | 久草网首页| 精品产品国产在线不卡 | 在线久草视频 | 日本系列中文字幕 | 天堂黄色片 | 国产成人三级一区二区在线观看一 | 日韩高清精品免费观看 | 欧美9999 | 国产精品嫩草影视久久久 | 手机av电影在线观看 | 日本在线观看视频一区 | 国产精品一区专区欧美日韩 | 久久av不卡| 免费福利影院 | 中文字幕亚洲高清 | 精品久久99 | 国产精品久久久久av | 91精品视频在线观看免费 | 国产精品久久久一区二区三区网站 | www.夜夜| 999热线在线观看 | 国产精品国产亚洲精品看不卡 | 亚洲视频久久久久 | 国偷自产中文字幕亚洲手机在线 | 国产亚洲精品精品精品 | 波多野结衣久久精品 | 狠狠色伊人亚洲综合网站野外 | 天天草天天干天天射 | 久久久久久久久久久网 | 亚洲国产中文字幕在线观看 | 中文字幕丝袜一区二区 | 美女视频一区二区 | 欧美成人999 | 国产在线播放不卡 | 一区二区三区电影 | 国产成人区 | 欧美精品久久久久性色 | 91中文字幕视频 | 99视频导航 | 久久草av| 视频一区在线免费观看 | a久久久久久 | 欧美精品第一 | 婷婷天天色| 一区二区三区高清在线观看 | 久久 亚洲视频 | 麻豆影视网站 | 国产91精品一区二区麻豆亚洲 | a级国产乱理伦片在线播放 久久久久国产精品一区 | 午夜av免费看 | 久久成年人 | 日韩一级黄色片 | 国产精品久久网站 | 亚洲欧美激情精品一区二区 | 国产精品毛片久久久久久久久久99999999 | 日韩,精品电影 | 欧美孕交vivoestv另类 | 亚洲韩国一区二区三区 | 99精品欧美一区二区三区黑人哦 | 久久精品免费 | 欧美亚洲精品在线观看 | 日日射天天射 | 婷婷在线不卡 | 国产视频午夜 | 精品国产123 | 日日干 天天干 | 国产精品久久久久久久免费 | 在线激情小视频 | 亚洲免费观看在线视频 | 天堂中文在线视频 | 国产91电影在线观看 | 亚av在线 | 国产成人精品国内自产拍免费看 | 色婷婷成人 | 97免费视频在线 | 中文字幕人成人 | 日韩草比 | 99视频在线观看视频 | 免费一级特黄毛大片 | www视频在线播放 | 91传媒在线观看 | 黄色影院在线免费观看 | 91网免费观看 | 精品久久五月天 | 国产黄色片久久久 | 欧美精品久久天天躁 | 在线观看黄污 | 久久激五月天综合精品 | 久久精品国产亚洲精品 | 国产精品久久久久久久久久三级 | 在线观看中文字幕一区二区 | 亚洲小视频在线观看 | 日韩动漫免费观看高清完整版在线观看 | 久久精品99精品国产香蕉 | 中文在线√天堂 | 亚洲在线精品 | 成人动图 | 天天干夜夜夜操天 | 狠狠狠色丁香婷婷综合激情 | 国产伦精品一区二区三区高清 | 色在线免费观看 | 精品在线观看视频 | 一区三区在线欧 | 麻豆综合网 | 亚洲国内在线 | 国产一区二区视频在线播放 | 九九久久成人 | 99精品在线视频观看 | 国产成人精品不卡 | 亚洲成av人片在线观看无 | 色一级片 | 欧美日韩另类视频 | 99热网站| 黄色免费网站 | 久久99中文字幕 | 九七视频在线 | 粉嫩av一区二区三区四区五区 | 久久色在线播放 | 日韩18p| 毛片一级免费一级 | 丝袜制服天堂 | av中文字幕第一页 | 91亚洲精品久久久蜜桃网站 | 国产精品福利在线观看 | 国产又粗又长又硬免费视频 | 国产呻吟在线 | 欧美a级在线播放 | 久久免费看a级毛毛片 | 欧美国产不卡 | 亚洲黄色一级视频 | 久久99国产精品二区护士 | 亚洲精品h| 在线观看91精品国产网站 | 成人免费视频播放 | 亚洲精品国内 | 在线国产中文字幕 | 四虎永久国产精品 | 国产97在线视频 | 久久情爱 | 国产少妇在线观看 | 亚洲欧美日本国产 | 激情喷水 | 91亚洲影院| 久香蕉| 热久久免费国产视频 | 九九九热精品免费视频观看网站 | 91福利区一区二区三区 | 欧美午夜久久久 | 亚洲一区av | 国产手机视频精品 | 99久久综合狠狠综合久久 | 久久国产系列 | 免费视频一区 | 国产精品视频线看 | 国产做aⅴ在线视频播放 | 日韩成人欧美 | 少妇自拍av| 精品a级片| 69精品视频在线观看 | 日韩字幕 | 热久久精品在线 | 日韩av电影中文字幕在线观看 | 在线日韩中文 | 91精品一区在线观看 | 91精品国产自产老师啪 | 欧美精品久久久久久久久久久 | 在线免费观看视频一区二区三区 | 亚洲视频在线观看网站 | 日本高清免费中文字幕 | 蜜臀av性久久久久蜜臀aⅴ流畅 | 久久精品久久综合 | 一级淫片在线观看 | 一区 在线观看 | 久久久免费av | 黄色在线观看网站 | 搡bbbb搡bbb视频 | 不卡av免费在线观看 | 91超级碰碰 | av超碰在线 | 狠狠综合久久 | 全久久久久久久久久久电影 | 午夜影院一级 | 国产日产在线观看 | 天天干天天射天天插 | 国产色在线 | 亚洲成人黄色av | 永久黄网站色视频免费观看w | 国产精品精品国产色婷婷 | 国产小视频在线播放 | 欧美91精品久久久久国产性生爱 | 免费在线国产 | 久久国产免费看 | 免费午夜在线视频 | 日本亚洲国产 | 91亚洲欧美| 成人污视频在线观看 | 日韩一级黄色av | 亚洲三级在线播放 | 日产av在线播放 | 亚州黄色一级 | 粉嫩一区二区三区粉嫩91 | 免费视频成人 | 网站在线观看日韩 | 韩国三级av在线 | 美女视频久久久 | 久久情侣偷拍 | 国产中出在线观看 | 在线观看福利网站 | 日日夜夜av| 国产精品视频免费在线观看 | 婷婷丁香九月 | 久久久精品一区二区三区 | 97超级碰碰碰视频在线观看 | 久久国产精品二国产精品中国洋人 | 国内视频1区 | 免费看的国产视频网站 | 夜夜操网 | 在线韩国电影免费观影完整版 | 97超碰在线久草超碰在线观看 | 欧美日韩二区在线 | 超碰在线观看av | 国产在线观看99 | 亚洲成熟女人毛片在线 | 狠狠久久伊人 | 国产一区二区三区网站 | 婷婷色吧 | 天天看天天干 | 九九热久久免费视频 | 一区二精品 | 一级大片在线观看 | 麻豆成人小视频 | 国产精品久久久久久电影 | 丁香视频在线观看 | 91资源在线视频 | 免费影视大全推荐 | 在线观看网站av | 五月天九九 | 黄色www| 麻豆一二三精选视频 | 超碰国产在线播放 | 激情五月激情综合网 | 亚洲精品国产成人 | 综合激情网 | 色片网站在线观看 | 亚洲成人xxx| 国内成人综合 | 国产精品久久久久久超碰 | 国产成人精品999 | 久久免费a | 日本 在线 视频 中文 有码 | 丁香五月缴情综合网 | 精品视频一区在线 | 亚洲视频456| 婷婷丁香在线观看 | 麻豆视频成人 | 国产精品 国产精品 | 国产精品视频app | 91精品国产91久久久久久三级 | 97天堂网 | www.91成人| 99精品国产亚洲 | 国产精品视频资源 | 中文字幕在线免费看线人 | 亚洲japanese制服美女 | 91精品中文字幕 | 日韩黄色软件 | 久久免费在线观看视频 | 免费av在线网 | 2024av在线播放| 99久久精品国产欧美主题曲 | 国产成人av在线影院 | 开心激情婷婷 | 久久亚洲国产精品 | 免费观看视频的网站 | 视频在线观看日韩 | 欧洲黄色片| 久久免费视频在线观看30 | 国产五月色婷婷六月丁香视频 | 91亚洲精品久久久久图片蜜桃 | 综合久久久久久久久 | av网站手机在线观看 | 色综合中文综合网 | 99热这里是精品 | 91成人在线免费观看 | 国产精品久久久久久一区二区三区 | 97av影院| 亚洲人天堂 | 久久精品7 | 亚洲午夜在线视频 | 一色av| www国产亚洲精品久久网站 | 亚洲成人精品久久 | 亚洲狠狠丁香婷婷综合久久久 | 天堂av在线网站 | 久久精品一区八戒影视 | 欧美二区三区91 | 久久人人精 | 国产精品一区二区在线观看 | 久久久久久久毛片 | 人人cao| 国产成人亚洲在线观看 | 欧洲精品在线视频 | 亚洲精品久久久久中文字幕m男 | 欧美日韩国产在线精品 | 在线观看av大片 | 久久黄色美女 | www.夜夜爽 | 色婷婷 亚洲 | 四虎免费av | 天天透天天插 | 国产在线久久久 | 91精品秘密在线观看 | 日韩精品在线免费播放 | 免费情趣视频 | 免费视频黄色 | 伊人久久一区 | 国产免费看| 视频成人永久免费视频 | 亚洲精品国产精品国自产在线 | 精品少妇一区二区三区在线 | 国产中年夫妇高潮精品视频 | 麻豆视频国产 | 天天操天天操一操 | 国产成人精品一区二区三区免费 | 久久久99精品免费观看乱色 | 国产精品久久久久久吹潮天美传媒 | 97理论电影 | 精品免费在线视频 | 在线一二区| 亚洲精品高清一区二区三区四区 | 国产精品一区二区三区在线 | 国产精品一区免费观看 | 国产少妇在线观看 | 99c视频在线| 中文字幕 国产视频 | 又黄又爽的视频在线观看网站 | av在线电影播放 | av电影亚洲 | 国产美女在线精品免费观看 | a级成人毛片 | 成人黄色大片在线观看 | 国产一区二区视频在线播放 | 99久久99久国产黄毛片 | 色网站免费在线看 | 久久国产视频网 | 国产成人99久久亚洲综合精品 | 久久久亚洲精品 | 91av视频在线观看免费 | 2019天天干天天色 | 亚洲成年人av | 亚洲三级性片 | 国产一级二级av | 一个色综合网站 | 亚洲一区二区观看 | 99精品热视频| 69国产盗摄一区二区三区五区 | 亚洲第一久久久 | 亚洲在线网址 | caobi视频| 在线免费三级 | 国产亚洲精品成人 | 99视 | av一区在线| 99久热在线精品视频观看 | 亚洲经典视频 | 成人综合婷婷国产精品久久免费 | 午夜精品久久一牛影视 | 欧美激情精品久久久久久变态 | av成人免费 | 国产在线污 | 国产一二区视频 | 日本一区二区三区免费看 | 日韩中文久久 | 亚洲精品国产精品国自产在线 | bayu135国产精品视频 | 国产精品麻豆91 | 天天操天天草 | 玖玖视频国产 | 激情文学综合丁香 | 99精品视频播放 | 狠狠躁夜夜躁人人爽超碰97香蕉 | 在线看国产 | 国产69精品久久app免费版 | 国产精品岛国久久久久久久久红粉 | 中文字幕日韩有码 | 久久玖 | 最近的中文字幕大全免费版 | 中文字幕中文字幕中文字幕 | 国产成人一区二区三区久久精品 | 日韩中出在线 | 免费看黄色91 | 狠狠做深爱婷婷综合一区 | 在线亚洲精品 | 黄视频网站大全 | 97精品国产91久久久久久久 | 亚洲欧美国产精品va在线观看 | 欧美成人h版在线观看 | 奇米四色影狠狠爱7777 | 午夜精品一区二区三区在线 | 黄色免费网 | 久久激情综合 | 韩国av一区二区 | 精品国产乱码久久久久久1区二区 | 丁香午夜| 国产一区二区在线免费播放 | 天天操天天爱天天爽 | 99tvdz@gmail.com | 亚洲精品中文在线观看 | 天天射狠狠干 | 中文字幕在线看视频国产中文版 | 国产日韩欧美在线观看 | 91精品国产91久久久久久三级 | 免费h在线观看 | 亚洲1区在线 | a级国产片| 天天躁日日躁狠狠躁 | 麻豆91在线观看 | 亚洲专区 国产精品 | 成年人视频免费在线播放 | 免费精品视频在线 | 成人播放器 | 91视频在线国产 | 天天爽天天摸 | 黄污污网站| 日韩亚洲在线观看 | 伊人亚洲综合 | 精品久久毛片 | 婷婷激情影院 | 中文字幕在线看视频 | aaa日本高清在线播放免费观看 | 久久草在线视频国产 | 久久久久久久久综合 | 久草在线视频精品 | 亚洲综合视频在线 | 国产黄色片在线免费观看 | 在线观看成人福利 | 成人日韩av | 中文视频在线 | 婷婷综合影院 | 亚洲国产综合在线 | 啪啪免费试看 | 人人干人人草 | 成人免费视频网站 | 亚洲国产日韩一区 | 青草视频网 | 久久精品国产一区二区三 | 伊人射| 一区二区三区精品久久久 | 国产69久久精品成人看 | 久久久久在线观看 | 国产精品九九九九九九 | 免费av高清| 夜色资源站国产www在线视频 | 亚洲国内精品在线 | 中文字幕色在线视频 | 久久免费的视频 | 中文字幕在线观看网站 | 欧美疯狂性受xxxxx另类 | 亚洲精区二区三区四区麻豆 | 日日夜夜噜 | 久草在线观看视频免费 | 亚洲好视频 | 97香蕉超级碰碰久久免费软件 | 97免费视频在线播放 | 久久爱992xxoo| 午夜精品福利一区二区三区蜜桃 | 91日韩精品视频 | 日日夜夜天天射 | 狠狠网 | 国产视频一区在线 | 久久国产精品系列 | 国产精品video爽爽爽爽 | 日韩有码第一页 | 久久婷婷国产色一区二区三区 | 国产精品久久久久久久久久了 | 国产午夜精品久久久久久久久久 | 精品一区二区影视 | 精品在线一区二区三区 | 日韩av不卡在线 | 午夜精品中文字幕 | 久久综合色天天久久综合图片 | 亚洲日本中文字幕在线观看 | 国产精品欧美一区二区 | 99精品偷拍视频一区二区三区 | 亚洲激情网站免费观看 | 成人av午夜| 婷婷在线网站 | 五月婷婷丁香色 | 国产成人免费在线观看 | 国产亚州av | 天天综合亚洲 | 在线观看视频国产 | 日韩av电影中文字幕 | 青春草视频 | 久久99久久99精品中文字幕 | 精品999在线| 亚洲黄色大片 | 久草免费在线视频观看 | 97视频在线免费 | 91毛片在线观看 | 综合久久综合久久 | 一区二区三区精品久久久 | 国产精品第一页在线观看 | 最新中文字幕在线播放 | 激情久久综合 | 国产精品成人一区二区三区吃奶 | 亚洲 欧美 精品 | 91成年人在线观看 | 99这里只有久久精品视频 | 黄色片免费电影 | 色噜噜色噜噜 | 久久伊人八月婷婷综合激情 | 天天草网站 | 久久综合色婷婷 | 欧美日韩视频 | 国产九九精品视频 | 婷婷丁香在线视频 | 99久热在线精品视频成人一区 | 91精品一区二区在线观看 | 精品欧美一区二区三区久久久 | 久久精品电影院 | 午夜精品久久久久久久久久 | 成人小视频在线免费观看 | 久久黄色网址 | 极品久久久久久久 | 天天干天天上 | 91亚洲欧美| 国产九色在线播放九色 | 激情综合五月 | 免费观看91视频 | 亚洲午夜精品久久久 | 免费在线观看国产黄 | 色婷婷狠狠| 婷婷六月激情 | 久久综合久久综合这里只有精品 | 国产一区在线视频观看 | 精品久久99 | 狠狠狠色狠狠色综合 | 超碰在线天天 | 日韩久久久久久 | 色就是色综合 | 午夜狠狠操 | 91av精品| 久久久久久久久久国产精品 | 精品视频www | 天天操天天摸天天射 | 国产 一区二区三区 在线 | 中文字幕在线观看网址 | 精品成人久久 | 九九免费精品视频 | 天天射日 | 天天综合中文 | 婷婷精品在线视频 | 豆豆色资源网xfplay | 亚洲干视频在线观看 | 99久久精品免费看国产一区二区三区 | 91桃色国产在线播放 | 人人玩人人爽 | 在线97| 国产高清在线观看 | 亚洲激情校园春色 | 国产精品久久久久毛片大屁完整版 | 一区二区三区www | 久草av在线播放 | 黄色免费网站下载 | 国产精品一级在线 | 一本一本久久a久久精品综合妖精 | 国产成人一区二 | 在线免费观看成人 | 丁香婷婷电影 | 国产精品成人在线观看 | 久久久精选 | 色播六月天 | 91麻豆高清视频 | 亚洲一区二区黄色 | 国产精品久久久久久久久婷婷 | 成人动漫一区二区三区 | 九九免费在线观看 | 激情久久五月 | 免费观看一级成人毛片 | 成人精品福利 | 亚洲日b视频 | 亚洲精品视频一 | 毛片网站在线观看 | 色婷婷精品 | 久久成 | 在线观看免费黄色 | 91精品国产91久久久久 | www夜夜操| 五月婷婷激情综合网 | 国产高清视频在线 | 婷婷丁香狠狠爱 | 国精产品999国精产 久久久久 | 久久视频国产 | 中文字幕免费观看 | 亚洲精品自拍视频在线观看 | 最新一区二区三区 | 久久视频6 | www五月天com | 国产色视频一区二区三区qq号 | 久久精品站 | 日韩天天干 | 中文字幕字幕中文 | 国产免费又粗又猛又爽 | 欧美激情精品 | 久久av电影| 人人爽人人看 | www91在线 | 国内精品久久久久影院优 | 欧美性视频网站 | 麻豆国产精品va在线观看不卡 | 美女福利视频 | 亚洲91精品在线观看 | 亚洲日本欧美在线 | 国产成人三级一区二区在线观看一 | 亚洲综合视频在线 | 国产成人精品在线观看 | a级黄色片视频 | 婷婷久久婷婷 | 亚洲人成人天堂h久久 | 精品国产一区二区三区在线 | 伊人资源视频在线 | 免费国产一区二区 | 中文字幕人成不卡一区 | 日日狠狠 | 亚洲欧美日韩国产精品一区午夜 | 国产免费小视频 | 免费a v视频 | 狠狠色狠狠色终合网 | 草久久影院 | 日韩欧美69 | 久久久久久久久久电影 | 国产精品自产拍 | 97精品视频在线播放 | 国产v欧美 | 18国产精品福利片久久婷 | 国产美女网站在线观看 | 成人黄色在线看 | 欧美日韩午夜爽爽 | 欧美精品一区在线 | 亚洲国产理论片 | 亚洲电影黄色 | 中文字幕在线观看视频一区 | 亚洲永久精品国产 | 欧美日韩国产一区二 | 国产精品久久久久久久久久白浆 | 亚洲激色| 亚洲免费不卡 | 超碰97在线资源 | 综合伊人av| 久久久免费电影 | 97福利在线观看 | 成年人精品 | 天天色综合久久 | 国产视频精选在线 | 国产午夜亚洲精品 | 亚洲色视频 | av高清不卡 | 综合久久精品 | 国产精品av电影 | 亚洲精品免费观看视频 | 精品在线观看一区二区三区 | 亚洲高清av在线 | 97在线观看免费视频 | 91麻豆精品国产91久久久久久久久 | 婷婷精品视频 | 91超碰免费在线 | 婷婷av在线 | 91中文字幕一区 | 国产一级黄大片 | 国产黄色大全 | 8x8x在线观看视频 | 天天操天天色天天射 | 欧美激情xxxx性bbbb | 色资源网免费观看视频 | 精品视频在线免费 | 日韩网站在线 | 亚洲欧美日韩一级 | 日本中文字幕电影在线免费观看 | 中文字幕888| 99精品视频在线观看 | 国产69精品久久99的直播节目 | 91免费看黄色 | 99久热在线精品 | 国产高清黄 | 制服丝袜一区二区 | 亚洲清纯国产 | 99精品视频在线观看播放 | 久久成人在线 | 日本 在线 视频 中文 有码 | 午夜精品久久久久久久久久久久 | 欧美韩国日本在线观看 | 国产午夜精品视频 | 欧美色888| 中文综合在线 | 色网免费观看 | 亚洲影视资源 | 久久综合狠狠综合久久综合88 | 国产精品福利久久久 | 在线观看视频h | 日韩理论电影在线 | 深爱激情久久 | 综合久久久久久久 | 99在线热播精品免费99热 | 日韩成人精品一区二区三区 | 久久99久久99精品中文字幕 | 免费观看日韩 | 日韩视频免费在线观看 | 婷婷成人综合 | 婷婷精品国产欧美精品亚洲人人爽 | 一区二区三区久久精品 | 国产无吗一区二区三区在线欢 | 成人免费观看视频网站 | 久久伊人国产精品 | 久久精品久久久久久久 | 婷婷色在线视频 | 免费观看性生交大片3 | 国产不卡av在线播放 | 国产精品 日韩精品 | www.天天射 | 亚洲精品成人 | 国产成人精品日本亚洲999 | 西西www4444大胆在线 | 成人污视频在线观看 | 国产精品正在播放 | 国产小视频在线看 | 狠狠久久综合 | 国产精品一区二区视频 | 日本韩国在线不卡 | 色欧美成人精品a∨在线观看 | 日韩精品一区电影 | 人人草在线视频 | 国产不卡一 | 人人爽人人爽人人片 | 97视频在线免费观看 | 国产婷婷 | 国产只有精品 | 最近中文字幕完整高清 | 夜色资源站国产www在线视频 | 激情黄色一级片 | 一本一本久久a久久精品综合小说 | 天天操综 | 热久久最新地址 | 欧美一级xxxx | 久久精品男人的天堂 | 97成人在线 | 午夜av一区二区三区 | 伊人手机在线 | 黄色成人av| 午夜久久精品 | 九七视频在线观看 | 久久精品99国产精品亚洲最刺激 | 最近中文字幕免费观看 | 久久大片| 最近2019中文免费高清视频观看www99 | 99久热在线精品视频成人一区 | 在线看国产一区 | 免费看一级特黄a大片 | 又黄又刺激| 精品国产一区二区三区蜜臀 | 麻豆一区在线观看 | 不卡的av在线播放 | avove黑丝 | 日韩一区精品 | 国产精品成人一区二区 | 中文字幕永久 | 美女av电影 | 亚洲国产天堂av | 亚洲一区欧美精品 | www在线免费观看 | 欧美人体xx | 91精品视屏 | 2000xxx影视| 午夜精品一二区 | 国内精品久久久久影院日本资源 | 国产一区成人 | 91午夜精品 | 午夜狠狠操 | 亚洲精品看片 | 国产精品扒开做爽爽的视频 | 久久精品国产99国产 | 日韩在线网址 | 日韩在线观看你懂的 | 国产精品美女999 | av色综合网 | 久久综合九色综合久99 | 久久亚洲综合色 | 98超碰在线 | 免费日韩 精品中文字幕视频在线 | 黄a在线观看 | 最新久久免费视频 | 一级a毛片高清视频 | 欧美一级xxxx | 亚洲另类人人澡 | 视频三区| 午夜av免费在线观看 | 久久丁香 | 日韩欧美一区二区三区免费观看 | 在线观看中文字幕第一页 | 国产精品久久久区三区天天噜 | 在线观看蜜桃视频 | 欧美午夜精品久久久久久孕妇 | 国产精品久久久久婷婷 | 啪啪肉肉污av国网站 | 亚洲视频资源在线 | 99精品欧美一区二区蜜桃免费 | 国产不卡免费 | 亚洲精品乱码久久久久久蜜桃欧美 | 在线免费中文字幕 | 亚洲 精品在线视频 | 9草在线 | 少妇自拍av| 美女视频国产 | 中文字幕 婷婷 | av在线进入 | 国产区av在线 | 国产精品普通话 | 国产精国产精品 | 亚洲mv大片欧洲mv大片免费 | 国产人成看黄久久久久久久久 | a视频免费 | 免费国产ww | 亚洲精品国产拍在线 | 亚洲国产精品电影在线观看 | 久久精品欧美一 | 久久99精品热在线观看 | 最近日本mv字幕免费观看 | 狠狠色噜噜狠狠 | 91丨精品丨蝌蚪丨白丝jk | 日本韩国在线不卡 | 1024手机看片国产 | 黄色免费在线视频 | 国产福利久久 | 97免费| 日韩天天操 | 亚洲婷婷网 | 五月天中文字幕mv在线 | 国产精品理论在线观看 | 欧美激情综合色综合啪啪五月 | 美女黄频免费 | 久久艹艹| 日韩精品免费一区二区 | 黄色小说视频在线 | 国产精品黄色影片导航在线观看 | 激情网在线视频 | 天天av天天 | 午夜久久福利 | 久草视频在线资源站 | 99国产高清 | 免费精品| 天天拍天天干 | 一区二区三区四区五区在线 | 欧美少妇xxxxxx | 久久久麻豆精品一区二区 | 天天舔天天射天天操 | 亚洲国产字幕 | 国产精品1024 | 少妇bbw搡bbbb搡bbb | 国产a国产a国产a | av.com在线| 久久无码精品一区二区三区 | 超碰精品在线 | 色天天综合久久久久综合片 | 日本久久影视 | 久久精品国产一区二区三 | 免费看三级网站 | 在线免费黄网站 | 日韩精品久久一区二区 | 色综合久久久网 | 亚洲精品国产综合99久久夜夜嗨 | 四虎影视国产精品免费久久 | 狠狠干我 | 国产色婷婷精品综合在线手机播放 | 日日干美女 | 超碰在线97观看 | 2021国产视频 | 久久福利 | 国产亚洲精品久久久久秋 | 欧美最猛性xxx | 欧美日韩p片 | 黄色软件视频网站 | 伊人国产女 | 亚洲第一区在线播放 | 日韩精品一区二区三区三炮视频 | 国产不卡片| 少妇高潮流白浆在线观看 | 欧美精品v国产精品v日韩精品 | 久久精品牌麻豆国产大山 | 色综合久久88色综合天天 | 国产精品黄 | 亚洲最大激情中文字幕 | 麻豆免费视频网站 | 激情综合网五月婷婷 | 欧美小视频在线 |