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

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

這一次搞懂Spring代理創建及AOP鏈式調用過程操作

瀏覽:114日期:2023-08-19 09:21:07

前言

AOP,也就是面向切面編程,它可以將公共的代碼抽離出來,動態的織入到目標類、目標方法中,大大提高我們編程的效率,也使程序變得更加優雅。如事務、操作日志等都可以使用AOP實現。這種織入可以是在運行期動態生成代理對象實現,也可以在編譯期、類加載時期靜態織入到代碼中。而Spring正是通過第一種方法實現,且在代理類的生成上也有兩種方式:JDK Proxy和CGLIB,默認當類實現了接口時使用前者,否則使用后者;另外Spring AOP只能實現對方法的增強。

正文

基本概念

AOP的術語很多,雖然不清楚術語我們也能很熟練地使用AOP,但是要理解分析源碼,術語就需要深刻體會其含義。

增強(Advice):就是我們想要額外增加的功能

目標對象(Target):就是我們想要增強的目標類,如果沒有AOP,我們需要在每個目標對象中實現日志、事務管理等非業務邏輯

連接點(JoinPoint):程序執行時的特定時機,如方法執行前、后以及拋出異常后等等。

切點(Pointcut):連接點的導航,我們如何找到目標對象呢?切點的作用就在于此,在Spring中就是匹配表達式。

引介(Introduction):引介是一種特殊的增強,它為類添加一些屬性和方法。這樣,即使一個業務類原本沒有實現某個接口,通過AOP的引介功能,我們可以動態地為該業務類添加接口的實現邏輯,讓業務類成為這個接口的實現類。

織入(Weaving):即如何將增強添加到目標對象的連接點上,有動態(運行期生成代理)、靜態(編譯期、類加載時期)兩種方式。

代理(Proxy):目標對象被織入增強后,就會產生一個代理對象,該對象可能是和原對象實現了同樣的一個接口(JDK),也可能是原對象的子類(CGLIB)。

切面(Aspect、Advisor):切面由切點和增強組成,包含了這兩者的定義。

代理對象的創建

在熟悉了AOP術語后,下面就來看看Spring是如何創建代理對象的,是否還記得上一篇提到的AOP的入口呢?在AbstractAutowireCapableBeanFactory類的applyBeanPostProcessorsAfterInitialization方法中循環調用了BeanPostProcessor的postProcessAfterInitialization方法,其中一個就是我們創建代理對象的入口。

這里是Bean實例化完成去創建代理對象,理所當然應該這樣,但實際上在Bean實例化之前調用了一個resolveBeforeInstantiation方法,這里實際上我們也是有機會可以提前創建代理對象的,這里放到最后來分析,先來看主入口,進入到AbstractAutoProxyCreator類中:

public Object postProcessAfterInitialization(@Nullable Object bean, String beanName) { if (bean != null) { Object cacheKey = getCacheKey(bean.getClass(), beanName); if (!this.earlyProxyReferences.contains(cacheKey)) { return wrapIfNecessary(bean, beanName, cacheKey); } } return bean; } protected Object wrapIfNecessary(Object bean, String beanName, Object cacheKey) { //創建當前bean的代理,如果這個bean有advice的話,重點看 // Create proxy if we have advice. Object[] specificInterceptors = getAdvicesAndAdvisorsForBean(bean.getClass(), beanName, null); //如果有切面,則生成該bean的代理 if (specificInterceptors != DO_NOT_PROXY) { this.advisedBeans.put(cacheKey, Boolean.TRUE); //把被代理對象bean實例封裝到SingletonTargetSource對象中 Object proxy = createProxy( bean.getClass(), beanName, specificInterceptors, new SingletonTargetSource(bean)); this.proxyTypes.put(cacheKey, proxy.getClass()); return proxy; } this.advisedBeans.put(cacheKey, Boolean.FALSE); return bean; }

先從緩存中拿,沒有則調用wrapIfNecessary方法創建。在這個方法里面主要看兩個地方:getAdvicesAndAdvisorsForBean和createProxy。簡單一句話概括就是先掃描后創建,問題是掃描什么呢?你可以先結合上面的概念思考下,換你會怎么做。進入到子類AbstractAdvisorAutoProxyCreator的getAdvicesAndAdvisorsForBean方法中:

protected Object[] getAdvicesAndAdvisorsForBean( Class<?> beanClass, String beanName, @Nullable TargetSource targetSource) { //找到合格的切面 List<Advisor> advisors = findEligibleAdvisors(beanClass, beanName); if (advisors.isEmpty()) { return DO_NOT_PROXY; } return advisors.toArray(); } protected List<Advisor> findEligibleAdvisors(Class<?> beanClass, String beanName) { //找到候選的切面,其實就是一個尋找有@Aspectj注解的過程,把工程中所有有這個注解的類封裝成Advisor返回 List<Advisor> candidateAdvisors = findCandidateAdvisors(); //判斷候選的切面是否作用在當前beanClass上面,就是一個匹配過程。現在就是一個匹配 List<Advisor> eligibleAdvisors = findAdvisorsThatCanApply(candidateAdvisors, beanClass, beanName); extendAdvisors(eligibleAdvisors); if (!eligibleAdvisors.isEmpty()) { //對有@Order@Priority進行排序 eligibleAdvisors = sortAdvisors(eligibleAdvisors); } return eligibleAdvisors; }

在findEligibleAdvisors方法中可以看到有兩個步驟,第一先找到所有的切面,即掃描所有帶有@Aspect注解的類,并將其中的切點(表達式)和增強封裝為切面,掃描完成后,自然是要判斷哪些切面能夠連接到當前Bean實例上。下面一步步來分析,首先是掃描過程,進入到AnnotationAwareAspectJAutoProxyCreator類中:

protected List<Advisor> findCandidateAdvisors() { // 先通過父類AbstractAdvisorAutoProxyCreator掃描,這里不重要 List<Advisor> advisors = super.findCandidateAdvisors(); // 主要看這里 if (this.aspectJAdvisorsBuilder != null) { advisors.addAll(this.aspectJAdvisorsBuilder.buildAspectJAdvisors()); } return advisors; }

這里委托給了BeanFactoryAspectJAdvisorsBuilderAdapter類,并調用其父類的buildAspectJAdvisors方法創建切面對象:

public List<Advisor> buildAspectJAdvisors() { List<String> aspectNames = this.aspectBeanNames; if (aspectNames == null) { synchronized (this) { aspectNames = this.aspectBeanNames; if (aspectNames == null) { List<Advisor> advisors = new ArrayList<>(); aspectNames = new ArrayList<>(); //獲取spring容器中的所有bean的名稱BeanName String[] beanNames = BeanFactoryUtils.beanNamesForTypeIncludingAncestors( this.beanFactory, Object.class, true, false); for (String beanName : beanNames) { if (!isEligibleBean(beanName)) { continue; } Class<?> beanType = this.beanFactory.getType(beanName); if (beanType == null) { continue; } //判斷類上是否有@Aspect注解 if (this.advisorFactory.isAspect(beanType)) { aspectNames.add(beanName); AspectMetadata amd = new AspectMetadata(beanType, beanName); if (amd.getAjType().getPerClause().getKind() == PerClauseKind.SINGLETON) { // 當@Aspect的value屬性為''時才會進入到這里 // 創建獲取有@Aspect注解類的實例工廠,負責獲取有@Aspect注解類的實例 MetadataAwareAspectInstanceFactory factory = new BeanFactoryAspectInstanceFactory(this.beanFactory, beanName); //創建切面advisor對象 List<Advisor> classAdvisors = this.advisorFactory.getAdvisors(factory); if (this.beanFactory.isSingleton(beanName)) { this.advisorsCache.put(beanName, classAdvisors); } else { this.aspectFactoryCache.put(beanName, factory); } advisors.addAll(classAdvisors); } else { MetadataAwareAspectInstanceFactory factory = new PrototypeAspectInstanceFactory(this.beanFactory, beanName); this.aspectFactoryCache.put(beanName, factory); advisors.addAll(this.advisorFactory.getAdvisors(factory)); } } } this.aspectBeanNames = aspectNames; return advisors; } } } return advisors; }

這個方法里面首先從IOC中拿到所有Bean的名稱,并循環判斷該類上是否帶有@Aspect注解,如果有則將BeanName和Bean的Class類型封裝到BeanFactoryAspectInstanceFactory中,并調用ReflectiveAspectJAdvisorFactory.getAdvisors創建切面對象:

public List<Advisor> getAdvisors(MetadataAwareAspectInstanceFactory aspectInstanceFactory) { //從工廠中獲取有@Aspect注解的類Class Class<?> aspectClass = aspectInstanceFactory.getAspectMetadata().getAspectClass(); //從工廠中獲取有@Aspect注解的類的名稱 String aspectName = aspectInstanceFactory.getAspectMetadata().getAspectName(); validate(aspectClass); // 創建工廠的裝飾類,獲取實例只會獲取一次 MetadataAwareAspectInstanceFactory lazySingletonAspectInstanceFactory = new LazySingletonAspectInstanceFactoryDecorator(aspectInstanceFactory); List<Advisor> advisors = new ArrayList<>(); //這里循環沒有@Pointcut注解的方法 for (Method method : getAdvisorMethods(aspectClass)) { //非常重要重點看看 Advisor advisor = getAdvisor(method, lazySingletonAspectInstanceFactory, advisors.size(), aspectName); if (advisor != null) { advisors.add(advisor); } } if (!advisors.isEmpty() && lazySingletonAspectInstanceFactory.getAspectMetadata().isLazilyInstantiated()) { Advisor instantiationAdvisor = new SyntheticInstantiationAdvisor(lazySingletonAspectInstanceFactory); advisors.add(0, instantiationAdvisor); } //判斷屬性上是否有引介注解,這里可以不看 for (Field field : aspectClass.getDeclaredFields()) { //判斷屬性上是否有DeclareParents注解,如果有返回切面 Advisor advisor = getDeclareParentsAdvisor(field); if (advisor != null) { advisors.add(advisor); } } return advisors; } private List<Method> getAdvisorMethods(Class<?> aspectClass) { final List<Method> methods = new ArrayList<>(); ReflectionUtils.doWithMethods(aspectClass, method -> { // Exclude pointcuts if (AnnotationUtils.getAnnotation(method, Pointcut.class) == null) { methods.add(method); } }); methods.sort(METHOD_COMPARATOR); return methods; }

根據Aspect的Class拿到所有不帶@Pointcut注解的方法對象(為什么是不帶@Pointcut注解的方法?仔細想想不難理解),另外要注意這里對method進行了排序,看看這個METHOD_COMPARATOR比較器:

private static final Comparator<Method> METHOD_COMPARATOR; static { Comparator<Method> adviceKindComparator = new ConvertingComparator<>( new InstanceComparator<>( Around.class, Before.class, After.class, AfterReturning.class, AfterThrowing.class), (Converter<Method, Annotation>) method -> { AspectJAnnotation<?> annotation = AbstractAspectJAdvisorFactory.findAspectJAnnotationOnMethod(method); return (annotation != null ? annotation.getAnnotation() : null); }); Comparator<Method> methodNameComparator = new ConvertingComparator<>(Method::getName); METHOD_COMPARATOR = adviceKindComparator.thenComparing(methodNameComparator); }

關注InstanceComparator構造函數參數,記住它們的順序,這就是AOP鏈式調用中同一個@Aspect類中Advice的執行順序。接著往下看,在getAdvisors方法中循環獲取到的methods,分別調用getAdvisor方法,也就是根據方法逐個去創建切面:

public Advisor getAdvisor(Method candidateAdviceMethod, MetadataAwareAspectInstanceFactory aspectInstanceFactory, int declarationOrderInAspect, String aspectName) { validate(aspectInstanceFactory.getAspectMetadata().getAspectClass()); //獲取pointCut對象,最重要的是從注解中獲取表達式 AspectJExpressionPointcut expressionPointcut = getPointcut( candidateAdviceMethod, aspectInstanceFactory.getAspectMetadata().getAspectClass()); if (expressionPointcut == null) { return null; } //創建Advisor切面類,這才是真正的切面類,一個切面類里面肯定要有1、pointCut 2、advice //這里pointCut是expressionPointcut, advice 增強方法是 candidateAdviceMethod return new InstantiationModelAwarePointcutAdvisorImpl(expressionPointcut, candidateAdviceMethod, this, aspectInstanceFactory, declarationOrderInAspect, aspectName); } private static final Class<?>[] ASPECTJ_ANNOTATION_CLASSES = new Class<?>[] { Pointcut.class, Around.class, Before.class, After.class, AfterReturning.class, AfterThrowing.class}; private AspectJExpressionPointcut getPointcut(Method candidateAdviceMethod, Class<?> candidateAspectClass) { //從候選的增強方法里面 candidateAdviceMethod 找有有注解 //Pointcut.class, Around.class, Before.class, After.class, AfterReturning.class, AfterThrowing.class //并把注解信息封裝成AspectJAnnotation對象 AspectJAnnotation<?> aspectJAnnotation = AbstractAspectJAdvisorFactory.findAspectJAnnotationOnMethod(candidateAdviceMethod); if (aspectJAnnotation == null) { return null; } //創建一個PointCut類,并且把前面從注解里面解析的表達式設置進去 AspectJExpressionPointcut ajexp = new AspectJExpressionPointcut(candidateAspectClass, new String[0], new Class<?>[0]); ajexp.setExpression(aspectJAnnotation.getPointcutExpression()); if (this.beanFactory != null) { ajexp.setBeanFactory(this.beanFactory); } return ajexp; }

之前就說過切面的定義,是切點和增強的組合,所以這里首先通過getPointcut獲取到注解對象,然后new了一個Pointcut對象,并將表達式設置進去。然后在getAdvisor方法中最后new了一個InstantiationModelAwarePointcutAdvisorImpl對象:

public InstantiationModelAwarePointcutAdvisorImpl(AspectJExpressionPointcut declaredPointcut, Method aspectJAdviceMethod, AspectJAdvisorFactory aspectJAdvisorFactory, MetadataAwareAspectInstanceFactory aspectInstanceFactory, int declarationOrder, String aspectName) { this.declaredPointcut = declaredPointcut; this.declaringClass = aspectJAdviceMethod.getDeclaringClass(); this.methodName = aspectJAdviceMethod.getName(); this.parameterTypes = aspectJAdviceMethod.getParameterTypes(); this.aspectJAdviceMethod = aspectJAdviceMethod; this.aspectJAdvisorFactory = aspectJAdvisorFactory; this.aspectInstanceFactory = aspectInstanceFactory; this.declarationOrder = declarationOrder; this.aspectName = aspectName; if (aspectInstanceFactory.getAspectMetadata().isLazilyInstantiated()) { // Static part of the pointcut is a lazy type. Pointcut preInstantiationPointcut = Pointcuts.union( aspectInstanceFactory.getAspectMetadata().getPerClausePointcut(), this.declaredPointcut); // Make it dynamic: must mutate from pre-instantiation to post-instantiation state. // If it’s not a dynamic pointcut, it may be optimized out // by the Spring AOP infrastructure after the first evaluation. this.pointcut = new PerTargetInstantiationModelPointcut( this.declaredPointcut, preInstantiationPointcut, aspectInstanceFactory); this.lazy = true; } else { // A singleton aspect. this.pointcut = this.declaredPointcut; this.lazy = false; //這個方法重點看看,創建advice對象 this.instantiatedAdvice = instantiateAdvice(this.declaredPointcut); } }

這個就是我們的切面類,在其構造方法的最后通過instantiateAdvice創建了Advice對象。注意這里傳進來的declarationOrder參數,它就是循環method時的序號,其作用就是賦值給這里的declarationOrder屬性以及Advice的declarationOrder屬性,在后面排序時就會通過這個序號來比較,因此Advice的執行順序是固定的,至于為什么要固定,后面分析完AOP鏈式調用過程自然就明白了。

public Advice getAdvice(Method candidateAdviceMethod, AspectJExpressionPointcut expressionPointcut, MetadataAwareAspectInstanceFactory aspectInstanceFactory, int declarationOrder, String aspectName) { //獲取有@Aspect注解的類 Class<?> candidateAspectClass = aspectInstanceFactory.getAspectMetadata().getAspectClass(); validate(candidateAspectClass); //找到candidateAdviceMethod方法上面的注解,并且包裝成AspectJAnnotation對象,這個對象中就有注解類型 AspectJAnnotation<?> aspectJAnnotation = AbstractAspectJAdvisorFactory.findAspectJAnnotationOnMethod(candidateAdviceMethod); if (aspectJAnnotation == null) { return null; } AbstractAspectJAdvice springAdvice; //根據不同的注解類型創建不同的advice類實例 switch (aspectJAnnotation.getAnnotationType()) { case AtPointcut: if (logger.isDebugEnabled()) { logger.debug('Processing pointcut ’' + candidateAdviceMethod.getName() + '’'); } return null; case AtAround: //實現了MethodInterceptor接口 springAdvice = new AspectJAroundAdvice( candidateAdviceMethod, expressionPointcut, aspectInstanceFactory); break; case AtBefore: //實現了MethodBeforeAdvice接口,沒有實現MethodInterceptor接口 springAdvice = new AspectJMethodBeforeAdvice( candidateAdviceMethod, expressionPointcut, aspectInstanceFactory); break; case AtAfter: //實現了MethodInterceptor接口 springAdvice = new AspectJAfterAdvice( candidateAdviceMethod, expressionPointcut, aspectInstanceFactory); break; case AtAfterReturning: //實現了AfterReturningAdvice接口,沒有實現MethodInterceptor接口 springAdvice = new AspectJAfterReturningAdvice( candidateAdviceMethod, expressionPointcut, aspectInstanceFactory); AfterReturning afterReturningAnnotation = (AfterReturning) aspectJAnnotation.getAnnotation(); if (StringUtils.hasText(afterReturningAnnotation.returning())) { springAdvice.setReturningName(afterReturningAnnotation.returning()); } break; case AtAfterThrowing: //實現了MethodInterceptor接口 springAdvice = new AspectJAfterThrowingAdvice( candidateAdviceMethod, expressionPointcut, aspectInstanceFactory); AfterThrowing afterThrowingAnnotation = (AfterThrowing) aspectJAnnotation.getAnnotation(); if (StringUtils.hasText(afterThrowingAnnotation.throwing())) { springAdvice.setThrowingName(afterThrowingAnnotation.throwing()); } break; default: throw new UnsupportedOperationException( 'Unsupported advice type on method: ' + candidateAdviceMethod); } // Now to configure the advice... springAdvice.setAspectName(aspectName); springAdvice.setDeclarationOrder(declarationOrder); String[] argNames = this.parameterNameDiscoverer.getParameterNames(candidateAdviceMethod); if (argNames != null) { springAdvice.setArgumentNamesFromStringArray(argNames); } //計算argNames和類型的對應關系 springAdvice.calculateArgumentBindings(); return springAdvice; }

這里邏輯很清晰,就是拿到方法上的注解類型,根據類型創建不同的增強Advice對象:AspectJAroundAdvice、AspectJMethodBeforeAdvice、AspectJAfterAdvice、AspectJAfterReturningAdvice、AspectJAfterThrowingAdvice。完成之后通過calculateArgumentBindings方法進行參數綁定,感興趣的可自行研究。這里主要看看幾個Advice的繼承體系:

這一次搞懂Spring代理創建及AOP鏈式調用過程操作

可以看到有兩個Advice是沒有實現MethodInterceptor接口的:AspectJMethodBeforeAdvice和AspectJAfterReturningAdvice。而MethodInterceptor有一個invoke方法,這個方法就是鏈式調用的核心方法,但那兩個沒有實現該方法的Advice怎么處理呢?稍后會分析。

到這里切面對象就創建完成了,接下來就是判斷當前創建的Bean實例是否和這些切面匹配以及對切面排序。匹配過程比較復雜,對理解主流程也沒什么幫助,所以這里就不展開分析,感興趣的自行分析(AbstractAdvisorAutoProxyCreator.findAdvisorsThatCanApply())。

下面看看排序的過程,回到AbstractAdvisorAutoProxyCreator.findEligibleAdvisors方法:

protected List<Advisor> findEligibleAdvisors(Class<?> beanClass, String beanName) { //找到候選的切面,其實就是一個尋找有@Aspectj注解的過程,把工程中所有有這個注解的類封裝成Advisor返回 List<Advisor> candidateAdvisors = findCandidateAdvisors(); //判斷候選的切面是否作用在當前beanClass上面,就是一個匹配過程。。現在就是一個匹配 List<Advisor> eligibleAdvisors = findAdvisorsThatCanApply(candidateAdvisors, beanClass, beanName); extendAdvisors(eligibleAdvisors); if (!eligibleAdvisors.isEmpty()) { //對有@Order@Priority進行排序 eligibleAdvisors = sortAdvisors(eligibleAdvisors); } return eligibleAdvisors; }

sortAdvisors方法就是排序,但這個方法有兩個實現:當前類AbstractAdvisorAutoProxyCreator和子類AspectJAwareAdvisorAutoProxyCreator,應該走哪個呢?

這一次搞懂Spring代理創建及AOP鏈式調用過程操作

通過類圖我們可以肯定是進入的AspectJAwareAdvisorAutoProxyCreator類,因為AnnotationAwareAspectJAutoProxyCreator的父類是它。

protected List<Advisor> sortAdvisors(List<Advisor> advisors) { List<PartiallyComparableAdvisorHolder> partiallyComparableAdvisors = new ArrayList<>(advisors.size()); for (Advisor element : advisors) { partiallyComparableAdvisors.add( new PartiallyComparableAdvisorHolder(element, DEFAULT_PRECEDENCE_COMPARATOR)); } List<PartiallyComparableAdvisorHolder> sorted = PartialOrder.sort(partiallyComparableAdvisors); if (sorted != null) { List<Advisor> result = new ArrayList<>(advisors.size()); for (PartiallyComparableAdvisorHolder pcAdvisor : sorted) { result.add(pcAdvisor.getAdvisor()); } return result; } else { return super.sortAdvisors(advisors); } }

這里排序主要是委托給PartialOrder進行的,而在此之前將所有的切面都封裝成了PartiallyComparableAdvisorHolder對象,注意傳入的DEFAULT_PRECEDENCE_COMPARATOR參數,這個就是比較器對象:

private static final Comparator<Advisor> DEFAULT_PRECEDENCE_COMPARATOR = new AspectJPrecedenceComparator();

所以我們直接看這個比較器的compare方法:

public int compare(Advisor o1, Advisor o2) { int advisorPrecedence = this.advisorComparator.compare(o1, o2); if (advisorPrecedence == SAME_PRECEDENCE && declaredInSameAspect(o1, o2)) { advisorPrecedence = comparePrecedenceWithinAspect(o1, o2); } return advisorPrecedence; } private final Comparator<? super Advisor> advisorComparator; public AspectJPrecedenceComparator() { this.advisorComparator = AnnotationAwareOrderComparator.INSTANCE; }

第一步先通過AnnotationAwareOrderComparator去比較,點進去看可以發現是對實現了PriorityOrdered和Ordered接口以及標記了Priority和Order注解的非同一個@Aspect類中的切面進行排序。這個和之前分析BeanFacotryPostProcessor類是一樣的原理。而對同一個@Aspect類中的切面排序主要是comparePrecedenceWithinAspect方法:

private int comparePrecedenceWithinAspect(Advisor advisor1, Advisor advisor2) { boolean oneOrOtherIsAfterAdvice = (AspectJAopUtils.isAfterAdvice(advisor1) || AspectJAopUtils.isAfterAdvice(advisor2)); int adviceDeclarationOrderDelta = getAspectDeclarationOrder(advisor1) - getAspectDeclarationOrder(advisor2); if (oneOrOtherIsAfterAdvice) { // the advice declared last has higher precedence if (adviceDeclarationOrderDelta < 0) { // advice1 was declared before advice2 // so advice1 has lower precedence return LOWER_PRECEDENCE; } else if (adviceDeclarationOrderDelta == 0) { return SAME_PRECEDENCE; } else { return HIGHER_PRECEDENCE; } } else { // the advice declared first has higher precedence if (adviceDeclarationOrderDelta < 0) { // advice1 was declared before advice2 // so advice1 has higher precedence return HIGHER_PRECEDENCE; } else if (adviceDeclarationOrderDelta == 0) { return SAME_PRECEDENCE; } else { return LOWER_PRECEDENCE; } } } private int getAspectDeclarationOrder(Advisor anAdvisor) { AspectJPrecedenceInformation precedenceInfo = AspectJAopUtils.getAspectJPrecedenceInformationFor(anAdvisor); if (precedenceInfo != null) { return precedenceInfo.getDeclarationOrder(); } else { return 0; } }

這里就是通過precedenceInfo.getDeclarationOrder拿到在創建InstantiationModelAwarePointcutAdvisorImpl對象時設置的declarationOrder屬性,這就驗證了之前的說法(實際上這里排序過程非常復雜,不是簡單的按照這個屬性進行排序)。

當上面的一切都進行完成后,就該創建代理對象了,回到AbstractAutoProxyCreator.wrapIfNecessary,看關鍵部分代碼:

//如果有切面,則生成該bean的代理 if (specificInterceptors != DO_NOT_PROXY) { this.advisedBeans.put(cacheKey, Boolean.TRUE); //把被代理對象bean實例封裝到SingletonTargetSource對象中 Object proxy = createProxy( bean.getClass(), beanName, specificInterceptors, new SingletonTargetSource(bean)); this.proxyTypes.put(cacheKey, proxy.getClass()); return proxy; }

注意這里將被代理對象封裝成了一個SingletonTargetSource對象,它是TargetSource的實現類。

protected Object createProxy(Class<?> beanClass, @Nullable String beanName, @Nullable Object[] specificInterceptors, TargetSource targetSource) { if (this.beanFactory instanceof ConfigurableListableBeanFactory) { AutoProxyUtils.exposeTargetClass((ConfigurableListableBeanFactory) this.beanFactory, beanName, beanClass); } //創建代理工廠 ProxyFactory proxyFactory = new ProxyFactory(); proxyFactory.copyFrom(this); if (!proxyFactory.isProxyTargetClass()) { if (shouldProxyTargetClass(beanClass, beanName)) { //proxyTargetClass 是否對類進行代理,而不是對接口進行代理,設置為true時,使用CGLib代理。 proxyFactory.setProxyTargetClass(true); } else { evaluateProxyInterfaces(beanClass, proxyFactory); } } //把advice類型的增強包裝成advisor切面 Advisor[] advisors = buildAdvisors(beanName, specificInterceptors); proxyFactory.addAdvisors(advisors); proxyFactory.setTargetSource(targetSource); customizeProxyFactory(proxyFactory); 用來控制代理工廠被配置后,是否還允許修改代理的配置,默認為false proxyFactory.setFrozen(this.freezeProxy); if (advisorsPreFiltered()) { proxyFactory.setPreFiltered(true); } //獲取代理實例 return proxyFactory.getProxy(getProxyClassLoader()); }

這里通過ProxyFactory對象去創建代理實例,這是工廠模式的體現,但在創建代理對象之前還有幾個準備動作:需要判斷是JDK代理還是CGLIB代理以及通過buildAdvisors方法將擴展的Advice封裝成Advisor切面。準備完成則通過getProxy創建代理對象:

public Object getProxy(@Nullable ClassLoader classLoader) { //根據目標對象是否有接口來判斷采用什么代理方式,cglib代理還是jdk動態代理 return createAopProxy().getProxy(classLoader); } protected final synchronized AopProxy createAopProxy() { if (!this.active) { activate(); } return getAopProxyFactory().createAopProxy(this); } public AopProxy createAopProxy(AdvisedSupport config) throws AopConfigException { if (config.isOptimize() || config.isProxyTargetClass() || hasNoUserSuppliedProxyInterfaces(config)) { Class<?> targetClass = config.getTargetClass(); if (targetClass == null) { throw new AopConfigException('TargetSource cannot determine target class: ' + 'Either an interface or a target is required for proxy creation.'); } if (targetClass.isInterface() || Proxy.isProxyClass(targetClass)) { return new JdkDynamicAopProxy(config); } return new ObjenesisCglibAopProxy(config); } else { return new JdkDynamicAopProxy(config); } }

首先通過配置拿到對應的代理類:ObjenesisCglibAopProxy和JdkDynamicAopProxy,然后再通過getProxy創建Bean的代理,這里以JdkDynamicAopProxy為例:

public Object getProxy(@Nullable ClassLoader classLoader) { //advised是代理工廠對象 Class<?>[] proxiedInterfaces = AopProxyUtils.completeProxiedInterfaces(this.advised, true); findDefinedEqualsAndHashCodeMethods(proxiedInterfaces); return Proxy.newProxyInstance(classLoader, proxiedInterfaces, this); }

這里的代碼你應該不陌生了,就是JDK的原生API,newProxyInstance方法傳入的InvocationHandler對象是this,因此,最終AOP代理的調用就是從該類中的invoke方法開始。至此,代理對象的創建就完成了,下面來看下整個過程的時序圖:

這一次搞懂Spring代理創建及AOP鏈式調用過程操作

小結

代理對象的創建過程整體來說并不復雜,首先找到所有帶有@Aspect注解的類,并獲取其中沒有@Pointcut注解的方法,循環創建切面,而創建切面需要切點和增強兩個元素,其中切點可簡單理解為我們寫的表達式,增強則是根據@Before、@Around、@After等注解創建的對應的Advice類。切面創建好后則需要循環判斷哪些切面能對當前的Bean實例的方法進行增強并排序,最后通過ProxyFactory創建代理對象。

AOP鏈式調用

熟悉JDK動態代理的都知道通過代理對象調用方法時,會進入到InvocationHandler對象的invoke方法,所以我們直接從JdkDynamicAopProxy的這個方法開始:

public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { MethodInvocation invocation; Object oldProxy = null; boolean setProxyContext = false; //從代理工廠中拿到TargetSource對象,該對象包裝了被代理實例bean TargetSource targetSource = this.advised.targetSource; Object target = null; try { //被代理對象的equals方法和hashCode方法是不能被代理的,不會走切面 ....... Object retVal; // 可以從當前線程中拿到代理對象 if (this.advised.exposeProxy) { // Make invocation available if necessary. oldProxy = AopContext.setCurrentProxy(proxy); setProxyContext = true; } //這個target就是被代理實例 target = targetSource.getTarget(); Class<?> targetClass = (target != null ? target.getClass() : null); //從代理工廠中拿過濾器鏈 Object是一個MethodInterceptor類型的對象,其實就是一個advice對象 List<Object> chain = this.advised.getInterceptorsAndDynamicInterceptionAdvice(method, targetClass); //如果該方法沒有執行鏈,則說明這個方法不需要被攔截,則直接反射調用 if (chain.isEmpty()) { Object[] argsToUse = AopProxyUtils.adaptArgumentsIfNecessary(method, args); retVal = AopUtils.invokeJoinpointUsingReflection(target, method, argsToUse); } else { invocation = new ReflectiveMethodInvocation(proxy, target, method, args, targetClass, chain); retVal = invocation.proceed(); } // Massage return value if necessary. Class<?> returnType = method.getReturnType(); if (retVal != null && retVal == target && returnType != Object.class && returnType.isInstance(proxy) && !RawTargetAccess.class.isAssignableFrom(method.getDeclaringClass())) { retVal = proxy; } return retVal; } finally { if (target != null && !targetSource.isStatic()) { // Must have come from TargetSource. targetSource.releaseTarget(target); } if (setProxyContext) { // Restore old proxy. AopContext.setCurrentProxy(oldProxy); } } }

這段代碼比較長,我刪掉了不關鍵的地方。首先來看this.advised.exposeProxy這個屬性,這在@EnableAspectJAutoProxy注解中可以配置,當為true時,會將該代理對象設置到當前線程的ThreadLocal對象中,這樣就可以通過AopContext.currentProxy拿到代理對象。這個有什么用呢?我相信有經驗的Java開發都遇到過這樣一個BUG,在Service實現類中調用本類中的另一個方法時,事務不會生效,這是因為直接通過this調用就不會調用到代理對象的方法,而是原對象的,所以事務切面就沒有生效。因此這種情況下就可以從當前線程的ThreadLocal對象拿到代理對象,不過實際上直接使用@Autowired注入自己本身也可以拿到代理對象。

接下來就是通過getInterceptorsAndDynamicInterceptionAdvice拿到執行鏈,看看具體做了哪些事情:

public List<Object> getInterceptorsAndDynamicInterceptionAdvice( Advised config, Method method, @Nullable Class<?> targetClass) { AdvisorAdapterRegistry registry = GlobalAdvisorAdapterRegistry.getInstance(); //從代理工廠中獲得該被代理類的所有切面advisor,config就是代理工廠對象 Advisor[] advisors = config.getAdvisors(); List<Object> interceptorList = new ArrayList<>(advisors.length); Class<?> actualClass = (targetClass != null ? targetClass : method.getDeclaringClass()); Boolean hasIntroductions = null; for (Advisor advisor : advisors) { //大部分走這里 if (advisor instanceof PointcutAdvisor) { // Add it conditionally. PointcutAdvisor pointcutAdvisor = (PointcutAdvisor) advisor; //如果切面的pointCut和被代理對象是匹配的,說明是切面要攔截的對象 if (config.isPreFiltered() || pointcutAdvisor.getPointcut().getClassFilter().matches(actualClass)) { MethodMatcher mm = pointcutAdvisor.getPointcut().getMethodMatcher(); boolean match; if (mm instanceof IntroductionAwareMethodMatcher) { if (hasIntroductions == null) { hasIntroductions = hasMatchingIntroductions(advisors, actualClass); } match = ((IntroductionAwareMethodMatcher) mm).matches(method, actualClass, hasIntroductions); } else { //接下來判斷方法是否是切面pointcut需要攔截的方法 match = mm.matches(method, actualClass); } //如果類和方法都匹配 if (match) { //獲取到切面advisor中的advice,并且包裝成MethodInterceptor類型的對象 MethodInterceptor[] interceptors = registry.getInterceptors(advisor); if (mm.isRuntime()) { for (MethodInterceptor interceptor : interceptors) { interceptorList.add(new InterceptorAndDynamicMethodMatcher(interceptor, mm)); } } else { interceptorList.addAll(Arrays.asList(interceptors)); } } } } //如果是引介切面 else if (advisor instanceof IntroductionAdvisor) { IntroductionAdvisor ia = (IntroductionAdvisor) advisor; if (config.isPreFiltered() || ia.getClassFilter().matches(actualClass)) { Interceptor[] interceptors = registry.getInterceptors(advisor); interceptorList.addAll(Arrays.asList(interceptors)); } } else { Interceptor[] interceptors = registry.getInterceptors(advisor); interceptorList.addAll(Arrays.asList(interceptors)); } } return interceptorList; }

這也是個長方法,看關鍵的部分,因為之前我們創建的基本上都是InstantiationModelAwarePointcutAdvisorImpl對象,該類是PointcutAdvisor的實現類,所以會進入第一個if判斷里,這里首先進行匹配,看切點和當前對象以及該對象的哪些方法匹配,如果能匹配上,則調用getInterceptors獲取執行鏈:

private final List<AdvisorAdapter> adapters = new ArrayList<>(3); public DefaultAdvisorAdapterRegistry() { registerAdvisorAdapter(new MethodBeforeAdviceAdapter()); registerAdvisorAdapter(new AfterReturningAdviceAdapter()); registerAdvisorAdapter(new ThrowsAdviceAdapter()); } public MethodInterceptor[] getInterceptors(Advisor advisor) throws UnknownAdviceTypeException { List<MethodInterceptor> interceptors = new ArrayList<>(3); Advice advice = advisor.getAdvice(); //如果是MethodInterceptor類型的,如:AspectJAroundAdvice //AspectJAfterAdvice //AspectJAfterThrowingAdvice if (advice instanceof MethodInterceptor) { interceptors.add((MethodInterceptor) advice); } //處理 AspectJMethodBeforeAdvice AspectJAfterReturningAdvice for (AdvisorAdapter adapter : this.adapters) { if (adapter.supportsAdvice(advice)) { interceptors.add(adapter.getInterceptor(advisor)); } } if (interceptors.isEmpty()) { throw new UnknownAdviceTypeException(advisor.getAdvice()); } return interceptors.toArray(new MethodInterceptor[0]); }

這里我們可以看到如果是MethodInterceptor的實現類,則直接添加到鏈中,如果不是,則需要通過適配器去包裝后添加,剛好這里有MethodBeforeAdviceAdapter和AfterReturningAdviceAdapter兩個適配器對應上文兩個沒有實現MethodInterceptor接口的類。最后將Interceptors返回。

if (chain.isEmpty()) { Object[] argsToUse = AopProxyUtils.adaptArgumentsIfNecessary(method, args); retVal = AopUtils.invokeJoinpointUsingReflection(target, method, argsToUse);}else { // We need to create a method invocation... invocation = new ReflectiveMethodInvocation(proxy, target, method, args, targetClass, chain); // Proceed to the joinpoint through the interceptor chain. retVal = invocation.proceed();}

返回到invoke方法后,如果執行鏈為空,說明該方法不需要被增強,所以直接反射調用原對象的方法(注意傳入的是TargetSource封裝的被代理對象);反之,則通過ReflectiveMethodInvocation類進行鏈式調用,關鍵方法就是proceed:

private int currentInterceptorIndex = -1; public Object proceed() throws Throwable { //如果執行鏈中的advice全部執行完,則直接調用joinPoint方法,就是被代理方法 if (this.currentInterceptorIndex == this.interceptorsAndDynamicMethodMatchers.size() - 1) { return invokeJoinpoint(); } Object interceptorOrInterceptionAdvice = this.interceptorsAndDynamicMethodMatchers.get(++this.currentInterceptorIndex); if (interceptorOrInterceptionAdvice instanceof InterceptorAndDynamicMethodMatcher) { InterceptorAndDynamicMethodMatcher dm = (InterceptorAndDynamicMethodMatcher) interceptorOrInterceptionAdvice; Class<?> targetClass = (this.targetClass != null ? this.targetClass : this.method.getDeclaringClass()); if (dm.methodMatcher.matches(this.method, targetClass, this.arguments)) { return dm.interceptor.invoke(this); } else { return proceed(); } } else { //調用MethodInterceptor中的invoke方法 return ((MethodInterceptor) interceptorOrInterceptionAdvice).invoke(this); } }

這個方法的核心就在兩個地方:invokeJoinpoint和interceptorOrInterceptionAdvice.invoke(this)。當增強方法調用完后就會通過前者調用到被代理的方法,否則則是依次調用Interceptor的invoke方法。下面就分別看看每個Interceptor是怎么實現的。

AspectJAroundAdvice public Object invoke(MethodInvocation mi) throws Throwable { if (!(mi instanceof ProxyMethodInvocation)) { throw new IllegalStateException('MethodInvocation is not a Spring ProxyMethodInvocation: ' + mi); } ProxyMethodInvocation pmi = (ProxyMethodInvocation) mi; ProceedingJoinPoint pjp = lazyGetProceedingJoinPoint(pmi); JoinPointMatch jpm = getJoinPointMatch(pmi); return invokeAdviceMethod(pjp, jpm, null, null); }

MethodBeforeAdviceInterceptor -> AspectJMethodBeforeAdvice public Object invoke(MethodInvocation mi) throws Throwable { this.advice.before(mi.getMethod(), mi.getArguments(), mi.getThis()); return mi.proceed(); } public void before(Method method, Object[] args, @Nullable Object target) throws Throwable { invokeAdviceMethod(getJoinPointMatch(), null, null); }

AspectJAfterAdvice public Object invoke(MethodInvocation mi) throws Throwable { try { return mi.proceed(); } finally { invokeAdviceMethod(getJoinPointMatch(), null, null); } }

AfterReturningAdviceInterceptor -> AspectJAfterReturningAdvice public Object invoke(MethodInvocation mi) throws Throwable { Object retVal = mi.proceed(); this.advice.afterReturning(retVal, mi.getMethod(), mi.getArguments(), mi.getThis()); return retVal; } public void afterReturning(@Nullable Object returnValue, Method method, Object[] args, @Nullable Object target) throws Throwable { if (shouldInvokeOnReturnValueOf(method, returnValue)) { invokeAdviceMethod(getJoinPointMatch(), returnValue, null); } }

AspectJAfterThrowingAdvice public Object invoke(MethodInvocation mi) throws Throwable { try { return mi.proceed(); } catch (Throwable ex) { if (shouldInvokeOnThrowing(ex)) { invokeAdviceMethod(getJoinPointMatch(), null, ex); } throw ex; } }

這里的調用順序是怎樣的呢?其核心就是通過proceed方法控制流程,每執行完一個Advice就會回到proceed方法中調用下一個Advice。可以思考一下,怎么才能讓調用結果滿足如下圖的執行順序。

這一次搞懂Spring代理創建及AOP鏈式調用過程操作

以上就是AOP的鏈式調用過程,但是這只是只有一個切面類的情況,如果有多個@Aspect類呢,這個調用過程又是怎樣的?其核心思想和“棧”一樣,就是“先進后出,后進先出”。

AOP擴展知識

一、自定義全局攔截器Interceptor

在上文創建代理對象的時候有這樣一個方法:

protected Advisor[] buildAdvisors(@Nullable String beanName, @Nullable Object[] specificInterceptors) { //自定義MethodInterceptor.拿到setInterceptorNames方法注入的Interceptor對象 Advisor[] commonInterceptors = resolveInterceptorNames(); List<Object> allInterceptors = new ArrayList<>(); if (specificInterceptors != null) { allInterceptors.addAll(Arrays.asList(specificInterceptors)); if (commonInterceptors.length > 0) { if (this.applyCommonInterceptorsFirst) { allInterceptors.addAll(0, Arrays.asList(commonInterceptors)); } else { allInterceptors.addAll(Arrays.asList(commonInterceptors)); } } } Advisor[] advisors = new Advisor[allInterceptors.size()]; for (int i = 0; i < allInterceptors.size(); i++) { //對自定義的advice要進行包裝,把advice包裝成advisor對象,切面對象 advisors[i] = this.advisorAdapterRegistry.wrap(allInterceptors.get(i)); } return advisors; }

這個方法的作用就在于我們可以擴展我們自己的Interceptor,首先通過resolveInterceptorNames方法獲取到通過setInterceptorNames方法設置的Interceptor,然后調用DefaultAdvisorAdapterRegistry.wrap方法將其包裝為DefaultPointcutAdvisor對象并返回:

public Advisor wrap(Object adviceObject) throws UnknownAdviceTypeException { if (adviceObject instanceof Advisor) { return (Advisor) adviceObject; } if (!(adviceObject instanceof Advice)) { throw new UnknownAdviceTypeException(adviceObject); } Advice advice = (Advice) adviceObject; if (advice instanceof MethodInterceptor) { return new DefaultPointcutAdvisor(advice); } for (AdvisorAdapter adapter : this.adapters) { if (adapter.supportsAdvice(advice)) { return new DefaultPointcutAdvisor(advice); } } throw new UnknownAdviceTypeException(advice); } public DefaultPointcutAdvisor(Advice advice) { this(Pointcut.TRUE, advice); }

需要注意DefaultPointcutAdvisor構造器里面傳入了一個Pointcut.TRUE,表示這種擴展的Interceptor是全局的攔截器。下面來看看如何使用:

public class MyMethodInterceptor implements MethodInterceptor { @Override public Object invoke(MethodInvocation invocation) throws Throwable { System.out.println('自定義攔截器'); return invocation.proceed(); }}

首先寫一個類實現MethodInterceptor 接口,在invoke方法中實現我們的攔截邏輯,然后通過下面的方式測試,只要UserService 有AOP攔截就會發現自定義的MyMethodInterceptor也生效了。

public void costomInterceptorTest() { AnnotationAwareAspectJAutoProxyCreator bean = applicationContext.getBean(AnnotationAwareAspectJAutoProxyCreator.class); bean.setInterceptorNames('myMethodInterceptor '); UserService userService = applicationContext.getBean(UserService.class); userService.queryUser('dark'); }

但是如果換個順序,像下面這樣:

public void costomInterceptorTest() { UserService userService = applicationContext.getBean(UserService.class); AnnotationAwareAspectJAutoProxyCreator bean = applicationContext.getBean(AnnotationAwareAspectJAutoProxyCreator.class); bean.setInterceptorNames('myMethodInterceptor '); userService.queryUser('dark'); }

這時自定義的全局攔截器就沒有作用了,這是為什么呢?因為當執行getBean的時候,如果有切面匹配就會通過ProxyFactory去創建代理對象,注意Interceptor是存到這個Factory對象中的,而這個對象和代理對象是一一對應的,因此調用getBean時,還沒有myMethodInterceptor這個對象,自定義攔截器就沒有效果了,也就是說要想自定義攔截器生效,就必須在代理對象生成之前注冊進去。

二、循環依賴三級緩存存在的必要性

在上一篇文章我分析了Spring是如何通過三級緩存來解決循環依賴的問題的,但你是否考慮過第三級緩存為什么要存在?我直接將bean存到二級不就行了么,為什么還要存一個ObjectFactory對象到第三級緩存中?一個是因為不是每個Bean都會出現循環依賴,所以三級緩存只存了一個工廠對象;二是我們在@Autowired對象時,想要注入的不一定是Bean本身,而是想要注入一個修改過后的對象,如代理對象。在AbstractAutowireCapableBeanFactory.getEarlyBeanReference方法中循環調用了SmartInstantiationAwareBeanPostProcessor.getEarlyBeanReference方法,AbstractAutoProxyCreator對象就實現了該方法:

public Object getEarlyBeanReference(Object bean, String beanName) { Object cacheKey = getCacheKey(bean.getClass(), beanName); if (!this.earlyProxyReferences.contains(cacheKey)) { this.earlyProxyReferences.add(cacheKey); } // 創建代理對象 return wrapIfNecessary(bean, beanName, cacheKey); }

因此,當我們想要對循壞依賴的Bean做出修改時,就可以像AOP這樣做。

三、如何在Bean創建之前提前創建代理對象

Spring的代理對象基本上都是在Bean實例化完成之后創建的,但在文章開始我就說過,Spring也提供了一個機會在創建Bean對象之前就創建代理對象,在AbstractAutowireCapableBeanFactory.resolveBeforeInstantiation方法中:

protected Object resolveBeforeInstantiation(String beanName, RootBeanDefinition mbd) { Object bean = null; if (!Boolean.FALSE.equals(mbd.beforeInstantiationResolved)) { // Make sure bean class is actually resolved at this point. if (!mbd.isSynthetic() && hasInstantiationAwareBeanPostProcessors()) { Class<?> targetType = determineTargetType(beanName, mbd); if (targetType != null) { bean = applyBeanPostProcessorsBeforeInstantiation(targetType, beanName); if (bean != null) { bean = applyBeanPostProcessorsAfterInitialization(bean, beanName); } } } mbd.beforeInstantiationResolved = (bean != null); } return bean; } protected Object applyBeanPostProcessorsBeforeInstantiation(Class<?> beanClass, String beanName) { for (BeanPostProcessor bp : getBeanPostProcessors()) { if (bp instanceof InstantiationAwareBeanPostProcessor) { InstantiationAwareBeanPostProcessor ibp = (InstantiationAwareBeanPostProcessor) bp; Object result = ibp.postProcessBeforeInstantiation(beanClass, beanName); if (result != null) { return result; } } } return null; }

主要是InstantiationAwareBeanPostProcessor.postProcessBeforeInstantiation方法中,這里又會進入到AbstractAutoProxyCreator類中:

public Object postProcessBeforeInstantiation(Class<?> beanClass, String beanName) { TargetSource targetSource = getCustomTargetSource(beanClass, beanName); if (targetSource != null) { if (StringUtils.hasLength(beanName)) { this.targetSourcedBeans.add(beanName); } Object[] specificInterceptors = getAdvicesAndAdvisorsForBean(beanClass, beanName, targetSource); Object proxy = createProxy(beanClass, beanName, specificInterceptors, targetSource); this.proxyTypes.put(cacheKey, proxy.getClass()); return proxy; } return null; } protected TargetSource getCustomTargetSource(Class<?> beanClass, String beanName) { // We can’t create fancy target sources for directly registered singletons. if (this.customTargetSourceCreators != null && this.beanFactory != null && this.beanFactory.containsBean(beanName)) { for (TargetSourceCreator tsc : this.customTargetSourceCreators) { TargetSource ts = tsc.getTargetSource(beanClass, beanName); if (ts != null) { return ts; } } } // No custom TargetSource found. return null; }

看到這里大致應該明白了,先是獲取到一個自定義的TargetSource對象,然后創建代理對象,所以我們首先需要自己實現一個TargetSource類,這里直接繼承一個抽象類,getTarget方法則返回原始對象:

public class MyTargetSource extends AbstractBeanFactoryBasedTargetSource { @Override public Object getTarget() throws Exception { return getBeanFactory().getBean(getTargetBeanName()); }}

但這還不夠,上面首先判斷了customTargetSourceCreators!=null,而這個屬性是個數組,可以通過下面這個方法設置進來:

public void setCustomTargetSourceCreators(TargetSourceCreator... targetSourceCreators) { this.customTargetSourceCreators = targetSourceCreators; }

所以我們還要實現一個TargetSourceCreator類,同樣繼承一個抽象類實現,并只對userServiceImpl對象進行攔截:

public class MyTargetSourceCreator extends AbstractBeanFactoryBasedTargetSourceCreator { @Override protected AbstractBeanFactoryBasedTargetSource createBeanFactoryBasedTargetSource(Class<?> beanClass, String beanName) { if (getBeanFactory() instanceof ConfigurableListableBeanFactory) { if(beanName.equalsIgnoreCase('userServiceImpl')) { return new MyTargetSource(); } } return null; }}

createBeanFactoryBasedTargetSource方法是在AbstractBeanFactoryBasedTargetSourceCreator.getTargetSource中調用的,而getTargetSource就是在上面getCustomTargetSource中調用的。以上工作做完后,還需要將其設置到AnnotationAwareAspectJAutoProxyCreator對象中,因此需要我們注入這個對象:

@Configurationpublic class TargetSourceCreatorBean { @Autowired private BeanFactory beanFactory; @Bean public AnnotationAwareAspectJAutoProxyCreator annotationAwareAspectJAutoProxyCreator() { AnnotationAwareAspectJAutoProxyCreator creator = new AnnotationAwareAspectJAutoProxyCreator(); MyTargetSourceCreator myTargetSourceCreator = new MyTargetSourceCreator(); myTargetSourceCreator.setBeanFactory(beanFactory); creator.setCustomTargetSourceCreators(myTargetSourceCreator); return creator; }}

這樣,當我們通過getBean獲取userServiceImpl的對象時,就會優先生成代理對象,然后在調用執行鏈的過程中再通過TargetSource.getTarget獲取到被代理對象。但是,為什么我們在getTarget方法中調用getBean就能拿到被代理對象呢?

繼續探究,通過斷點我發現從getTarget進入時,在resolveBeforeInstantiation方法中返回的bean就是null了,而getBeanPostProcessors方法返回的Processors中也沒有了AnnotationAwareAspectJAutoProxyCreator對象,也就是沒有進入到AbstractAutoProxyCreator.postProcessBeforeInstantiation方法中,所以不會再次獲取到代理對象,那AnnotationAwareAspectJAutoProxyCreator對象是在什么時候移除的呢?

帶著問題,我開始反推,發現在AbstractBeanFactoryBasedTargetSourceCreator類中有這樣一個方法buildInternalBeanFactory:

protected DefaultListableBeanFactory buildInternalBeanFactory(ConfigurableBeanFactory containingFactory) { DefaultListableBeanFactory internalBeanFactory = new DefaultListableBeanFactory(containingFactory); // Required so that all BeanPostProcessors, Scopes, etc become available. internalBeanFactory.copyConfigurationFrom(containingFactory); // Filter out BeanPostProcessors that are part of the AOP infrastructure, // since those are only meant to apply to beans defined in the original factory. internalBeanFactory.getBeanPostProcessors().removeIf(beanPostProcessor -> beanPostProcessor instanceof AopInfrastructureBean); return internalBeanFactory; }

在這里移除掉了所有AopInfrastructureBean的子類,而AnnotationAwareAspectJAutoProxyCreator就是其子類,那這個方法是在哪里調用的呢?繼續反推:

protected DefaultListableBeanFactory getInternalBeanFactoryForBean(String beanName) { synchronized (this.internalBeanFactories) { DefaultListableBeanFactory internalBeanFactory = this.internalBeanFactories.get(beanName); if (internalBeanFactory == null) { internalBeanFactory = buildInternalBeanFactory(this.beanFactory); this.internalBeanFactories.put(beanName, internalBeanFactory); } return internalBeanFactory; } } public final TargetSource getTargetSource(Class<?> beanClass, String beanName) { AbstractBeanFactoryBasedTargetSource targetSource = createBeanFactoryBasedTargetSource(beanClass, beanName); // 創建完targetSource后就移除掉AopInfrastructureBean類型的BeanPostProcessor對象,如AnnotationAwareAspectJAutoProxyCreator DefaultListableBeanFactory internalBeanFactory = getInternalBeanFactoryForBean(beanName); ...... return targetSource; }

至此,關于TargetSource接口擴展的原理就搞明白了。

總結

本篇篇幅比較長,主要搞明白Spring代理對象是如何創建的以及AOP鏈式調用過程,而后面的擴展則是對AOP以及Bean創建過程中一些疑惑的補充,可根據實際情況學習掌握。希望大家多多支持好吧啦網。

標簽: Spring
相關文章:
主站蜘蛛池模板: 大j8福利视频导航 | 成人免费视频xbxb入口 | 日韩在线观看一区 | 国产午夜精品久久久久免费视 | 国产手机视频在线 | 成人免费毛片男人用品 | 少妇裸体性生交免费 | 国产极品福利 | 久久久精品人妻无码专区不卡 | 2020最新无码福利视频 | 五月婷婷六月综合 | 亚洲色图欧美自拍 | 精品无码无人网站免费视频 | 中文字幕大香视频蕉免费 | 亚洲专区中文字幕 | 成人区人妻精品一熟女 | а√在线中文网新版地址在线 | 成人久久久精品国产乱码一区二区 | 24小时日本在线www免费的 | 色一情一区二区三区四区 | 少妇精品蜜桃偷拍高潮系列 | 国产精品ai换脸张天爱 | 男女无遮挡做爰猛烈黄文 | 亚洲中文字幕av无码专区 | jjzz在线 | 天堂网中文字幕 | 日本美女久久 | 性女次台湾三级 | av无码精品一区二区三区宅噜噜 | 五月天婷婷综合 | 国产成人在线网站 | 国产精品高潮呻吟久久av郑州 | 韩国久久久久久 | av有码在线 | se94se亚洲精品setu | www.婷婷亚洲基地 | 欧美日韩在线一区二区 | 香蕉影院在线观看 | 欧美日韩一区精品 | 13女裸体慰在线观看 | 亚洲狠狠色丁香婷婷综合 | 美女脱免费看网站女同 | 日韩精品免费看 | 色婷婷av一区二区三区影片 | 精品一区二区三区无码免费直播 | www.色94色.com| 国产精品久久久久久无毒偷食禁果 | 91精品啪在线观看国产手机 | 思九九爱九九 | 日本中文字幕乱码免费 | 久草热久草在线 | 亚洲国产午夜 | 国产偷国产偷亚洲高清app | 久久久久久久久久久久久国产 | 毛片一毛片二毛片三国产片 | 色欧美综合 | 北条麻妃精品久久中文字幕 | 欧美性在线视频 | 亚洲免费不卡视频 | 日日摸日日碰夜夜爽亚洲精品蜜乳 | 一级片在线放映 | www中文在线 | 日韩女同强女同hd | 窝窝午夜精品一区二区 | 大陆少妇xxxx做受高清 | yw尤物av无码国产在线观看 | 国产真实的和子乱拍在线观看 | 国产精品调教 | 国模少妇一区二区三区 | 欧洲激情网 | 波多野结衣绝顶大高潮 | 国产精品日韩精品欧美精品 | 成人精品啪啪欧美成 | 亚洲精品xxx | 国产suv精品一区二人妻 | 国内自拍一二三四2021 | 免费观看全黄做爰的视在线观看 | 国产熟女一区二区三区四区五区 | 亚洲国产成人极品综合 | 成人免费在线网站 | 久久九九免费 | 一二三四国产精品 | 国产精品九九九九九 | 亚洲国产精品无码久久电影 | 色天堂在线视频 | 三级色网 | 美女内射毛片在线看免费人动物 | 天天做天天躁天天躁 | 亚洲性大片 | 国产婷婷色综合av蜜臀av | 97视频人人 | 欧洲黄色网 | 久久人人爽人人爽人人 | 激情视频一区 | 天生舞男在线 | 人人澡人摸人人添 | 日日干日日摸 | 国产成人久久精品77777综合 | 国产精品一区二区福利视频 | www.com毛片 | 日韩精品一区在线 | 女人被狂躁到高潮视频免费软件 | 无码免费一区二区三区免费播放 | 蜜臀av夜夜澡人人爽人人 | 亚洲成人av网址 | 久久人人爽亚洲精品天堂 | 师尊双性精跪趴灌满h视频 湿女导航福利av导航 | 香蕉97视频观看在线观看 | 久久婷婷色综合 | 亚洲天堂中文字幕在线观看 | 亚洲男女在线 | 亚洲熟妇久久精品 | 国产乱人伦app精品久久 | 久久高清精品 | 潘金莲aa毛片一区二区 | 中文字幕视频网站 | 中文字幕日韩精品一区二区三区 | 成 人影片 免费观看 | 无码国产乱人伦偷精品视频 | 久久影视 | 97久人人做人人妻人人玩精品 | 免费成人激情视频 | 中文字幕网址在线 | 男人天堂网在线观看 | 国产小视频自拍 | 精品日韩视频 | 国产乱人伦真实精品视频 | 无码h黄动漫在线播放网站 国产精品高潮露脸在线观看 | jjzzjjzz在线观看 | 欧美色综合色 | 亚洲国语 | 女人性做爰24姿势视频 | 国产激情视频一区二区三区 | 黄色片免费 | 麻豆tube| 91精品国产91久久久久久吃药 | 国产一级一区二区 | 国产精品久久久久久精 | 久久高清| 黄色一级片免费播放 | 日韩精品日韩激情日韩综合 | 国产在视频线精品视频 | 欧美日韩国产成人在线观看 | 森林影视官网在线观看 | 美女黄网站18禁免费看 | 看免费真人视频网站 | 婷婷性多多影院 | 国产特黄大片aaaa毛片 | 人妻少妇精品久久 | 免费观看性生交大片女神 | 欧美黄色网 | 亚洲精品成人久久久 | 窝窝午夜精品一区二区 | 9999人体做爰大胆视频摄影 | 精品日本一区二区三区免费 | 日本网站在线免费观看 | 久久亚洲欧美日韩精品专区 | 日日躁狠狠躁夜夜躁av中文字幕 | 欧洲国产伦久久久久久久 | 亚洲天堂网络 | 五月天色站 | 涩涩屋www视频在线观看高清 | 九九精品在线播放 | 中日韩毛片| 国产成人精品亚洲线观看 | 日日噜噜噜噜夜夜爽亚洲精品 | 国产精品国产三级国产不产一地 | 日韩深夜影院 | 成人免费ā片在线观看 | 日日射影院 | 亚洲国产一区在线 | 黄色软件链接 | 国产精品视频播放 | 国产精品毛片在线 | 奶头挺立呻吟高潮视频 | 又粗又硬又大又爽免费视频播放 | 又黄又爽又色视频免费 | 国产在线国偷精品免费看 | 日本精品婷婷久久爽一下 | 11月流出美女撒尿偷拍在线播放 | 国产精品粉嫩懂色av | 在线播放无码高潮的视频 | 亚洲久热无码av中文字幕 | 日韩视频网站在线观看 | 国产精品色网 | 日本少妇xxxx动漫 | www成人在线观看 | 亚洲一区二区三区视频在线 | 91社区在线播放 | 天天干com | 把女邻居弄到潮喷的性经历 | 国产日韩欧美精品一区二区三区 | 成人在线免费看片 | 亚洲色图国产视频 | 婷婷五月综合色视频 | 免费在线观看av网址 | 亚洲精品成人av在线 | 成人黄性视频 | 久久老女人 | 亚洲欧美另类综合 | 久久久久久亚洲精品 | 热99在线视频 | 美女久久久久久久 | 日韩理论在线观看 | 日韩成人福利视频 | 亚洲 制服 丝袜 无码 | 免费又黄又粗又爽大片69 | 成人在线视屏 | av免费看在线 | 久久精品国产麻豆 | 黑白配高清在线观看免费版中文 | 免费看成人毛片 | 69xx国产 | 欧美国产日韩在线观看成人 | 日韩诱惑| 无码专区男人本色 | 黑人干亚洲女人 | 丁香八月婷婷 | 国产精品大尺度 | 少妇被躁爽到高潮 | 韩国三级中文字幕hd浴缸戏 | 美日韩毛片| 中文字幕网伦射乱中文 | 日日夜夜狠狠操 | 国产av一区二区精品久久凹凸 | 91免费国产 | 日韩欧美精品在线 | 一二三区乱码2021 | 国产911| 国产91玉足脚交在线播放 | 天天射,天天干 | 男人的天堂在线视频 | 免费看成人aa片无码视频羞羞网 | 丰满少妇高潮叫久久国产 | 色8久久精品久久久久久葡萄av | 精品少妇一区二区三区四区五区 | 欧美精品黑人猛交高潮 | 在线免费成人 | 妲己艳史淫片免费看 | 久久久久久免费观看 | 黄色一极视频 | 久久婷婷一区二区 | 欧美精品一区二区蜜臀亚洲 | 免费三片在线视频 | 国产乱轮视频 | 日本三级成本人网站 | 国精产品一二三三区入口 | 日韩高清dvd| 天天舔天天操天天干 | 成人高清免费观看mv | 亚洲精品久久久无码一区二区 | 日韩在线视频在线观看 | 日韩欧美在线精品 | 黑人粗长大战亚洲女 | 国产精品久久国产三级国不卡顿 | 紧身裙女教师三上悠亚红杏 | 日韩视频一区二区三区 | 婷婷爱五月天 | 中文字幕在线免费视频 | 和朋友换娶妻一区二区 | 野外偷拍做爰全过程 | 男人的天堂黄色 | av在线短片| 中文字幕在线不卡视频 | 99精品久久久久久久免费看蜜月 | 一区二区免费在线 | 中文 在线 日韩 亚洲 欧美 | 成人青青草 | 雨宫琴音一区二区三区 | 国产三级午夜理伦三级连载时间 | 精品人伦一区二区三区潘金莲 | 欧美性猛交xxxxx水多 | 久久av资源 | 国产在线资源 | 深夜福利网站在线观看 | 福利第一页 | 国产专业剧情av在线 | 天天躁日日躁狠狠躁av | 国产无遮挡又黄又爽免费视频 | 五月婷婷网 | 97中文字幕在线观看 | 欧美精品播放 | 欧美极品jiizzhd欧美爆 | 欧美亚洲视频一区 | 日韩亚州 | 肥熟一91porny丨九色丨 | 国产av一区二区三区 | 国产精品对白交换绿帽视频 | 午夜男女无遮挡拍拍视频 | 韩日视频| 日本三级一区 | 九九久久免费视频 | 亚洲图片在线播放 | 日韩中文字幕影院 | 操mm影院| 97自拍视频 | 日本v片 | 亚洲国产成人久久精品大牛影视 | aⅴ一区二区三区无卡无码 aⅴ在线免费观看 | 欧美乱大交xxxxx春色视频 | 师生出轨h灌满了1v1 | 欧美性受xxxx黑人猛交 | 久久99久久99久久 | 国产刺激出水片 | 色女生影院 | 成年午夜性影院 | 精品国产三级a∨在线 | 美女被张开双腿日出白浆 | 五月天国色天香国语版 | 国产亚洲精品久久久久5区 成熟了的熟妇毛茸茸 | 午夜黄色福利 | 人人看人人干 | 久久精品国产精品亚洲 | 欧美九九视频 | 欧美精品一区二区三区一线天视频 | 尤物av无码色av无码 | 中文字幕人妻熟女在线 | 偷窥自拍亚洲色图 | 日本一丰满一bbw | 国产精品69av | 狠狠cao日日穞夜夜穞av | 少妇欧美激情一区二区三区 | 久久久久久久久久影视 | 亚洲另类春色校园小说 | 69精品在线 | www色99| 91免费视频黄 | 性做久久久久久久免费看 | 日本做爰高潮视频 | 嫩草视频在线播放 | 96国产精品久久久久aⅴ四区 | 日韩中文字幕在线一区二区 | 天天想夜夜操 | 9.1在线观看免费 | 色婷婷久久综合中文久久蜜桃av | 性欧美videos另类极品小说 | 天天看片天天av免费观看 | 性日本xxx | 国模和精品嫩模私拍视频 | 国产欧美日韩一区二区三区在线 | 一级黄色网 | 91香蕉视频在线看 | 少妇撒尿一区二区在线视频 | 日本三级日本三级韩国三级视 | 午夜视频在线免费观看 | 日韩精品无码一区二区三区av | 国产av激情无码久久 | 亚洲国产成人久久综合电影 | 亚洲国产成人精品激情在线 | 激情午夜视频 | 天堂资源最新在线 | 天天曰夜夜曰 | 国产精品乱码在线观看 | 国产精品4| 99热免费精品 | 香蕉视频网页版 | 中文国产在线观看 | 日韩xxxxxxxxx | 一级做a视频 | 性生交大全免费看 | 玖玖精品在线 | 黄色片在线免费观看视频 | 水野朝阳av一区二区三区 | 欧美精品亚洲精品日韩传电影 | 国产三级高清 | 久久一区国产 | 免费美女av | 97色伦97色伦国产欧美空 | 欧美大片在线免费观看 | 日本美女影院 | 性的免费视频 | 欧美日韩亚洲精品瑜伽裤 | 男人狂躁女人爽的尖叫的免费视频 | 亚洲人成人7777在线播放 | 开心五月色婷婷综合开心网 | 美女一级黄 | 亚洲无套 | 中文字幕被公侵犯的漂亮人妻 | 午夜福利试看120秒体验区 | 青青草综合网 | 亚洲精品视频播放 | 中文有码在线观看 | 成年人看的黄色 | 亚洲成av人片在线观看无 | 午夜日韩视频 | 久久久久青草 | 国产无遮挡又黄又爽免费网站 | 伊人手机在线 | 天天av综合 | 国产午夜精品无码一区二区 | 伊人色综合久久天天网 | 日韩精品中文字幕一区二区三区 | 艹逼在线观看 | 亚洲熟妇av乱码在线观看 | 川上优av一区二区线观看 | 乳女教师の诱惑julia | 中国亲与子乱ay中文 | 欧美激情导航 | 人人爽人人爽人人片av亚洲 | 国产精品毛片一区二区在线看 | 欧美有码视频 | 日本中文字幕一区二区有限公司 | 亚洲第一aaaaa片 | 黄av在线免费观看 | 鲁一鲁av2019在线 | 三攻一受h啪肉np文 三级av毛片 | 鲁大师影院在线观看 | 国产白浆视频 | 97视频网址 | av网站观看 | 亚洲国产无| 国产精品自产拍高潮在线观看 | 国产成人av无码精品 | 中文视频在线 | 久久国产经典 | 青草青草久热 | 国模一二区 | 99久久夜色精品国产亚洲1000部 | av免费网 | 另类av在线 | 日韩小视频在线观看 | 天天干天天爽天天操 | 亚洲欧美视频一区 | 亚洲国产视频一区二区三区 | 影音先锋亚洲天堂 | 免费av网址大全 | 色网站在线看 | 亚洲国产天堂一区二区三区 | 欧美一区二区三区免费在线观看 | 欧美精品久久久久久久久老牛影院 | 国产美女无遮挡裸色视频 | 国产色视频播放网站www | 色av综合av综合无码网站 | 91日韩在线视频 | 国产精品国产三级国产在线观看 | 亚洲高清av一区二区三区 | 成人乱人伦精品小说 | 农村女人乱淫免费视频麻豆 | 亚洲精品国产熟女久久久 | 先锋影音一区二区三区 | 张津瑜警花国产精品一区 | 成人在线播放av | 艳妇av | 美丽姑娘国语版在线播放 | 国产性猛交xx乱老孕妇 | 欧美最猛黑人xxxx黑人表情 | 日日干网站 | 大学生三级中国dvd 大学生一级片 | 亚洲性少妇 | 天天艹夜夜艹 | 成人在线观看网站 | 99视频精品在线 | 亚洲国产精品无码久久电影 | jizz久久 | 日本伦片免费观看 | 一边添奶一边摸pp爽快视频 | 调教驯服丰满美艳麻麻在线视频 | 中文在线免费观看 | 少妇被躁爽到高潮无码文 | 国产精品日日摸夜夜添夜夜av | 亚洲精品久久久久av无码 | 男人午夜影院 | 国产欧美一区二区三区网站 | 久久精品国产欧美亚洲人人爽 | 亚洲人成网站色www 久久在线视频免费观看 | 欧美高清性xxxxhd | 91污网站 | 日韩激情视频网站 | 国产欧美一区二区三区不卡视频 | 国产欧美一区二区三区鸳鸯浴 | 婷婷一级片 | 婷婷五月六月激情综合色中文字幕 | 国产91久久婷婷一区二区 | 亚洲欧美激情在线 | 538精品一线 | 国产在线精品一区二区 | 91性生活| 日韩av免费网址 | 久久精品岛国av一区二区无码 | а√资源新版在线天堂 | 久久久久久久久福利 | 国色天香一卡2卡三卡4卡乱码 | 欧美日韩一区二区在线播放 | 免费看美女被靠到爽的视频 | 亚洲日韩国产成网在线观看 | 精品日产乱码久久久久久仙踪林 | 欧美日韩国产一级片 | 波多野结衣大战黑人8k经典 | 四虎永久免费观看 | a天堂在线观看视频 | 久久精品国产99精品国产亚洲性色 | 国产精品16p | 国产欧美一区二区三区沐欲 | 亚洲精品av久久久久久久影院 | 欧美亚洲精品在线 | 成 人 网 站国产免费观看 | r级无码视频在线观看 | 国产激情视频一区 | 成人影片网址 | 国产精品ai换脸张天爱 | 在线国产一区 | 夜色.com| 国产精品国产三级国产专业不 | 亚洲一区二区三区成人网站 | 国产凸凹视频一区二区 | 在线观看第一页 | 国模小黎自慰gogo人体 | av大全免费 | 中国少妇xxxxxx做受 | 日韩人妻精品一区二区三区视频 | 欧洲色视频 | 少妇特黄a片一区二区三区 精品香蕉一区二区三区 | 久久久久久九九九九 | 中文字幕一区二区三区乱码图片 | 午夜精品小视频 | 日本中文字幕免费 | 国产真实交换夫妇视频 | 亚洲美女福利视频 | 日韩精品久久久久久久的张开腿让 | 一区免费视频 | 少妇被多人c夜夜爽爽 | 阿v天堂在线 | 在线视频成人 | 久久九九热 | 国产国产精品久久久久 | 麻豆国产尤物av尤物在线观看 | 99国产精品久久久久久 | 婷婷综合五月 | 小嫩草张开腿让我爽了一夜 | 黄色一集片 | 海量av资源| 都市激情综合 | 国产欧美一区二区精品秋霞影院 | 91久久国产| 天天干夜夜操视频 | 亚洲丁香色 | 欧美人与动牲交片免费 | 超爱碰在线资源 | 欧美变态绿帽cuckold | 91国内精品野花午夜精品 | 青青草原综合久久大伊人精品 | 99精品国产免费久久久久久按摩 | 天堂国产欧美一区二区三区 | 天天干妹子 | 久久久久久国产精品亚洲78 | 色综合视频在线观看 | 性折磨bdsm欧美激情另类 | 日本在线视频免费 | 先锋影音av资源在线观看 | 国产乱色精品成人免费视频 | 粉嫩在线一区二区三区视频 | 视频在线观看一区二区三区 | 欧美激情一区在线 | a欧美在线 | 91精品综合久久久久m3u8 | 国产极品美女高潮无套嗷嗷叫酒店 | 91在线精品啪婷婷 | 国产精品乱码一区二区三区视频 | 欧美一区视频在线 | 亚洲最新无码中文字幕久久 | 伊人无码精品久久一区二区 | 日韩加勒比一本无码精品 | 日本妞一区 | 乱人伦人妻精品一区二区 | 国产理论一区二区三区 | 91com在线观看| 欧美久久久久 | 狠狠操av| 麻豆av一区二区 | 51真实女性私密spa按摩偷拍 | 国产激情久久久久影院 | 青青草国产 | 精品中文字幕一区二区三区av | 三级黄色av| 亚洲乱码一区av春药高潮 | 国产最露的三级 | 91午夜少妇三级全黄 | 青青草原在线免费 | 欧美亚洲国产一区二区三区 | 五月婷婷激情第四季 | 日本精品一区二区三区视频 | 色婷婷国产 | 网站黄色在线免费观看 | 性欧美8khd高清极品 | 欧美v国产v亚洲v日韩九九 | 欧美亚洲视频在线观看 | 国产一区二区免费播放 | 天天摸天天看天天做天天爽 | 久久中文在线 | 免费看涩涩视频软件 | 成人自拍视频在线观看 | 天天干天天噜 | 日操干 | 秋霞欧洲 | 伊人免费在线观看 | 久久精品一区二区免费播放 | 成人激情开心 | 任你躁国产自任一区二区三区 | 欧美国产乱视频 | 又湿又紧又大又爽a视频国产 | 久久久亚洲欧洲日产国码aⅴ | 午夜资源| 久久精品国内一区二区三区 | 日韩精品在线观看免费 | 国产精品二区一区二区aⅴ污介绍 | 人成在线观看 | 国产女人18毛片水真多1 | 少妇被爽到高潮喷水久久欧美精品 | 久久久伦理片 | 免费看毛片基地 | 性猛交xxxx乱大交孕妇2十 | 乱子伦一区二区 | 久久精品国产亚卅av嘿嘿 | 久久天天躁夜夜躁狠狠i女人 | 亚洲欧美综合一区二区三区 |