Java 由于无状态bean引用,有状态bean钝化失败

Java 由于无状态bean引用,有状态bean钝化失败,java,ejb,weblogic,weblogic12c,ejb-passivation,Java,Ejb,Weblogic,Weblogic12c,Ejb Passivation,我正在使用WebLogic12C(具体来说是12.1.3)部署我的(EJB3.1)应用程序 在我的应用程序中,我有一个@Statefulbean,它分别使用@PersistenceContext和@EJB注释来注入对EntityManager和其他@无状态bean的引用 我的问题是,当我的有状态bean被钝化并序列化到磁盘时,Weblogic也尝试序列化对无状态bean的引用,并抛出一个引用Weblogic注入的bean代理的NotSerializableException。作为比较-Entit

我正在使用WebLogic12C(具体来说是12.1.3)部署我的(EJB3.1)应用程序

在我的应用程序中,我有一个
@Stateful
bean,它分别使用
@PersistenceContext
@EJB
注释来注入对
EntityManager
和其他
@无状态
bean的引用

我的问题是,当我的有状态bean被钝化并序列化到磁盘时,Weblogic也尝试序列化对无状态bean的引用,并抛出一个引用Weblogic注入的bean代理的
NotSerializableException
。作为比较-EntityManager引用被钝化和重新激活,没有任何问题。只有无状态bean才会引起问题

我知道我可以定义
@PrePassivate
@PostActivate
方法来让我的代码工作,但是有没有办法让容器自己处理无状态bean引用

附上示例代码,为我重现问题

远程bean接口:

import javax.ejb.Remote;

@Remote
public interface Passivate {

    public void doSomething();

}
import javax.ejb.Local;

@Local
public interface NoState {

    public void foo();

}
有状态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
    }
}
import javax.ejb.Stateless;

@Stateless(mappedName = "NoStateBean")
public class NoStateBean implements NoState {

    @Override
    public void foo() {
        System.out.println("foo");
    }

}
NoState本地接口:

import javax.ejb.Remote;

@Remote
public interface Passivate {

    public void doSomething();

}
import javax.ejb.Local;

@Local
public interface NoState {

    public void foo();

}
nostatebean实现:

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
    }
}
import javax.ejb.Stateless;

@Stateless(mappedName = "NoStateBean")
public class NoStateBean implements NoState {

    @Override
    public void foo() {
        System.out.println("foo");
    }

}
最后,当钝化Bean被钝化到磁盘时,我会遇到一个异常:

<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中的一个实际错误/限制,因此我将发布我的解决方案作为可能的解决方案

为了使
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");
}

如果在未来几周左右没有任何评论或其他答案,我将把这个答案标记为解决方案。

EJB规范说,即使本地和远程EJB不可序列化,也应该能够保留对它们的引用:(参见“被动状态”)……至少这是理论,但我不知道为什么会出现这种错误,因此,显然还有更多的事情要做。一个状态完整bean的所有字段都必须实现Serializable。使用
实例
注入或直接将它们定义为瞬态,并在预激活时恢复lookup@fantarama-正如@hugh所指出的,根据EJB3.1(可能在此之前,不确定)
这个列表中的类型(及其子类型)是由钝化机制专门处理的。它们不需要序列化;它们将通过钝化来维护,并在bean实例被激活时自动恢复。
所以唯一的问题是我的无状态bean是否应该实现一些东西,以使容器以不同的方式处理它?但是
钝化bean
实现
可序列化
?对
Stateful
bean来实现
Serializable
。如果没有对
无状态
bean的引用,那么上面的代码在钝化/重新激活方面工作得完美无缺。