【问题标题】:Stateful bean passivation fails because of Stateless beans references由于无状态 bean 引用,有状态 bean 钝化失败
【发布时间】:2015-10-03 00:42:02
【问题描述】:

我正在使用 Weblogic 12c(具体来说是 12.1.3)来部署我的 (EJB 3.1) 应用程序。

在我的应用程序中,我有一个 @Stateful bean,它包含对 EntityManager 和其他 @Stateless bean 的引用,分别使用 @PersistenceContext@EJB 注释注入。

我的问题是,当我的有状态 bean 被钝化并序列化到磁盘时,Weblogic 也会尝试序列化对无状态 bean 的引用,并抛出一个 NotSerializableException 引用由 Weblogic 注入的 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)

【问题讨论】:

  • EJB 规范说您应该能够保留对本地和远程 EJB 的引用,即使它们不可序列化:what-when-how.com/enterprise-javabeans-3-1/…(参见“钝化状态”)...至少这是理论,但我不知道你为什么会收到这个错误,所以显然还有更多。
  • 有状态 bean 的所有字段都必须实现 Serializable。将它们定义为瞬态并使用Instance 注入或直接查找在预激活时恢复
  • @fantarama - 正如@hugh 指出的那样,根据 ejb 3.1(可能在此之前,不确定) - The types in this list (and their subtypes) are handled specially by the passivation mechanism. They do not need to be serializable; they will be maintained through passivation and restored automatically when the bean instance is activated. 所以唯一的问题是我的无状态 bean 是否应该实现一些东西来制作容器处理方式不同?
  • 但是PassivateBean 实现了Serializable??
  • 没有要求Stateful bean 实现Serializable。如果没有对Stateless bean 的引用,上述代码可以完美地使用钝化/重新激活。

标签: java ejb weblogic weblogic12c ejb-passivation


【解决方案1】:

到目前为止,这似乎是 Weblogic 12.1.3 中的一个实际错误/限制,因此我将我的解决方法发布为可能的解决方案。

要使Stateful bean 成功通过钝化,需要实现带有javax.ejb.PrePassivatejavax.ejb.PostActivate 注释的方法。 @PrePassivate 方法将使无状态 bean 引用指向 null,当 bean 再次被激活时,@PostActivate 方法将对该 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");
}

如果在接下来的几周左右没有 cmets 或其他答案,我会将这个答案标记为解决方案。

【讨论】:

    猜你喜欢
    • 2015-05-15
    • 2015-01-04
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2014-10-01
    • 1970-01-01
    • 2011-05-29
    相关资源
    最近更新 更多