This is the 17th day of my participation in the More text Challenge. For more details, see more text Challenge

>>>> ๐Ÿ˜œ๐Ÿ˜œ๐Ÿ˜œ Making: ๐Ÿ‘‰ github.com/black-ant

Current case Source: ๐Ÿ‘‰ Github AOP Source

A. The preface

After taking stock of the processes involved in IOC, this article finally talks about AOP. The series starts with:

  • AOP initialization
  • Creation of an AOP proxy class
  • AOP interception
  • Proxy method invocation of AOP

The above chapters are described respectively

2. AOP invocation entry

Aop initialization refers to the point at which Aop configuration begins:

Combined with what IOC has learned Before, the core of Aop is completed through postProcess, which is divided into Before and After processes

  • ApplyBeanPostProcessorsBeforeInstantiation createBean (create)
  • ApplyBeanPostProcessorsAfterInitialization (initializeBean processing)

// Route 1: Handle Aware, which can be generated in advance by getCustomTargetSource

  • C- AbstractAutowireCapableBeanFactory # createBean
  • C- AbstractAutowireCapableBeanFactory # resolveBeforeInstantiation
  • C- AbstractAutowireCapableBeanFactory # applyBeanPostProcessorsBeforeInstantiation
  • C- AnnotationAwareAspectJAutoProxyCreator # postProcessBeforeInstantiation

// Route 2: process the BeanPostProcessor and create the logic normally

  • C- AbstractAutowireCapableBeanFactory # createBean
  • C- AbstractAutowireCapableBeanFactory # doCreateBean
  • C- AbstractAutowireCapableBeanFactory # initializeBean
  • C- AbstractAutowireCapableBeanFactory # applyBeanPostProcessorsAfterInitialization
/ / Step 1: in the createBean applyBeanPostProcessorsBeforeInstantiation for processing
protected Object createBean(String beanName, RootBeanDefinition mbd, @Nullable Object[] args)
			throws BeanCreationException {

    / /... Omit some logic and initialize the proxy class here
    Object bean = resolveBeforeInstantiation(beanName, mbdToUse);
    
    // In this case, the Step 2 creation logic is called and the proxy class is processed
    Object beanInstance = doCreateBean(beanName, mbdToUse, args);

    / /...
    return beanInstance;

}


// Step 2: In the IOC process initializeBean, PostProcessors are processed
protected Object initializeBean(final String beanName, final Object bean, @Nullable RootBeanDefinition mbd) {

    / /...
    invokeInitMethods(beanName, wrappedBean, mbd);

    / / call applyBeanPostProcessorsAfterInitialization AfterPostProcessors processing
    if (mbd == null| |! mbd.isSynthetic()) { wrappedBean = applyBeanPostProcessorsAfterInitialization(wrappedBean, beanName); }return wrappedBean;
}


Copy the code

After processing, we delegate to the specified interceptors before calling the bean itself. There are two types of interceptors:

  • Common: Shared by all agents created for it
  • Specific: Unique for each bean instance

Proxy restrictions: Subclasses can apply any policy to determine whether a bean is to be proxied, such as by type, name, definition details, and so on

Why is the getCustomTargetSource generated ahead of time?

GetCustomTargetSource is used to generate a proxy object for the Bean before it is instantiated using the TargetSourceCreator object.

This pattern needs to be customized. See appendix: Creating a customTargetSource

2.1 applyBeanPostProcessorsBeforeInstantiation

Step 1: call PostProcessorsBefore C – AbstractAutowireCapableBeanFactory

The main invokes the org. Springframework. Aop. Aspectj. The annotation. AnnotationAwareAspectJAutoProxyCreator for processing

for (BeanPostProcessor bp : getBeanPostProcessors()) {
    if (bp instanceof InstantiationAwareBeanPostProcessor) {
         // idp -> org.springframework.aop.aspectj.annotation.AnnotationAwareAspectJAutoProxyCreator
        InstantiationAwareBeanPostProcessor ibp = (InstantiationAwareBeanPostProcessor) bp;
        Object result = ibp.postProcessBeforeInstantiation(beanClass, beanName);
         / /...}}Copy the code

Step 2: Configure a prerequisite c-AbstractautoProxyCreator

public Object postProcessBeforeInstantiation(Class
        beanClass, String beanName) {
    Object cacheKey = getCacheKey(beanClass, beanName);

    if(! StringUtils.hasLength(beanName) || !this.targetSourcedBeans.contains(beanName)) {
         // If it has been processed, return it directly
        if (this.advisedBeans.containsKey(cacheKey)) {
            return null;
        }
        // isInfrastructureClass -> Infrastructure class that should not be broiled
         // shouldSkip -> Note that subclasses override shouldSkip, which is used to skip automatic proxies
        if (isInfrastructureClass(beanClass) || shouldSkip(beanClass, beanName)) {
             // PRO21001 -> advisedBeans
            this.advisedBeans.put(cacheKey, Boolean.FALSE);
            return null; }}// Create a proxy here if you have a custom TargetSource
    // TargetSource will handle the target instance in a custom manner
    TargetSource targetSource = getCustomTargetSource(beanClass, beanName);
    if(targetSource ! =null) {
        // This is where the agent is created after customizing the CustomTargetSource logic
        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;
}


Copy the code

Step 2-1: Return the target instance

C-getcustomtargetsource: Creates the target source for the bean instance. If set, use anyTargetSourceCreators
protected TargetSource getCustomTargetSource(Class
        beanClass, String beanName) {
    // This is where the creator is required and customization needs to be performed
    if (this.customTargetSourceCreators ! =null &&
        this.beanFactory ! =null && this.beanFactory.containsBean(beanName)) {
        for (TargetSourceCreator tsc : this.customTargetSourceCreators) {
            // Return the target object proxy class through the creator
            TargetSource ts = tsc.getTargetSource(beanClass, beanName);
            if(ts ! =null) {
                returnts; }}}// The undefined customTargetSource is returned here
    return null;
}
Copy the code

2.2 applyBeanPostProcessorsAfterInitialization

We said that the proxy is created before Bean instantiation, but more often than not, the proxy logic is created after Bean instantiation

Step 1: Entry of IOC initializeBean

IOC initializeBean, call applyBeanPostProcessorsAfterInitialization logic

if (mbd == null| |! mbd.isSynthetic()) { wrappedBean = applyBeanPostProcessorsAfterInitialization(wrappedBean, beanName); }Copy the code

Step 2: treated BeanPostProcessor applyBeanPostProcessorsAfterInitialization in cycles

for (BeanPostProcessor processor : getBeanPostProcessors()) {
    // BeanPostProcessor -> org.springframework.aop.aspectj.annotation.AnnotationAwareAspectJAutoProxyCreator
    Object current = processor.postProcessAfterInitialization(result, beanName);
    / /...
}
Copy the code

PS: as you can see, here to Bean is AnnotationAwareAspectJAutoProxyCreator PostProcessor

Step 3: # AbstractAutoProxyCreator postProcessAfterInitialization build proxy objects

If the bean is identified as a proxy, the proxy is created using the configured interceptor

public Object postProcessAfterInitialization(@Nullable Object bean, String beanName) {
    if(bean ! =null) {
         // cacheKey is used to identify
        Object cacheKey = getCacheKey(bean.getClass(), beanName);
        if (this.earlyProxyReferences.remove(cacheKey) ! = bean) {returnwrapIfNecessary(bean, beanName, cacheKey); }}return bean;
}
Copy the code

Step 4: wrapIfNecessary to create an official agent class c-AbstractautoProxyCreator


protected Object wrapIfNecessary(Object bean, String beanName, Object cacheKey) {
    // 
    if (StringUtils.hasLength(beanName) && this.targetSourcedBeans.contains(beanName)) {
        return bean;
    }
    if (Boolean.FALSE.equals(this.advisedBeans.get(cacheKey))) {
        return bean;
    }
    if (isInfrastructureClass(bean.getClass()) || shouldSkip(bean.getClass(), beanName)) {
        this.advisedBeans.put(cacheKey, Boolean.FALSE);
        return bean;
    }

    // If notification is configured, create a Proxy
    Object[] specificInterceptors = getAdvicesAndAdvisorsForBean(bean.getClass(), beanName, null);
    if(specificInterceptors ! = DO_NOT_PROXY) {this.advisedBeans.put(cacheKey, Boolean.TRUE);
         // Core logic, create proxy -> 2.2 Aop proxy class creation
         SingletonTargetSource -> PRO21002: What is the TargetSource?
        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;
}
Copy the code

PRO21001 -> advisedBeans function

// advisedBeans is used to identify the proxy status of the current Bean. The object key is obtained by getCacheKey
Map<Object, Boolean> advisedBeans = new ConcurrentHashMap<>(256); -> AdvisedBeans.put (cacheKey, Boolean) -> AdvisedBeans.put (cacheKey, Boolean) Boolean) - No final action (annotated without annotations) -> AdvisedBeans.put (cacheKey, boolea.false)Copy the code

PRO21002: What is the TargetSource?

As you can see, here is actually similar to the following applyBeanPostProcessorsAfterInitialization, then deal with what is above?

The TargetSource is used to get the current “target” of an AOP call. It does not directly create an object instance of the target bean when Spring proxys the target bean. Instead, you wrap the target bean around an object of type TargetSource, and then get the target object through getTarget

TargetSource is an interface whose architecture can be seen below:

Three. Added point: Advisors, explanation of getAdvicesAndAdvisorsForBean process

This section of getAdvicesAndAdvisorsForBean process in detail

Step 1: Screen the Advices C-AbstractautoproxyCreator

M- getAdvicesAndAdvisorsForBean Object[] getAdvicesAndAdvisorsForBean(Class<? > beanClass, String beanName,TargetSource customTargetSource) ? - Returns whether the given bean is to be propped, what additional advice (such as an AOP Alliance interceptor) and advisors to apply/ / the class there are mainly two implementation class: AbstractAdvisorAutoProxyCreator, BeanNameAutoProxyCreator
    
protectedObject[] getAdvicesAndAdvisorsForBean( Class<? > beanClass, String beanName,@Nullable TargetSource targetSource) {

    List<Advisor> advisors = findEligibleAdvisors(beanClass, beanName);
    if (advisors.isEmpty()) {
        return DO_NOT_PROXY;
    }
    return advisors.toArray();
}

Copy the code

Step 2: Find all qualified Advisors for the automatic proxy class

protected List<Advisor> findEligibleAdvisors(Class
        beanClass, String beanName) {
    // Step 2-1: Query all notifications corresponding to the aspect
    List<Advisor> candidateAdvisors = findCandidateAdvisors();
    // Step 2-2: filter the pointcuts corresponding to the class
    List<Advisor> eligibleAdvisors = findAdvisorsThatCanApply(candidateAdvisors, beanClass, beanName);
    // Empty object, to be implemented
    extendAdvisors(eligibleAdvisors);
    if(! eligibleAdvisors.isEmpty()) { eligibleAdvisors = sortAdvisors(eligibleAdvisors); }return eligibleAdvisors;
}
Copy the code

Step 2-1: BeanFactoryAdvisorRetrievalHelper get notification object

2-1-1: first call the implementation class (AnnotationAwareAspectJAutoProxyCreator)

๐Ÿ‘‰ Sections are scanned and all notification points are cached during this process

protected List<Advisor> findCandidateAdvisors(a) {
    // Add all found Spring notifications according to the superclass rules
    List<Advisor> advisors = super.findCandidateAdvisors();
    // Build notifications for all AspectJ aspects in the bean factory
    if (this.aspectJAdvisorsBuilder ! =null) {
         / / 2-1-1
        advisors.addAll(this.aspectJAdvisorsBuilder.buildAspectJAdvisors());
    }
    return advisors;
}

// 2-1-1-1: build the notifier
public List<Advisor> buildAspectJAdvisors(a) {
    List<String> aspectNames = this.aspectBeanNames;

    if (aspectNames == null) {
        synchronized (this) {
            aspectNames = this.aspectBeanNames;
            if (aspectNames == null) {
                List<Advisor> advisors = new ArrayList<>();
                aspectNames = new ArrayList<>();
                String[] beanNames = BeanFactoryUtils.beanNamesForTypeIncludingAncestors(
                            this.beanFactory, Object.class, true.false);
                for (String beanName : beanNames) {
                    if(! isEligibleBean(beanName)) {continue;
                    }
                    // needs to be processed before the cache is instantiatedClass<? > beanType =this.beanFactory.getType(beanName);
                    if (beanType == null) {
                        continue;
                    }
                    // Filter out the section Bean, debug will get my own section
                    if (this.advisorFactory.isAspect(beanType)) {
                        aspectNames.add(beanName);
                        // Get the section metadata -> PIC0001
                        AspectMetadata amd = new AspectMetadata(beanType, beanName);
                        if (amd.getAjType().getPerClause().getKind() == PerClauseKind.SINGLETON) {
                            // Subinterface of AspectInstanceFactory, which returns AspectMetadata associated with an AspectJ annotated class
                            MetadataAwareAspectInstanceFactory factory =
                                        new BeanFactoryAspectInstanceFactory(this.beanFactory, beanName);
                            // Get all notification points -> PIC0002
                            List<Advisor> classAdvisors = this.advisorFactory.getAdvisors(factory);
                            if (this.beanFactory.isSingleton(beanName)) {
                                // Here it is successfully placed in the cache
                                this.advisorsCache.put(beanName, classAdvisors);
                            }
                            else {
                                this.aspectFactoryCache.put(beanName, factory);
                            }
                            advisors.addAll(classAdvisors);
                        } else {
                            // Per target or per this.
                            if (this.beanFactory.isSingleton(beanName)) {
                                throw new IllegalArgumentException("Bean with name '" + beanName +
                                            "' is a singleton, but aspect instantiation model is not singleton");
                            }
                            MetadataAwareAspectInstanceFactory factory =
                                new PrototypeAspectInstanceFactory(this.beanFactory, beanName);
                            this.aspectFactoryCache.put(beanName, factory);
                            advisors.addAll(this.advisorFactory.getAdvisors(factory)); }}}this.aspectBeanNames = aspectNames;
                returnadvisors; }}}if (aspectNames.isEmpty()) {
        return Collections.emptyList();
    }
    
    // Loop through the aspect aspect to add advice
    List<Advisor> advisors = new ArrayList<>();
    for (String aspectName : aspectNames) {
        List<Advisor> cachedAdvisors = this.advisorsCache.get(aspectName);
        if(cachedAdvisors ! =null) {
            advisors.addAll(cachedAdvisors);
        } else {
            MetadataAwareAspectInstanceFactory factory = this.aspectFactoryCache.get(aspectName);
            advisors.addAll(this.advisorFactory.getAdvisors(factory)); }}return advisors;
}


Copy the code

PIC0001: section metadata

PIC0002: the pointcut to be notified

2-1-2: calls the parent class

protected List<Advisor> findCandidateAdvisors(a) {
    return this.advisorRetrievalHelper.findAdvisorBeans();
}
Copy the code

2-1-3: the actual method

public List<Advisor> findAdvisorBeans(a) {
    // Determine the list of Advisor bean names
    String[] advisorNames = this.cachedAdvisorBeanNames;
    if (advisorNames == null) {
        // Do not initialize FactoryBeans here: We need to leave all regular beans
        // uninitialized to let the auto-proxy creator apply to them!
        advisorNames = BeanFactoryUtils.beanNamesForTypeIncludingAncestors(
                    this.beanFactory, Advisor.class, true.false);
        this.cachedAdvisorBeanNames = advisorNames;
    }
    if (advisorNames.length == 0) {
        return new ArrayList<>();
    }

    List<Advisor> advisors = new ArrayList<>();
    for (String name : advisorNames) {
        if (isEligibleBean(name)) {
            if (this.beanFactory.isCurrentlyInCreation(name)) {
                / / the log omitted
            } else {
                try {
                    advisors.add(this.beanFactory.getBean(name, Advisor.class));
                } catch (BeanCreationException ex) {
                    Throwable rootCause = ex.getMostSpecificCause();
                    if (rootCause instanceof BeanCurrentlyInCreationException) {
                        BeanCreationException bce = (BeanCreationException) rootCause;
                        String bceBeanName = bce.getBeanName();
                        if(bceBeanName ! =null && this.beanFactory.isCurrentlyInCreation(bceBeanName)) {
                            // Ignore: indicates a reference back to the bean we're trying to advise.
                            // We want to find advisors other than the currently created bean itself.
                            continue; }}throwex; }}}}return advisors;
}
Copy the code

Step 2-2: Search the given candidate Advisor to find all the advisors applicable to the specified bean


protected List<Advisor> findAdvisorsThatCanApply( List
       
         candidateAdvisors, Class
         beanClass, String beanName)
        {

    ProxyCreationContext.setCurrentProxiedBeanName(beanName);
    try {
        return AopUtils.findAdvisorsThatCanApply(candidateAdvisors, beanClass);
    } finally {
        ProxyCreationContext.setCurrentProxiedBeanName(null); }}Copy the code

Step 2-3: Sort sort

protected List<Advisor> sortAdvisors(List<Advisor> advisors) {
    AnnotationAwareOrderComparator.sort(advisors);
    return advisors;
}

/ / / Pro: AnnotationAwareOrderComparator is what?AnnotationAwareOrderComparator is an extension of the OrderComparator, it supports Spring Ordered interface and the Order and PriorityCopy the code

conclusion

Now that the AOP initialization phase is almost complete, let’s look at the creation of proxy classes. To summarize all the concepts in this document:

  • InitializeBean turns on the AOP’s main processing logic
  • AbstractAutowireCapableBeanFactory call applyBeanPostProcessorsBeforeInstantiation create custom proxy class
  • AbstractAutowireCapableBeanFactory call applyBeanPostProcessorsAfterInitialization screening notifier

Appendix: Creating a customTargetSource

Refer to the address @blog.csdn.net/qq_39002724…

Step 1: Prepare the CustomTargetSource

public class DefaultCustomTargetSource extends AbstractBeanFactoryBasedTargetSource {

    private Logger logger = LoggerFactory.getLogger(this.getClass());

    @Override
    public Object getTarget(a) throws Exception {
        logger.info("-- -- -- -- -- - > enter [DefaultCustomTargetSource], returns the current target < -- -- -- -- -- -- --");
        returngetBeanFactory().getBean(getTargetBeanName()); }}Copy the code

Step 2: Prepare the creator

This object is used to return the CustomTargetSource that was created

public class DefaultCustomerTargetSourceCreator extends AbstractBeanFactoryBasedTargetSourceCreator {

    private Logger logger = LoggerFactory.getLogger(this.getClass());

    @Override
    protected AbstractBeanFactoryBasedTargetSource createBeanFactoryBasedTargetSource(Class
        beanClass, String beanName) {
        logger.info("-- -- -- -- -- - > CustomerTargetSourceCreator build: [creation process into the process, to return to the default resource class DefaultCustomTargetSource] < -- -- -- -- -- -- --");
        if (getBeanFactory() instanceof ConfigurableListableBeanFactory) {
            if (beanClass.isAssignableFrom(OtherService.class)) {
                return newDefaultCustomTargetSource(); }}return null; }}Copy the code

Step 3: Configure the CustomTargetSource resource

@Component
public class CustomTargetSourceCreatorConfig implements BeanPostProcessor.PriorityOrdered.BeanFactoryAware {

    private Logger logger = LoggerFactory.getLogger(this.getClass());

    private BeanFactory beanFactory;

    @Override
    public int getOrder(a) {
        return 45;
    }

    @Override
    public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
        logger.info("-- -- -- -- -- - > enter CustomTargetSourceCreatorConfig initialization loading logic, return to the default creator < -- -- -- -- -- -- --");
        if (bean instanceof AnnotationAwareAspectJAutoProxyCreator) {
            AnnotationAwareAspectJAutoProxyCreator annotationAwareAspectJAutoProxyCreator = (AnnotationAwareAspectJAutoProxyCreator) bean;
            DefaultCustomerTargetSourceCreator customTargetSourceCreator = new DefaultCustomerTargetSourceCreator();
            customTargetSourceCreator.setBeanFactory(beanFactory);
            annotationAwareAspectJAutoProxyCreator.setCustomTargetSourceCreators(customTargetSourceCreator);
        }
        return bean;
    }

    @Override
    public void setBeanFactory(BeanFactory beanFactory) throws BeansException {
        this.beanFactory = beanFactory; }}Copy the code