由于无状态 bean 引用,有状态 bean 钝化失败
Stateful bean passivation fails because of Stateless beans references
我正在使用 Weblogic 12c(具体来说是 12.1.3)来部署我的 (EJB 3.1) 应用程序。
在我的应用程序中,我有一个 @Stateful
bean,它包含对 EntityManager
和其他 @Stateless
bean 的引用,使用 @PersistenceContext
和 [=19= 注入]分别注解。
我的问题是,当我的有状态 bean 被钝化并序列化到磁盘时,Weblogic 也尝试序列化对无状态 bean 的引用,并抛出一个 NotSerializableException
引用被注入的 bean 代理通过网络逻辑。作为比较 - EntityManager 引用被钝化和重新激活,完全没有问题。只有无状态 Bean 会导致问题。
我知道我可以定义 @PrePassivate
和 @PostActivate
方法来让我的代码工作,但是有什么方法可以让容器自己处理无状态 bean 引用吗?
附上为我重现问题的示例代码。
远程 bean 接口:
import javax.ejb.Remote;
@Remote
public interface Passivate {
public void doSomething();
}
有状态 Bean 实现:
import javax.ejb.EJB;
import javax.ejb.PostActivate;
import javax.ejb.PrePassivate;
import javax.ejb.Stateful;
@Stateful(mappedName = "PassivateBean")
public class PassivateBean implements Passivate {
@EJB(mappedName = "NoStateBean")
private NoState noStateBean;
@Override
public void doSomething() {
System.out.println("Hello world");
}
@PrePassivate
public void prePassivate() {
// as a workaround - can set noStateBean to null here
}
@PostActivate
public void postActivate() {
// as a workaround - can lookup and set noStateBean here manually
}
}
NoState 本地接口:
import javax.ejb.Local;
@Local
public interface NoState {
public void foo();
}
NoState bean 实现:
import javax.ejb.Stateless;
@Stateless(mappedName = "NoStateBean")
public class NoStateBean implements NoState {
@Override
public void foo() {
System.out.println("foo");
}
}
最后,当 PassivateBean 被钝化到磁盘时我得到的异常:
<Jul 14, 2015 2:36:20 PM IDT> <Error> <EJB> <BEA-010024> <Error occurred during passivation: java.io.NotSerializableException: passivateTest.NoStateBean_i02rk_Impl
at java.io.ObjectOutputStream.writeObject0(ObjectOutputStream.java:1184)
at java.io.ObjectOutputStream.writeObject(ObjectOutputStream.java:348)
at weblogic.ejb.container.utils.Serializer.writeObject(Serializer.java:52)
at passivateTest.PassivateBean_dmn8u8_Impl.writeObject(Unknown Source)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:483)
at java.io.ObjectStreamClass.invokeWriteObject(ObjectStreamClass.java:988)
at java.io.ObjectOutputStream.writeSerialData(ObjectOutputStream.java:1496)
at java.io.ObjectOutputStream.writeOrdinaryObject(ObjectOutputStream.java:1432)
at java.io.ObjectOutputStream.writeObject0(ObjectOutputStream.java:1178)
at java.io.ObjectOutputStream.writeObject(ObjectOutputStream.java:348)
at weblogic.ejb.container.swap.PassivationUtils.write(PassivationUtils.java:84)
at weblogic.ejb.container.swap.DiskSwap.write(DiskSwap.java:173)
at weblogic.ejb.container.manager.StatefulSessionManager.swapOut(StatefulSessionManager.java:1246)
at weblogic.ejb.container.cache.LRUCache.passivateNode(LRUCache.java:199)
at weblogic.ejb.container.cache.LRUCache.timerExpired(LRUCache.java:187)
at weblogic.timers.internal.TimerImpl.run(TimerImpl.java:304)
at weblogic.work.SelfTuningWorkManagerImpl$WorkAdapterImpl.run(SelfTuningWorkManagerImpl.java:548)
at weblogic.work.ExecuteThread.execute(ExecuteThread.java:311)
at weblogic.work.ExecuteThread.run(ExecuteThread.java:263)
截至目前,这似乎是 Weblogic 12.1.3 中的实际 bug/limitation,因此我将我的解决方法作为可能的解决方案发布。
要使Stateful
bean 成功通过钝化,需要实现用javax.ejb.PrePassivate
和javax.ejb.PostActivate
注释的方法。 @PrePassivate
方法将使无状态 bean 引用指向 null
,而 @PostActivate
方法将在再次激活该 bean 时对该 bean 执行查找。
@PrePassivate
public void prePassivate() {
noStateBean = null;
}
@PostActivate
public void postActivate() {
// The lookup string is correct assuming the ejb module is deployed within an ear. If your setup is different the JNDI lookup name may be slightly different.
noStateBean = InitialContext.doLookup("java:module/NoStateBean!NoState");
}
如果在接下来的几周内没有评论或其他答案,我会将此答案标记为解决方案。
我正在使用 Weblogic 12c(具体来说是 12.1.3)来部署我的 (EJB 3.1) 应用程序。
在我的应用程序中,我有一个 @Stateful
bean,它包含对 EntityManager
和其他 @Stateless
bean 的引用,使用 @PersistenceContext
和 [=19= 注入]分别注解。
我的问题是,当我的有状态 bean 被钝化并序列化到磁盘时,Weblogic 也尝试序列化对无状态 bean 的引用,并抛出一个 NotSerializableException
引用被注入的 bean 代理通过网络逻辑。作为比较 - EntityManager 引用被钝化和重新激活,完全没有问题。只有无状态 Bean 会导致问题。
我知道我可以定义 @PrePassivate
和 @PostActivate
方法来让我的代码工作,但是有什么方法可以让容器自己处理无状态 bean 引用吗?
附上为我重现问题的示例代码。
远程 bean 接口:
import javax.ejb.Remote;
@Remote
public interface Passivate {
public void doSomething();
}
有状态 Bean 实现:
import javax.ejb.EJB;
import javax.ejb.PostActivate;
import javax.ejb.PrePassivate;
import javax.ejb.Stateful;
@Stateful(mappedName = "PassivateBean")
public class PassivateBean implements Passivate {
@EJB(mappedName = "NoStateBean")
private NoState noStateBean;
@Override
public void doSomething() {
System.out.println("Hello world");
}
@PrePassivate
public void prePassivate() {
// as a workaround - can set noStateBean to null here
}
@PostActivate
public void postActivate() {
// as a workaround - can lookup and set noStateBean here manually
}
}
NoState 本地接口:
import javax.ejb.Local;
@Local
public interface NoState {
public void foo();
}
NoState bean 实现:
import javax.ejb.Stateless;
@Stateless(mappedName = "NoStateBean")
public class NoStateBean implements NoState {
@Override
public void foo() {
System.out.println("foo");
}
}
最后,当 PassivateBean 被钝化到磁盘时我得到的异常:
<Jul 14, 2015 2:36:20 PM IDT> <Error> <EJB> <BEA-010024> <Error occurred during passivation: java.io.NotSerializableException: passivateTest.NoStateBean_i02rk_Impl
at java.io.ObjectOutputStream.writeObject0(ObjectOutputStream.java:1184)
at java.io.ObjectOutputStream.writeObject(ObjectOutputStream.java:348)
at weblogic.ejb.container.utils.Serializer.writeObject(Serializer.java:52)
at passivateTest.PassivateBean_dmn8u8_Impl.writeObject(Unknown Source)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:483)
at java.io.ObjectStreamClass.invokeWriteObject(ObjectStreamClass.java:988)
at java.io.ObjectOutputStream.writeSerialData(ObjectOutputStream.java:1496)
at java.io.ObjectOutputStream.writeOrdinaryObject(ObjectOutputStream.java:1432)
at java.io.ObjectOutputStream.writeObject0(ObjectOutputStream.java:1178)
at java.io.ObjectOutputStream.writeObject(ObjectOutputStream.java:348)
at weblogic.ejb.container.swap.PassivationUtils.write(PassivationUtils.java:84)
at weblogic.ejb.container.swap.DiskSwap.write(DiskSwap.java:173)
at weblogic.ejb.container.manager.StatefulSessionManager.swapOut(StatefulSessionManager.java:1246)
at weblogic.ejb.container.cache.LRUCache.passivateNode(LRUCache.java:199)
at weblogic.ejb.container.cache.LRUCache.timerExpired(LRUCache.java:187)
at weblogic.timers.internal.TimerImpl.run(TimerImpl.java:304)
at weblogic.work.SelfTuningWorkManagerImpl$WorkAdapterImpl.run(SelfTuningWorkManagerImpl.java:548)
at weblogic.work.ExecuteThread.execute(ExecuteThread.java:311)
at weblogic.work.ExecuteThread.run(ExecuteThread.java:263)
截至目前,这似乎是 Weblogic 12.1.3 中的实际 bug/limitation,因此我将我的解决方法作为可能的解决方案发布。
要使Stateful
bean 成功通过钝化,需要实现用javax.ejb.PrePassivate
和javax.ejb.PostActivate
注释的方法。 @PrePassivate
方法将使无状态 bean 引用指向 null
,而 @PostActivate
方法将在再次激活该 bean 时对该 bean 执行查找。
@PrePassivate
public void prePassivate() {
noStateBean = null;
}
@PostActivate
public void postActivate() {
// The lookup string is correct assuming the ejb module is deployed within an ear. If your setup is different the JNDI lookup name may be slightly different.
noStateBean = InitialContext.doLookup("java:module/NoStateBean!NoState");
}
如果在接下来的几周内没有评论或其他答案,我会将此答案标记为解决方案。