4

Weblogic 12c (具体的には 12.1.3) を使用して (EJB 3.1) アプリケーションをデプロイしています。

私のアプリケーションには、およびアノテーションをそれぞれ使用して注入されたおよび他のBean@Statefulへの参照を保持する Bean があります。EntityManager@Stateless@PersistenceContext@EJB

私の問題は、ステートフル Bean がパッシベーションされてディスクにシリアル化されているときに、Weblogic がステートレス Bean への参照もシリアル化しようとし、WeblogicNotSerializableExceptionによって注入された 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)
4

1 に答える 1

1

現時点では、これは Weblogic 12.1.3 の実際のバグ/制限のように思われるため、可能な解決策として回避策を投稿しています。

StatefulBean のパッシベーションを成功させるには、 および のアノテーションが付けられたメソッドを実装する必要がありjavax.ejb.PrePassivateますjavax.ejb.PostActivate。この@PrePassivateメソッドは、ステートレス Bean の参照ポイントをに設定し、Bean が再びアクティブ化されるときに、メソッドはその Bean のルックアップを実行しますnull@PostActivate

@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");
}

今後数週間ほどでコメントやその他の回答がない場合は、この回答を解決策としてマークします.

于 2015-08-03T11:51:18.563 に答える