春季原型豆是否需要手动销毁?

我注意到我的原型范围的Spring bean的钩子没有被执行。@PreDestroy

从那以后,我在这里读到这实际上是设计使然。Spring容器将销毁单一豆子,但不会破坏原型豆。我不清楚为什么。如果Spring容器将创建我的原型豆并执行其钩子,为什么当容器关闭时,它也不会破坏我的豆子?一旦我的Spring容器被关闭,继续使用它的任何豆子是否有意义?我看不到您希望在完成其bean之前关闭容器的情况。在容器关闭后,是否有可能继续使用原型Spring bean?@PostConstruct

上面描述了我的主要问题的令人费解的背景,即:如果Spring容器没有破坏原型豆,这是否意味着可能发生内存泄漏?还是原型豆会在某个时候被垃圾收集?

春季文档指出:

客户端代码必须清理原型范围的对象,并释放原型 Bean 所持有的昂贵资源。要让 Spring 容器释放原型范围的 Bean 所持有的资源,请尝试使用自定义 Bean 后处理器,该处理器包含对需要清理的 Bean 的引用。

那是什么意思?文本向我暗示,作为程序员,我负责明确(手动)销毁我的原型豆。这是正确的吗?如果是这样,我该怎么做?


答案 1

为了其他人的利益,我将在下面介绍我从调查中收集到的信息:

只要原型 Bean 本身不保存对另一个资源(如数据库连接或会话对象)的引用,一旦删除了对该对象的所有引用或该对象超出范围,它就会被垃圾回收。因此,通常不需要显式销毁原型豆。

但是,在如上所述可能发生内存泄漏的情况下,可以通过创建单例 Bean 后处理器来销毁原型 Bean,其销毁方法显式调用原型 Bean 的销毁钩子。因为后处理器本身是单例范围的,所以它的破坏钩子将被Spring调用:

  1. 创建一个 Bean 后处理器来处理所有原型 Bean 的销毁。这是必要的,因为Spring不会破坏原型bean,因此代码中的任何@PreDestroy钩子都不会被容器调用。

  2. 实现以下接口:

    1.BeanFactoryAware
    此接口提供了一个接收 Beanfactory 对象的回调方法。此 BeanFactory 对象在后处理器类中用于通过其 BeanFactory.isPrototype(String beanName) 方法来标识所有原型 Bean。

    2. 一次性Bean
    这个接口提供了一个由Spring容器调用的Destroy()回调方法。我们将从此方法中调用所有原型 bean 的 Destroy() 方法。

    3. BeanPostProcessor
    实现此接口提供了对后处理回调的访问,从中,我们准备一个内部 List<>由 Spring 容器实例化的所有原型对象。稍后我们将循环浏览此列表<>销毁我们的每个原型豆。


3. 最后,在每个原型 bean 中实现 DisposableBean 接口,提供此合约所需的 Destroy() 方法。

为了说明这个逻辑,我在下面提供了一些从本文中摘录的代码:

/**
* Bean PostProcessor that handles destruction of prototype beans
*/
@Component
public class DestroyPrototypeBeansPostProcessor implements BeanPostProcessor, BeanFactoryAware, DisposableBean {

    private BeanFactory beanFactory;

    private final List<Object> prototypeBeans = new LinkedList<>();

    @Override
    public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
        return bean;
    }

    @Override
    public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
        if (beanFactory.isPrototype(beanName)) {
            synchronized (prototypeBeans) {
                prototypeBeans.add(bean);
            }
        }
        return bean;
    }

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

    @Override
    public void destroy() throws Exception {
        synchronized (prototypeBeans) {
            for (Object bean : prototypeBeans) {
                if (bean instanceof DisposableBean) {
                    DisposableBean disposable = (DisposableBean)bean;
                    try {
                        disposable.destroy();
                    } catch (Exception e) {
                        e.printStackTrace();
                    }
                }
            }
            prototypeBeans.clear();
        }
    }
}

答案 2

你的答案很好。我还想分享一些关于替代解决方案的注意事项,该解决方案允许原型成员通过使用内部bean由Spring IoC容器生命周期进行本机管理。

我最近写了一个关于内豆的单独问题的答案。内部 Bean 是通过将 Bean 属性值指定为对象来创建的。Bean 定义属性值会自动解析为它们定义的 Bean 的实例(作为托管单例 Bean)。BeanDefinition(inner)

以下 XML 上下文配置元素可用于为将要管理的每个引用创建不同的自动连接 Bean(将在上下文关闭时调用):ForkJoinPool@PreDestroy

<!-- Prototype-scoped bean for creating distinct FJPs within the application -->
<bean id="forkJoinPool" class="org.springframework.beans.factory.support.GenericBeanDefinition" scope="prototype">
    <property name="beanClass" value="org.springframework.scheduling.concurrent.ForkJoinPoolFactoryBean" />
</bean>

但是,此行为取决于作为 Bean 定义的属性值分配的引用。这意味着默认情况下,构造函数注入不与此一起使用,因为这些自动布线方法会立即解析值,而不是使用 AbstractAutowireCapableBeanFactory#applyPropertyValues 中的属性值解析。按类型自动布线也不起作用,因为类型解析不会通过 bean 传播,这些 bean 是用来查找生成的类型。@AutowiredBeanDefinition

仅当满足以下两个条件之一时,此方法才有效:

  • 依赖 Bean 也在 XML 中定义
  • 或者,如果自动连线模式设置为AutowireCapableBeanFactory#AUTOWIRE_BY_NAME

<!-- Setting bean references through XML -->
<beans ...>
    <bean id="myOtherBean" class="com.example.demo.ForkJoinPoolContainer">
        <property name="forkJoinPool" ref="forkJoinPool" />
    </bean>
</beans>

<!-- Or setting the default autowire mode -->
<beans default-autowire="byName" ...>
    ...
</beans>

可能会进行两项额外的更改,以实现构造函数注入和构造注入。@Autowired

  • 构造函数注入:

    Bean 工厂分配一个用于构造函数注入。可以覆盖默认值 () 以应用候选解析器,该解析器查找用于注入的符合条件的豆类。AutowireCandidateResolverContextAnnotationAutowireCandidateResolverDefaultListableBeanFactory#setAutowireCandidateResolverBeanDefinition

  • @Autowired注射:

    Bean 后处理器直接设置 Bean 值,而不解析内部 Bean。可以覆盖此后处理器,也可以创建单独的 Bean 后处理器来处理托管原型 Bean 的自定义注释(例如,)。AutowiredAnnotationBeanPostProcessorBeanDefinition@AutowiredManagedPrototype