这是在WAS8.0上运行的JSF 2应用程序。这是一个页面的“支持”bean的代码。
@Named("mySessionBean")
@SessionScoped
@Stateful
@LocalBean
@StatefulTimeout(unit = TimeUnit.MINUTES, value = 10)
public class MySessionBean implements Serializable {
@PostConstruct
public void init()
{
System.out.println("start MySessionBean: " + this.hashCode());
}
@PreDestroy
public void cleanup()
{
System.out.println("destroy MySessionBean: " + this.hashCode());
}
....
}
web.xml中设置的会话超时值小于bean的超时时间。当我运行应用程序时,我看到来自@PostConstruct的打印输出,但从未看到来自@PreDestroy的打印输出。我尝试了以下两种方案:1。logout - invalidateSession; 2.只需等到会话到期。
我不是应用程序的设计者。设计者坚持将所有支持bean作为有状态会话bean。我认为更主流的方法就是让它们成为CDI bean。但无论如何,当我确实将注释更改为CDI时,我也开始从@PreDestroy获取打印输出
@Named("mySessionBean")
@SessionScoped
public class MySessionBean implements Serializable {
.....
我的问题是,在第一种情况下我没有得到@PreDestroy方法调用的原因是什么?如果我看不到@PreDestroy被调用,是否还有其他方法可以跟踪“支持”bean的生命周期(在这种情况下,是一个有状态会话bean)。谢谢!
答案 0 :(得分:4)
查看Java EE 6 Tutorial的这一部分,它显示了有状态会话bean的生命周期。仅当从客户端代码中显式删除bean,调用使用 @Remove 注释的方法时,才会调用 PreDestroy 方法。
答案 1 :(得分:2)
另一个答案链接到Java EE 6 Tutorial,甚至没有提到超时的存在。我也不认为该教程明确指出@PreDestroy方法仅在显式的@Remove方法调用之后调用。没有说明因果关系,它只是描述了两个事件,不一定是直接相关的:
在生命周期结束时,客户端调用注释@Remove的方法,EJB容器调用带注释的方法@PreDestroy(如果有的话)。
WebSphere docs提到以下内容:
调用remove方法时,会为有状态会话bean调用PreDestroy生命周期拦截器回调。还要记住,如果有状态会话bean在处于被动状态时超时,或者在bean上的方法调用期间发生意外异常并且bean被丢弃,则不会调用PreDestroy生命周期拦截器回调。
现在,它确实明确表示调用PreDestroy回调 调用remove方法时。同时,它明确指出如果SFSB在被动状态下超时,则不会调用PreDestroy回调。那么这意味着当SFSB不处于被动状态时它们会被调用吗?
现在让我们来看看JBoss 首先,如果removeTimeout<有关删除超时,则不会调用关于PreDestroy的JBoss issue {{3}}。空闲超时。似乎在5.2中已经修复 其次,我刚刚在JBoss 4.3上进行了测试,当一个钝化的SFSB在超时被删除时会发生什么 - 它会在销毁之前被激活。
所以似乎对超时和删除bean的行为方式存在多种解释。