Spring 中原型bean的使用

midoll 458 2022-07-14

Spring 中原型bean的使用

前言

我们知道在Spring当中bean的Scope类型包含有单例(singleton)和多例(prototype),后者又叫原型。当然,还有request类型、session类型等。其中单例类型是单个Spring容器只存在一个,而原型类型每次从容器中获取都会生成一个新的,因此从容器生命周期来看,单例类型的bean可以认为是longer-lived scoped bean(长生命周期bean),而原型类型的bean可以认为是shorter-lived scoped bean(短生命周期bean)。由此可以引出一个问题:在单例bean当中注入原型bean之后,这个原型bean还是原型bean吗?很多人可能会问,这个问题有意义吗?平时根本碰不到。其实很多人对于单例bean和原型bean适用什么样的场景是分不清的,好像单例bean与原型bean的区别只不过一个是存储在单例缓存中,所以每次获取都是一样的,而原型bean不存在啥缓存,所以每次从容器获取都会重新创建一个。但这是从结果来分析的,还没说清楚单例bean和原型bean的适用场景。在官方文档中有如下这句话:

As a rule, you should use the prototype scope for all stateful beans and the singleton scope for stateless beans.

这句话可以说很好的总结了单例bean和原型bean的适用场景,前者适用有状态的,后者适用无状态的。啥是有状态的?啥是无状态的?比如常见的服务类,除了依赖的bean和方法之外没有其他记录状态的属性,假如在一个服务类中存在一个boolean类型的属性,还将其作为单例bean,这样就变为有状态的了,因为这个boolean类型属性随时可能被另一个使用者改变,此时为了隔离,就必须考虑每个使用者使用一个了,也就是要使用原型bean了。我们平时之所以很少遇到单例bean注入多例bean的问题的主要原因就是我们平时设计的控制层、业务层和数据层尽量屏蔽了有状态。所以上面的问题其实要回答的是,在单例bean当中注入一个原型bean之后,这个原型bean之中的状态是共享的吗?

ApplicationContext#getBean()与@Autowired的简要对比

@Autowired getBean()
是否触发依赖注入 是。如果注入的对象还未注册到容器,则会先注册它。 否。如果注入的对象还未注册到容器,不会去注册它,只会获得一个null。
单例与多例 默认是单例。就算是将bean加注解改为多例,此时注入仍为单例。 默认是单例。但可将bean加注解改为多例,此时getBean()获取即为多例。

一、结果分析

首先定义一个原型bean,在这个原型bean当中,有一个stateCount属性,每次调用print都会将这个属性加一并输出。对于原型bean,每次都获取一个新的,这个方法只会输出1.


package org.spring.boot.example.entity;

import org.springframework.context.annotation.Scope;
import org.springframework.stereotype.Component;

import java.util.concurrent.atomic.AtomicInteger;

@Scope(scopeName = "prototype")
@Component
public class PrototypeInstance {

    AtomicInteger stateCount = new AtomicInteger(0);

    public void print() {
        System.out.println(stateCount.incrementAndGet());
    }

}

再定义一个单例bean,并注入上面的原型bean,在这个单例的printCode方法中调用原型bean的方法。


package org.spring.boot.example.entity;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;

@Component
public class SigletonInstance {

    @Autowired
    private PrototypeInstance prototypeInstance;

    public void setPrototypeInstance(PrototypeInstance prototypeInstance) {
        this.prototypeInstance = prototypeInstance;
    }

    public PrototypeInstance getPrototypeInstance() {
        return prototypeInstance;
    }

    public void printCode() {
        prototypeInstance.print();
    }
}

从容器中获取这个单例bean,并多次调用其中的printCode。
image-1689320079491
从结果来看,这个状态变量其实是共享的,也就是说这个单例bean里面的这个原型bean其实只有一个。仔细想一下,在这个单例bean在容器中只会创建一次,也就是说注入原型bean也只有一次机会,所以单例bean当中的prototypeInstance属性只可能是一个。所以最后注入的原型bean其实变成了一个单例bean了。那么这是我们想要的结果吗?当然不是,我们期待的结果是,每次调用prototypeInstance这个属性的方法,应该不会共享状态的。那么,可以办到吗?
image-1689320091566
像上面这样,我们在这个原型bean上面的Scope注解中添加属性值proxyMode = ScopedProxyMode.TARGET_CLASS,然后再次调用。
image-1689320127294
从结果来看,此时这个原型bean成了真正的原型bean,不再共享变量了。达到了我们的目标,那么,这里面的玄机是什么呢?

二、原理分析

按照道理来说,SigletonInstance单例bean中注入PrototypeInstance原型bean只有一次机会,这个prototypeInstance属性只可能是一个对象,而不可能每次都不一样,所以这个对象肯定不能是一个真实的PrototypeInstance对象,而是一个代理对象,这样才能在每次调用其方法的时候有机会被改变。那么Spring中是怎么做到的呢?首先Spring容器中为这个原型bean保存了两个bean实例。一个名称为prototypeInstance,另一个名称为scopedTarget.prototypeInstance,在前面的名称前面加了scopedTarget…如下图所示
image-1689320139624
仔细看下这两个bean的实际类型,前者为org.springframework.aop.scope.ScopedProxyFactoryBean,后者为org.spring.boot.example.entity.PrototypeInstance,是不是很奇怪,其实Spring就是将真实的原型bean改了名字,而原来的名字对应了一个FactoryBean。而获取这个FactoryBean的真实结果是通过getObject方法来获取的,如下所示


@Override
public Object getObject() {
	if (this.proxy == null) {
		throw new FactoryBeanNotInitializedException();
	}
	return this.proxy;
}

image-1689320161019

那么这个proxy属性是啥呢?其实这个proxy就是通过CGLIB生成的一个代理对象,在生成这个代理对象的时候会设置targetSource为SimpleBeanTargetSource类型的对象。上面的proxy和这个targetSource都是上面FactoryBean实例的属性,在调用这个代理的时候会从targetSource中获取到真实对象,然后调用真实对象的属性,Spring只要在这个时候从beanFactory中获取原型bean(此时的bean名称为scopedTarget.prototypeInstance),那么每次方法调用都是不一样的。而Spring正是这么实现的。
image-1689320185989
这个SimpleBeanTargetSource中属性如上所示,而getTarget方法源码如下


public class SimpleBeanTargetSource extends AbstractBeanFactoryBasedTargetSource {

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

}

在每次调用代理的方法时候,会进入到org.springframework.aop.framework.CglibAopProxy.DynamicAdvisedInterceptor#intercept方法中,如下所示
image-1689320196133
不难看出,这里就是通过注入一个代理对象,在每次调用这个代理对象的时候,获取一个原型Bean,这样就相当于注入了一个原型bean。

三、流程分析

1、注册阶段

在Spring扫描注册bean的时候会进入到方法ClassPathBeanDefinitionScanner#doScan当中,这个方法当中有如下代码

definitionHolder = AnnotationConfigUtils.applyScopedProxyMode(scopeMetadata, definitionHolder, this.registry);

而这个applyScopedProxyMode方法就会根据类上面的ScopedProxyMode执行不同的策略,默认值为ScopedProxyMode.NO


static BeanDefinitionHolder applyScopedProxyMode(
		ScopeMetadata metadata, BeanDefinitionHolder definition, BeanDefinitionRegistry registry) {

	ScopedProxyMode scopedProxyMode = metadata.getScopedProxyMode();
	if (scopedProxyMode.equals(ScopedProxyMode.NO)) {
		return definition;
	}
	boolean proxyTargetClass = scopedProxyMode.equals(ScopedProxyMode.TARGET_CLASS);
	return ScopedProxyCreator.createScopedProxy(definition, registry, proxyTargetClass);
}

从上面源码可以看出,只有当ScopedProxyMode为TARGET_CLASS才会进入到下面创建ScopedProxy的逻辑,这也是为啥我们在第一节中在原型bean的scope注解中添加proxyMode = ScopedProxyMode.TARGET_CLASS的原因,只有修改了proxyMode才会进入下面的逻辑。


public static BeanDefinitionHolder createScopedProxy(
		BeanDefinitionHolder definitionHolder, BeanDefinitionRegistry registry, boolean proxyTargetClass) {

	return ScopedProxyUtils.createScopedProxy(definitionHolder, registry, proxyTargetClass);
}

而createScopedProxy的逻辑其实就是将原来要注册的原型bean改个名字注册,另外注册一个ScopedProxyFactoryBean类型的bean并使用原来原型bean的名称。对应的源码如下


/**
 * Generate a scoped proxy for the supplied target bean, registering the target
 * bean with an internal name and setting 'targetBeanName' on the scoped proxy.
 * @param definition the original bean definition
 * @param registry the bean definition registry
 * @param proxyTargetClass whether to create a target class proxy
 * @return the scoped proxy definition
 * @see #getTargetBeanName(String)
 * @see #getOriginalBeanName(String)
 */
public static BeanDefinitionHolder createScopedProxy(BeanDefinitionHolder definition,
		BeanDefinitionRegistry registry, boolean proxyTargetClass) {

	String originalBeanName = definition.getBeanName();
	BeanDefinition targetDefinition = definition.getBeanDefinition();
	String targetBeanName = getTargetBeanName(originalBeanName);

	// Create a scoped proxy definition for the original bean name,
	// "hiding" the target bean in an internal target definition.
	RootBeanDefinition proxyDefinition = new RootBeanDefinition(ScopedProxyFactoryBean.class);
	proxyDefinition.setDecoratedDefinition(new BeanDefinitionHolder(targetDefinition, targetBeanName));
	proxyDefinition.setOriginatingBeanDefinition(targetDefinition);
	proxyDefinition.setSource(definition.getSource());
	proxyDefinition.setRole(targetDefinition.getRole());

	proxyDefinition.getPropertyValues().add("targetBeanName", targetBeanName);
	if (proxyTargetClass) {
		targetDefinition.setAttribute(AutoProxyUtils.PRESERVE_TARGET_CLASS_ATTRIBUTE, Boolean.TRUE);
		// ScopedProxyFactoryBean's "proxyTargetClass" default is TRUE, so we don't need to set it explicitly here.
	}
	else {
		proxyDefinition.getPropertyValues().add("proxyTargetClass", Boolean.FALSE);
	}

	// Copy autowire settings from original bean definition.
	proxyDefinition.setAutowireCandidate(targetDefinition.isAutowireCandidate());
	proxyDefinition.setPrimary(targetDefinition.isPrimary());
	if (targetDefinition instanceof AbstractBeanDefinition) {
		proxyDefinition.copyQualifiersFrom((AbstractBeanDefinition) targetDefinition);
	}

	// The target bean should be ignored in favor of the scoped proxy.
	targetDefinition.setAutowireCandidate(false);
	targetDefinition.setPrimary(false);

	// Register the target bean as separate bean in the factory.
	registry.registerBeanDefinition(targetBeanName, targetDefinition);

	// Return the scoped proxy definition as primary bean definition
	// (potentially an inner bean).
	return new BeanDefinitionHolder(proxyDefinition, originalBeanName, definition.getAliases());
}

在这里将targetDefinition的AutowireCandidate和Primary属性设置为false,这样在自动注入的时候就不会考虑原型bean了,而是对应的FactoryBean.

2、实例化阶段

在容器启动过程中,会实例化单例bean,这个时候需要注入这个原型bean,其实会触发ScopedProxyFactoryBean类型bean的初始化。我们看一下这个bean的定义。
image-1689320258079
其中ProxyConfig用于为生成代理提供配置,AopInfrastructureBean是一个标识接口,标识这个是Spring的AOP的一个基础结构,因此这个类不会被自动代理,即使被匹配上,也就是说这个类在Spring AOP框架中是不会再生成其他的代理的。另外实现了FactoryBean接口,这样这个bean的真实对象是通过getObject方法返回的,另外这个类实现了BeanFactoryAware接口,这样在这个bean初始化的时候会调用回调方法setBeanFactory,而这个回调方法也是生成代理的核心,如下所示


@Override
public void setBeanFactory(BeanFactory beanFactory) {
	if (!(beanFactory instanceof ConfigurableBeanFactory)) {
		throw new IllegalStateException("Not running in a ConfigurableBeanFactory: " + beanFactory);
	}
	ConfigurableBeanFactory cbf = (ConfigurableBeanFactory) beanFactory;

	this.scopedTargetSource.setBeanFactory(beanFactory);

	ProxyFactory pf = new ProxyFactory();
	pf.copyFrom(this);
	pf.setTargetSource(this.scopedTargetSource);

	Assert.notNull(this.targetBeanName, "Property 'targetBeanName' is required");
	Class<?> beanType = beanFactory.getType(this.targetBeanName);
	if (beanType == null) {
		throw new IllegalStateException("Cannot create scoped proxy for bean '" + this.targetBeanName +
				"': Target type could not be determined at the time of proxy creation.");
	}
	if (!isProxyTargetClass() || beanType.isInterface() || Modifier.isPrivate(beanType.getModifiers())) {
		pf.setInterfaces(ClassUtils.getAllInterfacesForClass(beanType, cbf.getBeanClassLoader()));
	}

	// Add an introduction that implements only the methods on ScopedObject.
	ScopedObject scopedObject = new DefaultScopedObject(cbf, this.scopedTargetSource.getTargetBeanName());
	pf.addAdvice(new DelegatingIntroductionInterceptor(scopedObject));

	// Add the AopInfrastructureBean marker to indicate that the scoped proxy
	// itself is not subject to auto-proxying! Only its target bean is.
	pf.addInterface(AopInfrastructureBean.class);

	this.proxy = pf.getProxy(cbf.getBeanClassLoader());
}

这里不但生成了代理,而且设置了SimpleBeanTargetSource,这样后面代理才能获取到真实的目标对象。最后实例化的对象如下所示
image-1689320279552
注册到单例bean中的其实就是对象中的proxy属性对象。

3、调用阶段

调用单例bean中多例bean的方法的时候,会调用proxy代理对象的方法,这是一个CGLIB代理对象,所以会首先进入org.springframework.aop.framework.CglibAopProxy.DynamicAdvisedInterceptor#intercept方法中,如下所示


@Override
@Nullable
public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
	Object oldProxy = null;
	boolean setProxyContext = false;
	Object target = null;
	TargetSource targetSource = this.advised.getTargetSource();
	try {
		if (this.advised.exposeProxy) {
			// Make invocation available if necessary.
			oldProxy = AopContext.setCurrentProxy(proxy);
			setProxyContext = true;
		}
		// Get as late as possible to minimize the time we "own" the target, in case it comes from a pool...
		target = targetSource.getTarget();
		Class<?> targetClass = (target != null ? target.getClass() : null);
		List<Object> chain = this.advised.getInterceptorsAndDynamicInterceptionAdvice(method, targetClass);
		Object retVal;
		// Check whether we only have one InvokerInterceptor: that is,
		// no real advice, but just reflective invocation of the target.
		if (chain.isEmpty() && Modifier.isPublic(method.getModifiers())) {
			// We can skip creating a MethodInvocation: just invoke the target directly.
			// Note that the final invoker must be an InvokerInterceptor, so we know
			// it does nothing but a reflective operation on the target, and no hot
			// swapping or fancy proxying.
			Object[] argsToUse = AopProxyUtils.adaptArgumentsIfNecessary(method, args);
			retVal = methodProxy.invoke(target, argsToUse);
		}
		else {
			// We need to create a method invocation...
			retVal = new CglibMethodInvocation(proxy, target, method, args, targetClass, chain, methodProxy).proceed();
		}
		retVal = processReturnType(proxy, target, method, retVal);
		return retVal;
	}
	finally {
		if (target != null && !targetSource.isStatic()) {
			targetSource.releaseTarget(target);
		}
		if (setProxyContext) {
			// Restore old proxy.
			AopContext.setCurrentProxy(oldProxy);
		}
	}
}

其中org.springframework.aop.TargetSource#getTarget就是获取原型bean的关键了。

总结

如果实在xml中配置的话


<!-- DefaultUserPreferences implements the UserPreferences interface -->
<bean id="prototypeInstance" class="org.spring.boot.example.entity.PrototypeInstance" scope="prototype">
    <aop:scoped-proxy proxy-target-class="true"/>
</bean>
<bean id="sigletonInstance" class="org.spring.boot.example.entity.SigletonInstance">
    <property name="prototypeInstance" ref="prototypeInstance"/>
</bean>

除了以上的方式解决单例bean注册原型bean的问题之外,还有另一种方式-方法注入,但是本质其实都是差不多的,都是通过CGLIB动态代理的方式,可以参考博客:Spring注入之方法注入Spring注入之方法注入


# 原型bean