Bean对象创建流程

调用流程:AbstractApplicationContext#refresh

=> AbstractApplicationContext#finishBeanFactoryInitialization

=> DefaultListableBeanFactory#preInstantiateSingletons

=> AbstractBeanFactory#getBean

=> AbstractBeanFactory#doGetBean

=> AbstractBeanFactory#createBean
创建Bean对象的方法入口: AbstractAutowireCapableBeanFactory#createBean

// AbstractBeanFactory#doGetBean
if (mbd.isSingleton()) {
    // 关键方法 createBean  getSingleton
    sharedInstance = this.getSingleton(beanName, () -> {
        try {
            return this.createBean(beanName, mbd, args);
        } catch (BeansException var5) {
            this.destroySingleton(beanName);
            throw var5;
        }
    });
    bean = this.getObjectForBeanInstance(sharedInstance, name, beanName, mbd);
} else if (mbd.isPrototype()) {
    var11 = null;

    Object prototypeInstance;
    try {
        this.beforePrototypeCreation(beanName);
        prototypeInstance = this.createBean(beanName, mbd, args);
    } finally {
        this.afterPrototypeCreation(beanName);
    }

    bean = this.getObjectForBeanInstance(prototypeInstance, name, beanName, mbd);
}

准备工作:

1、保证类以及加载完成了

2、处理lookup-method and replace-method(打上标识mo.setOverloaded(false);)

3、断定是否需要spring来管理这个类

4、创建Bean的关键方法(AbstractAutowireCapableBeanFactory#doCreateBean)

回调BeanPostProcessor(一)

第一次回调BeanPostProcessor(InstantiationAwareBeanPostProcessor 的 postProcessBeforeInstantiation 方法),用来断定是否需要spring来管理这个类

// AbstractAutowireCapableBeanFactory#createBean
Object bean = resolveBeforeInstantiation(beanName, mbdToUse);

这里就会返回这个接口中 postProcessBeforeInstantiation 方法返回的对象,spring中默认返回的是null。表示spring会去维护这个类中的一些依赖关系,如果我们不需要spring来维护这些但是又希望将spring将这个类放入容器中,就可以通过这个方法,将我们需要放入spring的对象返回,这里会直接返回这个对象,结束这个bean的创建

// 方法是判断是否向bean工厂中添加了 InstantiationAwareBeanPostProcessor 类型的后置处理器
protected Object resolveBeforeInstantiation(String beanName, RootBeanDefinition mbd) {
    if (!mbd.isSynthetic() && hasInstantiationAwareBeanPostProcessors()) {
        Class<?> targetType = determineTargetType(beanName, mbd);
        if (targetType != null) {
            // InstantiationAwareBeanPostProcessor#postProcessBeforeInstantiation
            bean = applyBeanPostProcessorsBeforeInstantiation(targetType, beanName);
            if (bean != null) {
                // 遍历所有的 BeanPostProcessor 执行它的 postProcessAfterInitialization方法
                bean = applyBeanPostProcessorsAfterInitialization(bean, beanName);
            }
        }
    }
}

关键方法AbstractAutowireCapableBeanFactory#doCreateBean

Object beanInstance = doCreateBean(beanName, mbdToUse, args);

doCreateBean的逻辑如下:

  • 检查是否存在与factoryBean的缓存中,
// AbstractAutowireCapableBeanFactory#doCreateBean
instanceWrapper = this.factoryBeanInstanceCache.remove(beanName);
  • 如果不在FactoryBean的缓存中,则创建bean
if (instanceWrapper == null) {
    instanceWrapper = createBeanInstance(beanName, mbd, args);
}
  • 创建Bean的核心方法
// AbstractAutowireCapableBeanFactory#createBeanInstance
protected BeanWrapper createBeanInstance(String beanName, RootBeanDefinition mbd, Object[] args) {
    // 确保bd中的class已经被加载进来了
    Class<?> beanClass = resolveBeanClass(mbd, beanName);
    // 检查类的访问权限
    if (beanClass != null && !Modifier.isPublic(beanClass.getModifiers()) && !mbd.isNonPublicAccessAllowed()) {throw xxx}
    if (mbd.getFactoryMethodName() != null) {
        // bd是否存在工厂方法,如果存在工厂方法直接通过工厂方法进行创建
        // 参考 @Bean标注的static方法,这个bean就设置了FactoryMethod
        // @Bean 标注的普通方法是设置了 UniqueFactoryMethod 这两者的区别??
        return instantiateUsingFactoryMethod(beanName, mbd, args);
    }
    // Shortcut when re-creating the same bean...
    // 创建一个快捷方式,当重新创建相同的bean时,比如创建prototype类型的bean时
    // 从这里可以知道(上一次创建时)是使用无参构造函数创建,还是使用有参构造函数创建bean实例对象
    boolean resolved = false;
    boolean autowireNecessary = false;
    if (args == null) {
        synchronized (mbd.constructorArgumentLock) {
            if (mbd.resolvedConstructorOrFactoryMethod != null) {
                resolved = true;
                autowireNecessary = mbd.constructorArgumentsResolved;
            }
        }
    }
    if (resolved) {
        // 上次次已经创建过了
        if (autowireNecessary) {
            // 使用构造函数注入的方式实例化bean
            return autowireConstructor(beanName, mbd, null, null);
        } else {
            // 使用无参构造函数实例化bean
            return instantiateBean(beanName, mbd);
        }
    }
    // 未完...
}

回调BeanPostProcessor(二)

Spring是如何知道是使用无参还是有参的构造函数?这与BeanPostProcessor的第二次回调有关

// Candidate constructors for autowiring?
// 这里是 BeanPostProcessor 的第二次回调,主要是推断构造方法
Constructor<?>[] ctors = determineConstructorsFromBeanPostProcessors(beanClass, beanName);

上面这一句的总体意思就是是否采用有参的构造函数进行注入
1、只存在无参的构造函数时,这里会返回一个null
2、只存在一个有参的构造函数时,这里会返回这个有参的构造函数
3、存在一个无参的构造函数和一个有参的构造函数 ,返回null
4、存在两个有参的构造函数,返回null
5、存在三个或者三个以上的构造方法,返回null

// AbstractAutowireCapableBeanFactory#determineConstructorsFromBeanPostProcessors
protected Constructor<?>[] determineConstructorsFromBeanPostProcessors(@Nullable Class<?> beanClass, String beanName) throws BeansException {
    // 执行 AutowiredAnnotationBeanPostProcessor.determineCandidateConstructors推断构造函数
    if (beanClass != null && hasInstantiationAwareBeanPostProcessors()) {
        for (BeanPostProcessor bp : getBeanPostProcessors()) {
            if (bp instanceof SmartInstantiationAwareBeanPostProcessor) {
                SmartInstantiationAwareBeanPostProcessor ibp = (SmartInstantiationAwareBeanPostProcessor) bp;
                Constructor<?>[] ctors = ibp.determineCandidateConstructors(beanClass, beanName);
                if (ctors != null) {
                    return ctors;
                }
            }
        }
    }
    return null;
}

推断构造函数总结:

当只存在一个构造函数时:
(1)默认的无参构造函数,这里会返回null
(2)自己写了一个有参的构造函数时,这里会返回这个有参的构造函数
当存在两个构造函数时
(1)两个都是有参的构造函数,这里会返回null
(2)一个有参的构造函数,一个无参的构造函数,这里还是返回null
在第二种情况中,怎么才能返回两个构造函数呢?
存在无参的构造函数,存在primaryConstructor,并且无参的构造函数和 primaryConstructor 不相等
这里的 primaryConstructor 是针对Kotlin的类的,所以在JAVA中这里永远都会返回一个null
所以在JAVA中这里是不可能返回两个构造函数的
当存在三个或者三个以上的构造函数时
这里一律返回null

所以要么只存在一个无参的构造方法,要么只存在一个有参的构造方法

// AutowiredAnnotationBeanPostProcessor.determineCandidateConstructors
// 1、记录Lookup注解的方法
// 获取类中的所有构造函数
rawCandidates = beanClass.getDeclaredConstructors();
// 遍历所有的构造函数
for (Constructor<?> candidate : rawCandidates) {
    // isSynthetic方法判断如果是合成的构造函数就返回true(java自己生成的),自己写的构造函数会返回false
    if (!candidate.isSynthetic()) {
        nonSyntheticConstructors++;// 统计构造函数的个数
    }else if (primaryConstructor != null) {
        // 存在primaryConstructor,会跳出循环
        continue;
    }
    // 获取构造方法上的注解
    AnnotationAttributes ann = findAutowiredAnnotation(candidate);
    if (ann == null) {
        // 构造方法上没有注解
        // 如果这个beanClass是被CGlib进行代理的class类型,则获取父类的构造函数
        // 进而拿到父类构造函数上的注解
        Class<?> userClass = ClassUtils.getUserClass(beanClass);
        if (userClass != beanClass) {
            Constructor<?> superCtor =
                        userClass.getDeclaredConstructor(candidate.getParameterTypes());
            ann = findAutowiredAnnotation(superCtor);
        }
    }
    if (ann != null) {
        if (requiredConstructor != null) {
            throw new BeanCreationException(beanName,requiredConstructor);
        }
        boolean required = determineRequiredStatus(ann);
        if (required) {
            if (!candidates.isEmpty()) {
                throw new BeanCreationException(beanName,candidate);
            }
            requiredConstructor = candidate;
        }
        // 只有在构造方法存在注解的情况下向 candidates 集合中添加了构造方法
        candidates.add(candidate);
    }else if (candidate.getParameterCount() == 0) {
        // 没有参数的构造函数,表示为默认的构造函数
        defaultConstructor = candidate;
    }
}
if (!candidates.isEmpty()) {
    // Add default constructor to list of optional constructors, as fallback.
    if (requiredConstructor == null) {
        if (defaultConstructor != null) {
            candidates.add(defaultConstructor);
        }else if (candidates.size() == 1 && logger.isInfoEnabled()) {
            logger.info("Inconsistent constructor declaration on bean with name");
        }
    }
    candidateConstructors = candidates.toArray(new Constructor<?>[0]);
}else if (rawCandidates.length == 1 && rawCandidates[0].getParameterCount() > 0) {
    // rawCandidates表示所有的构造函数,表示只有一个有参的构造函数
    candidateConstructors = new Constructor<?>[] {rawCandidates[0]};
}else if (nonSyntheticConstructors == 2 && primaryConstructor != null &&
    defaultConstructor != null && !primaryConstructor.equals(defaultConstructor)) {
    // 有两个构造函数,必须存在 一个 primaryConstructor 和一个 defaultConstructor
    // 并且两者不相等
    candidateConstructors = new Constructor<?>[] {primaryConstructor, defaultConstructor};
}else if (nonSyntheticConstructors == 1 && primaryConstructor != null) {
    // 有一个primaryConstructor,就返回这个primaryConstructor
    candidateConstructors = new Constructor<?>[] {primaryConstructor};
}else {
    // 不满足上述条件的一律视为不存在合适的构造函数 返回一个空数组
    candidateConstructors = new Constructor<?>[0];
}
this.candidateConstructorsCache.put(beanClass, candidateConstructors);
return (candidateConstructors.length > 0 ? candidateConstructors : null);

调用构造方法创建对象

以下四种情况只有存在一种,都会使用有参构造函数进行bean的实例化

1、存在且只有 有参的构造函数 比较绕
2、bd的装配模式设置为了构造方法装配模式
3、通过操作bd,给定了bd的构造方法的参数(mybatis中Mapper的生成,给定了Mapper构造函数的接口名称 => 通过构造函数去创建Mapper代理对象)
4、传入的args参数不为空,这里传入的参数是构造函数中对应的实参

// AbstractAutowireCapableBeanFactory#createBeanInstance
Constructor<?>[] ctors = determineConstructorsFromBeanPostProcessors(beanClass, beanName);
// 调用构造函数
if (ctors != null || mbd.getResolvedAutowireMode() == AUTOWIRE_CONSTRUCTOR ||
    mbd.hasConstructorArgumentValues() || !ObjectUtils.isEmpty(args)) {
    // 通过有参的构造函数创建bean
    return autowireConstructor(beanName, mbd, ctors, args);
}
// Preferred constructors for default construction? bd中的默认首选构造函数
// 在RootBeanDefinition中默认返回了null,表示默认使用无参的构造函数
ctors = mbd.getPreferredConstructors();
if (ctors != null) {
    return autowireConstructor(beanName, mbd, ctors, null);
}
// No special handling: simply use no-arg constructor.
// 使用无参的构造函数创建bean对象
return instantiateBean(beanName, mbd);

有参的构造函数调用

调用有参的构造函数ConstructorResolver#autowireConstructor

  • beanName bean对象的名称
  • mbd bean合并后的bd
  • chosenCtors 选中执行的构造函数
  • explicitArgs 使用bd中的构造函数参数
public BeanWrapper autowireConstructor(String beanName, RootBeanDefinition mbd,
            Constructor<?>[] chosenCtors, Object[] explicitArgs) {
    BeanWrapperImpl bw = new BeanWrapperImpl();
    this.beanFactory.initBeanWrapper(bw);
    // 将要使用的构造函数
    Constructor<?> constructorToUse = null;
    // 构造函数的形参列表
    ArgumentsHolder argsHolderToUse = null;
    // 构造函数的实际参数
    Object[] argsToUse = null;
    // explicitArgs 是传递过来的实际参数
    if (explicitArgs != null) {
        argsToUse = explicitArgs;
    }else {
        Object[] argsToResolve = null;
        synchronized (mbd.constructorArgumentLock) {
            // 首先从缓存中获取构造函数和实际参数
            constructorToUse = (Constructor<?>) mbd.resolvedConstructorOrFactoryMethod;
            if (constructorToUse != null && mbd.constructorArgumentsResolved) {
                // Found a cached constructor...
                argsToUse = mbd.resolvedConstructorArguments;
                if (argsToUse == null) {
                    argsToResolve = mbd.preparedConstructorArguments;
                }
            }
        }
        if (argsToResolve != null) {
            // 缓存中存在才会在这里解析实参
            argsToUse = resolvePreparedArguments(beanName, mbd, bw, constructorToUse, argsToResolve, true);
        }
    }
    // 缓存中没有获取到或者没有实参,都会进入这个if
    if (constructorToUse == null || argsToUse == null) {
        // Take specified constructors, if any.
        Constructor<?>[] candidates = chosenCtors;
        if (candidates == null) {
            Class<?> beanClass = mbd.getBeanClass();
            candidates = (mbd.isNonPublicAccessAllowed() ? 
                          beanClass.getDeclaredConstructors() : 
                          beanClass.getConstructors());
        }
        // mbd中不存在构造函数参数,并且只有一个构造函数
        if (candidates.length == 1 && explicitArgs == null && !mbd.hasConstructorArgumentValues()) {
            Constructor<?> uniqueCandidate = candidates[0];
            if (uniqueCandidate.getParameterCount() == 0) {
                // 直接调用这个构造函数,不需要去进行构造函数的匹配
                synchronized (mbd.constructorArgumentLock) {
                    mbd.resolvedConstructorOrFactoryMethod = uniqueCandidate;
                    mbd.constructorArgumentsResolved = true;
                    mbd.resolvedConstructorArguments = EMPTY_ARGS;
                }
                bw.setBeanInstance(instantiate(beanName, mbd, uniqueCandidate, EMPTY_ARGS));
                return bw;
            }
        }
        // 这里表示存在多个构造函数,需要进行构造函数的匹配
        // Need to resolve the constructor.一般是这样的 autowiring == true
        boolean autowiring = (chosenCtors != null ||
                              mbd.getResolvedAutowireMode() == AutowireCapableBeanFactory.AUTOWIRE_CONSTRUCTOR);
        ConstructorArgumentValues resolvedValues = null;
        // 最少的参数个数,是根据这个方法传入的实参确定
        // 如果传入的实参为null,使用 resolveConstructorArguments 方法来确定最少参数个数
        int minNrOfArgs;
        if (explicitArgs != null) {
            minNrOfArgs = explicitArgs.length;
        }else {
            ConstructorArgumentValues cargs = mbd.getConstructorArgumentValues();
            resolvedValues = new ConstructorArgumentValues();
            minNrOfArgs = resolveConstructorArguments(beanName, mbd, bw, cargs, resolvedValues);
        }
        /**
         * 对传入的构造函数集合进行排序
         * 安装访问权限由大到小,参数由多到少的顺序排序
         */
        AutowireUtils.sortConstructors(candidates);
        // 参数和构造方法的差异权重,越小代表越接近
        int minTypeDiffWeight = Integer.MAX_VALUE;
        Set<Constructor<?>> ambiguousConstructors = null;
        LinkedList<UnsatisfiedDependencyException> causes = null;
        for (Constructor<?> candidate : candidates) {
            // 拿到构造函数的参数类型数组 1、拿到形参个数 2、拿到新参类型数组
            Class<?>[] paramTypes = candidate.getParameterTypes();
            // 这个判断满足了,说明已经找到了合适的构造方法
            // 因为构造方法按照新参个数进行排序了,如果当前的构造函数的形参个数小于使用参数的个数
            // 那么在向下寻找也只会找到更少形参的构造方法,所以在这里直接 break 了
            if (constructorToUse != null && argsToUse != null && argsToUse.length > paramTypes.length) {
                // Already found greedy constructor that can be satisfied ->
                // do not look any further, there are only less greedy constructors left.
                break;
            }
            // 构造函数的形参个数小于了最小要求的实参个数
            // 比如实参有2个参数,这里直接过滤掉1个参数的构造函数
            if (paramTypes.length < minNrOfArgs) {
                continue;
            }
            // 封装形参和实现的对应关系
            ArgumentsHolder argsHolder;
            if (resolvedValues != null) {
                try {
                    // 获取形参的名称
                    String[] paramNames = ConstructorPropertiesChecker.evaluate(candidate, paramTypes.length);
                    if (paramNames == null) {
                        ParameterNameDiscoverer pnd = this.beanFactory.getParameterNameDiscoverer();
                        if (pnd != null) {
                            paramNames = pnd.getParameterNames(candidate);
                        }
                    }
                    argsHolder = createArgumentArray(beanName, mbd, 
                                          resolvedValues, bw, 
                                          paramTypes, paramNames,                                                                   getUserDeclaredConstructor(candidate), 
                                          autowiring, candidates.length == 1);
                }catch (UnsatisfiedDependencyException ex) {
                    // Swallow and try next constructor.
                    if (causes == null) {
                        causes = new LinkedList<>();
                    }
                    causes.add(ex);
                    continue;
                }
            }else {
                // Explicit arguments given -> arguments length must match exactly.
                if (paramTypes.length != explicitArgs.length) {
                    continue;
                }
                argsHolder = new ArgumentsHolder(explicitArgs);
            }
            // 类型差异权重
            int typeDiffWeight = (mbd.isLenientConstructorResolution() ?
                                  argsHolder.getTypeDifferenceWeight(paramTypes) : argsHolder.getAssignabilityWeight(paramTypes));
            // Choose this constructor if it represents the closest match.
            // 找到最小差异的构造方法和参数
            if (typeDiffWeight < minTypeDiffWeight) {
                constructorToUse = candidate;
                argsHolderToUse = argsHolder;
                argsToUse = argsHolder.arguments;
                minTypeDiffWeight = typeDiffWeight;
                ambiguousConstructors = null;
            } else if (constructorToUse != null && typeDiffWeight == minTypeDiffWeight) {
                if (ambiguousConstructors == null) {
                    ambiguousConstructors = new LinkedHashSet<>();
                    ambiguousConstructors.add(constructorToUse);
                }
                ambiguousConstructors.add(candidate);
            }
        }
        // 缓存构造函数和解析出来的参数
        if (explicitArgs == null && argsHolderToUse != null) {
            argsHolderToUse.storeCache(mbd, constructorToUse);
        }
    }
    Assert.state(argsToUse != null, "Unresolved constructor arguments");
    // 调用构造函数、实际参数创建对象
    bw.setBeanInstance(instantiate(beanName, mbd, constructorToUse, argsToUse));
    return bw;
}

最终调用SimpleInstantiationStrategy#instantiate判断如何进行实例化

  • 如果不存在方法的复写(lookup-method / replace-method )那么使用反射创建对象
  • 如果存在方法的复写,使用cglib进行实例化
// 反射调用构造方法
BeanUtils.instantiateClass(constructorToUse);
// 使用cglib
CglibSubclassingInstantiationStrategy#instantiateWithMethodInjection
// 最终返回
return new CglibSubclassCreator(bd, owner).instantiate(ctor, args);
// CglibSubclassingInstantiationStrategy.CglibSubclassCreator#instantiate
public Object instantiate(@Nullable Constructor<?> ctor, Object... args) {
    Class<?> subclass = createEnhancedSubclass(this.beanDefinition);
    Object instance;
    // ctor如果是null表示使用默认的无参构造函数,这里是和下面调用无参构造函数共用
    if (ctor == null) {
        instance = BeanUtils.instantiateClass(subclass);
    }else {
        Constructor<?> enhancedSubclassConstructor = subclass.getConstructor(ctor.getParameterTypes());
        instance = enhancedSubclassConstructor.newInstance(args);
    }
    // enhanced class (via the Enhancer) in order to avoid memory leaks.
    Factory factory = (Factory) instance;
    // 设置方法的拦截器
    factory.setCallbacks(new Callback[] {NoOp.INSTANCE,
                                         new LookupOverrideMethodInterceptor(this.beanDefinition, this.owner),
                                         new ReplaceOverrideMethodInterceptor(this.beanDefinition, this.owner)});
    return instance;
}

LookupOverrideMethodInterceptor:拦截@Lookup/lookup-method注解的方法(LookupOverride)

ReplaceOverrideMethodInterceptor:拦截配置的replace-method的方法(ReplaceOverride)

LookupOverrideMethodInterceptor中的intercept会从容器中取出这个Lookup方法返回值类型的bean

public Object intercept(Object obj, Method method, Object[] args, MethodProxy mp) {
    // Cast is safe, as CallbackFilter filters are used selectively.
    LookupOverride lo = (LookupOverride) getBeanDefinition().getMethodOverrides().getOverride(method);
    Object[] argsToUse = (args.length > 0 ? args : null); 
    if (StringUtils.hasText(lo.getBeanName())) {
        // owner=BeanFactory  从容器中获取
        return (argsToUse != null ?
                this.owner.getBean(lo.getBeanName(), argsToUse) :
                this.owner.getBean(lo.getBeanName()));
    }else {
        return (argsToUse != null ? 
                this.owner.getBean(method.getReturnType(), argsToUse) :
                this.owner.getBean(method.getReturnType()));
    }
}

ReplaceOverrideMethodInterceptor中的intercept方法会回调配置的MethodReplacer接口实现类的reimplement方法

<bean name="myMethodReplacer" class="top.gmfcj.component.MyMethodReplacer" />
<bean name="test2" class="top.gmfcj.component.UserTest2">
    <replaced-method name="test" replacer="myMethodReplacer" />
</bean>
public Object intercept(Object obj, Method method, Object[] args, MethodProxy mp) {
    ReplaceOverride ro = (ReplaceOverride) getBeanDefinition().getMethodOverrides().getOverride(method);
    Assert.state(ro != null, "ReplaceOverride not found");
    // TODO could cache if a singleton for minor performance optimization
    // 获取replaced-method标签中配置的replacer对应的bean
    MethodReplacer mr = this.owner.getBean(ro.getMethodReplacerBeanName(), MethodReplacer.class);
    // 回调bean的reimplement方法
    return mr.reimplement(obj, method, args);
}

无参的构造函数调用

调用无参的构造函数AbstractAutowireCapableBeanFactory#instantiateBean

protected BeanWrapper instantiateBean(final String beanName, final RootBeanDefinition mbd) {
    Object beanInstance;
    final BeanFactory parent = this;
    // 安全检查
    if (System.getSecurityManager() != null) {
       // 授权给当前类一个 getInstantiationStrategy().instantiate这个方法的执行权限
    }else {
        // 实例化 => SimpleInstantiationStrategy.instantiate
        beanInstance = getInstantiationStrategy().instantiate(mbd, beanName, parent);
    }
    // 包裹真实的对象并返回包装对象
    BeanWrapper bw = new BeanWrapperImpl(beanInstance);
    initBeanWrapper(bw);
    return bw;
}

实例化对象的逻辑SimpleInstantiationStrategy.instantiate

public Object instantiate(RootBeanDefinition bd, String beanName, BeanFactory owner) {
    // 如果不存在方法的复写(lookup-method / replace-method )那么使用反射创建对象
    // 存在方法复写,使用cglib进行实例化
    if (!bd.hasMethodOverrides()) {
        Constructor<?> constructorToUse;
        synchronized (bd.constructorArgumentLock) {
            // constructorToUse 表示用来实例对象的构造方法
            constructorToUse = (Constructor<?>) bd.resolvedConstructorOrFactoryMethod;
            if (constructorToUse == null) {
                final Class<?> clazz = bd.getBeanClass();
                if (clazz.isInterface()) {
                    throw new BeanInstantiationException(clazz, "Specified class is an interface");
                }
                bd.resolvedConstructorOrFactoryMethod = constructorToUse;
            }
        }
        // 通过构造函数实例出对象 => constructorToUse.newInstance(args)
        return BeanUtils.instantiateClass(constructorToUse);
    }else {
        // Must generate CGLIB subclass. 使用CGlib生成beanClass的子类,返回的是子类的对象
        // 这里逻辑和调用有参的构造函数创建cglib代理一致,只是传入的构造函数是一个null值
        return instantiateWithMethodInjection(bd, beanName, owner);
    }
}

至此,对象的实例化就完成了。

回调BeanPostProcessor(三)

当创建的非FactoryBean的对象创建完成之后,会返回一个BeanWrapper,这个类包装了bean对象。

// AbstractAutowireCapableBeanFactory#doCreateBean
if (instanceWrapper == null) {
    // 说明不是FactoryBean,在这里实例化bean
    instanceWrapper = createBeanInstance(beanName, mbd, args);
}
// bean是真实的对象
final Object bean = instanceWrapper.getWrappedInstance();
Class<?> beanType = instanceWrapper.getWrappedClass();

第三次后置处理器的回调:

  • 实例化对象之后,仅仅只是完成了类的初始化工作,还未涉及到数据的填充,初始化方法的执行。这里的回调就是缓存bean中的生命周期函数,解析@Resource @Inject @Autowired等注解,为后续注入依赖,回调声明周期函数做准备

  • 虽然会调用三个类的postProcessMergedBeanDefinition方法,但是主要看CommonAnnotationBeanPostProcessor#postProcessMergedBeanDefinition和AutowiredAnnotationBeanPostProcessor#postProcessMergedBeanDefinition

  • CommonAnnotationBeanPostProcessor:主要缓存bean的声明周期函数,bean内部的@Resource的注解

  • AutowiredAnnotationBeanPostProcessor:缓存一个类中所有@Autowired、@Value和@Inject注解的方法和字段。

applyMergedBeanDefinitionPostProcessors(mbd, beanType, beanName);
// AbstractAutowireCapableBeanFactory#applyMergedBeanDefinitionPostProcessors
protected void applyMergedBeanDefinitionPostProcessors(RootBeanDefinition mbd, Class<?> beanType, String beanName) {
    for (BeanPostProcessor bp : getBeanPostProcessors()) {
        if (bp instanceof MergedBeanDefinitionPostProcessor) {
            // CommonAnnotationBeanPostProcessor
            // AutowiredAnnotationBeanPostProcessor
            // ApplicationListenerDetector
            MergedBeanDefinitionPostProcessor bdp = (MergedBeanDefinitionPostProcessor) bp;
            bdp.postProcessMergedBeanDefinition(mbd, beanType, beanName);
        }
    }
}

CommonAnnotationBeanPostProcessor

public void postProcessMergedBeanDefinition(RootBeanDefinition beanDefinition, Class<?> beanType, String beanName) {
    // 检查@PostConstruct/@PreDestroy
    super.postProcessMergedBeanDefinition(beanDefinition, beanType, beanName);
    // 检查@Resource注解
    InjectionMetadata metadata = findResourceMetadata(beanName, beanType, null);
    metadata.checkConfigMembers(beanDefinition);
}

获取bean对象的中的生命周期回调函数

// 父类的postProcessMergedBeanDefinition方法
public void postProcessMergedBeanDefinition(RootBeanDefinition beanDefinition, Class<?> beanType, String beanName) {
    // 拿到 bd 对应的生命周期回调信息
    LifecycleMetadata metadata = findLifecycleMetadata(beanType);
    metadata.checkConfigMembers(beanDefinition);
}
private LifecycleMetadata findLifecycleMetadata(Class<?> clazz) {
    if (this.lifecycleMetadataCache == null) {
        // 初始化缓存map对象
        return buildLifecycleMetadata(clazz);
    }
    // 从Map获取生命周期回调的元数据,包括回调的类 初始方法集合  销毁方法集合
    // 第一次获取肯定是没有的
    LifecycleMetadata metadata = this.lifecycleMetadataCache.get(clazz);
    if (metadata == null) {
        synchronized (this.lifecycleMetadataCache) {
            metadata = this.lifecycleMetadataCache.get(clazz);
            if (metadata == null) {
                // 没有就创建(检查类中的注解,核心方法),并放入map中
                metadata = buildLifecycleMetadata(clazz);
                // 获取到回调函数之后会缓存起来
                this.lifecycleMetadataCache.put(clazz, metadata);
            }
            return metadata;
        }
    }
    return metadata;
}

关键方法

  • 在CommonAnnotationBeanPostProcessor对象创建时调用了父类的set方法, 设置了InitAnnotation和DestroyAnnotation
public CommonAnnotationBeanPostProcessor() {
    setOrder(Ordered.LOWEST_PRECEDENCE - 3);
    setInitAnnotationType(PostConstruct.class);
    setDestroyAnnotationType(PreDestroy.class);
    ignoreResourceType("javax.xml.ws.WebServiceContext");
}
  • 遍历类中的所有方法,判断是否存在initAnnotationType或者destroyAnnotationType
do {
    ReflectionUtils.doWithLocalMethods(targetClass, method -> {
        if (this.initAnnotationType != null && method.isAnnotationPresent(this.initAnnotationType)) {
            LifecycleElement element = new LifecycleElement(method);
            currInitMethods.add(element);
        }
        if (this.destroyAnnotationType != null && method.isAnnotationPresent(this.destroyAnnotationType)) {
            currDestroyMethods.add(new LifecycleElement(method));
        }
    });
    initMethods.addAll(0, currInitMethods);
    destroyMethods.addAll(currDestroyMethods);
}while (targetClass != null && targetClass != Object.class);
return new LifecycleMetadata(clazz, initMethods, destroyMethods);
  • @Resource注解和@Autowired注解处理与上述过程类似,这两者解析的结果都被缓存在一个相同的Map(injectionMetadataCache)中。在下次依赖注入时直接从缓存中就可以获取到依赖信息。

回调BeanPostProcessor(四)

第四次回调的作用:在对象没有放入singletonObjects之前调用,返回一个提前暴露的对象,在这个后置处理器中可以对这个bean进行修改

// AbstractAutowireCapableBeanFactory#doCreateBean
boolean earlySingletonExposure = (mbd.isSingleton() && this.allowCircularReferences &&
                                  isSingletonCurrentlyInCreation(beanName));
if (earlySingletonExposure) {
    // getEarlyBeanReference 后置处理器的第四次回调  => AutowiredAnnotationBeanPostProcessor
    // SmartInstantiationAwareBeanPostProcessor 的 getEarlyBeanReference 方法
    addSingletonFactory(beanName, () -> getEarlyBeanReference(beanName, mbd, bean));
}

关键在SmartInstantiationAwareBeanPostProcessor#getEarlyBeanReference

// SmartInstantiationAwareBeanPostProcessor#getEarlyBeanReference
public Object getEarlyBeanReference(Object bean, String beanName) throws BeansException {
    return bean;
}

这里的addSingletonFactory方法非常的关键,主要是为了解决Spring中循环依赖的问题。

接下来就是解决Bean对象的属性装配和生命周期函数回调

// AbstractAutowireCapableBeanFactory#doCreateBean
Object exposedObject = bean;
// 解决属性装配
populateBean(beanName, mbd, instanceWrapper);
// 调用 BeanPostProcessor实现类的方法和自定义的初始化回调方法
exposedObject = initializeBean(beanName, exposedObject, mbd);

回调BeanPostProcessor(五)

第五次回调后置处理器,断定是否需要进行属性装配,当然默认的都是返回true,表示会自动装配。

// AbstractAutowireCapableBeanFactory#populateBean
if (!mbd.isSynthetic() && hasInstantiationAwareBeanPostProcessors()) {
    for (BeanPostProcessor bp : getBeanPostProcessors()) {
        if (bp instanceof InstantiationAwareBeanPostProcessor) {
            // CommonAnnotationBeanPostProcessor.postProcessAfterInstantiation() => return true
            // AutowiredAnnotationBeanPostProcessor.postProcessAfterInstantiation() => return true
            InstantiationAwareBeanPostProcessor ibp = (InstantiationAwareBeanPostProcessor) bp;
            if (!ibp.postProcessAfterInstantiation(bw.getWrappedInstance(), beanName)) {
                continueWithPropertyPopulation = false;
                break;
            }
        }
    }
}
// 如果不自动注入,在这里就直接返回了
if (!continueWithPropertyPopulation) {
    return;
}

选择Bean的装配模式

PropertyValues pvs = (mbd.hasPropertyValues() ? mbd.getPropertyValues() : null);
// 选择bd的装配模式,默认是 AUTOWIRE_NO
if (mbd.getResolvedAutowireMode() == AUTOWIRE_BY_NAME || mbd.getResolvedAutowireMode() == AUTOWIRE_BY_TYPE) {
    MutablePropertyValues newPvs = new MutablePropertyValues(pvs);
    // Add property values based on autowire by name if applicable.
    if (mbd.getResolvedAutowireMode() == AUTOWIRE_BY_NAME) {
        autowireByName(beanName, mbd, bw, newPvs);
    }
    // Add property values based on autowire by type if applicable.
    if (mbd.getResolvedAutowireMode() == AUTOWIRE_BY_TYPE) {
        autowireByType(beanName, mbd, bw, newPvs);
    }
    pvs = newPvs;
}

回调BeanPostProcessor(六)

第六次回调正式进行属性的注入

// 检查是否存在 InstantiationAwareBeanPostProcessor 接口的实现类
boolean hasInstAwareBpps = hasInstantiationAwareBeanPostProcessors();
// 是否需要深度检查
boolean needsDepCheck = (mbd.getDependencyCheck() != AbstractBeanDefinition.DEPENDENCY_CHECK_NONE);
PropertyDescriptor[] filteredPds = null;
if (hasInstAwareBpps) {
    if (pvs == null) {
        pvs = mbd.getPropertyValues();
    }
    // 第六次后置处理器的回调 InstantiationAwareBeanPostProcessor 的 postProcessProperties 方法,处理属性
    // CommonAnnotationBeanPostProcessor 处理 Resource 注解
    // AutowiredAnnotationBeanPostProcessor 处理 Autowired Inject注解
    for (BeanPostProcessor bp : getBeanPostProcessors()) {
        if (bp instanceof InstantiationAwareBeanPostProcessor) {
            InstantiationAwareBeanPostProcessor ibp = (InstantiationAwareBeanPostProcessor) bp;
            PropertyValues pvsToUse = ibp.postProcessProperties(pvs, bw.getWrappedInstance(), beanName);
            if (pvsToUse == null) {
                if (filteredPds == null) {
                    filteredPds = filterPropertyDescriptorsForDependencyCheck(bw, mbd.allowCaching);
                }
                pvsToUse = ibp.postProcessPropertyValues(pvs, filteredPds, bw.getWrappedInstance(), beanName);
                if (pvsToUse == null) {
                    return;
                }
            }
            pvs = pvsToUse;
        }
    }
}
if (needsDepCheck) {
    if (filteredPds == null) {
        filteredPds = filterPropertyDescriptorsForDependencyCheck(bw, mbd.allowCaching);
    }
    // 深度检查依赖
    checkDependencies(beanName, mbd, filteredPds, pvs);
}

if (pvs != null) {
    applyPropertyValues(beanName, mbd, bw, pvs);
}

回调BeanPostProcessor(七、八)

第七次回调是在属性注入完成之后,执行BeanPostProcessor的 postProcessBeforeInitialization 方法,也就是init方法或者@PostConstruct方法。

第八次回调,执行BeanPostProcessor的postProcessAfterInitialization 方法,AOP的代理对象就是在这里生成的。

// AbstractAutowireCapableBeanFactory#initializeBean
protected Object initializeBean(final String beanName, final Object bean, RootBeanDefinition mbd) {
    // 回调 BeanNameAware/BeanClassLoaderAware/BeanFactoryAware 接口
    invokeAwareMethods(beanName, bean);
    Object wrappedBean = bean;
    if (mbd == null || !mbd.isSynthetic()) {
        // 第七次调用
        // beanPostProcessor 的 beanPostProcessorsBeforeInitialization 方法
        wrappedBean = applyBeanPostProcessorsBeforeInitialization(wrappedBean, beanName);
    }
    // 调用init方法  @PostConstruct
    invokeInitMethods(beanName, wrappedBean, mbd);
    if (mbd == null || !mbd.isSynthetic()) {
        // beanPostProcessor 的 beanPostProcessorsAfterInitialization 方法
        // AOP的代理对象就是在这里生成的
        // 第八次调用
        wrappedBean = applyBeanPostProcessorsAfterInitialization(wrappedBean, beanName);
    }
    return wrappedBean;
}

回调BeanPostProcessor(九)

第九次回调,在bean销毁之前的回调,在容器关闭时调用(context.close())

// DisposableBeanAdapter#destroy
public void destroy() {
    if (!CollectionUtils.isEmpty(this.beanPostProcessors)) {
        // 处理系统中的 DestructionAwareBeanPostProcessor 接口 回调beanPostProcessor
        for (DestructionAwareBeanPostProcessor processor : this.beanPostProcessors) {
            processor.postProcessBeforeDestruction(this.bean, this.beanName);
        }
    }
    if (this.destroyMethod != null) {
        // 处理销毁方法回调 @PreDestroy
        invokeCustomDestroyMethod(this.destroyMethod);
    }
    else if (this.destroyMethodName != null) {
        // 调用指定的销毁回调方法
        Method methodToInvoke = determineDestroyMethod(this.destroyMethodName);
        if (methodToInvoke != null) {
            invokeCustomDestroyMethod(ClassUtils.getInterfaceMethodIfPossible(methodToInvoke));
        }
    }
}

总结:在spring容器中,BeanPostProcessor一共回调了九次

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,717评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,501评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,311评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,417评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,500评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,538评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,557评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,310评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,759评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,065评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,233评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,909评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,548评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,172评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,420评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,103评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,098评论 2 352

推荐阅读更多精彩内容