2

JBoss6.1.Final、JSF 2.0(Mojarra)、Weld CDI、MyFaces CODI 1.0.5(ビューアクセススコープ用)を使用しています

ベストプラクティスを再考する実世界のJavaEEパターンのゲートウェイパターンのようなものを使用しています(残念ながら私はそれを持っていないので、ここで何かを台無しにした可能性があります)。基本的に、このアプリケーションを使用すると、ユーザーは「編集モード」に入り、永続コンテキストが拡張された@ViewAccessScopedバッキングBeanに保持されているユーザーのリスト(作成、編集、削除)を編集してから、フラッシュする「保存」コマンドリンクをクリックできます。データベースへのすべての変更。最初はViewExpiredExceptionsに問題がありました(ブラウザがセッションタイムアウト期間を過ぎてアイドル状態になり、その後さらにリクエストが実行された場合)が、セッションを維持するサーブレットにgetリクエストを作成するためにjQueryを追加しました(10と呼ばれます)。セッションタイムアウトの数秒前)。これは機能しているようですが、別の問題が発生しました。バッキングBeanもSFSBであり、アイドル時間の後、

13:06:22,063重大[javax.enterprise.resource.webcontainer.jsf.context] javax.el.E​​LException:/index.xhtml @ 27,81 render = "#{!conversationBean.editMode}":javax.ejb.NoSuchEJBException :ステートフルBeanが見つかりませんでした:43h1h2f-9c7qkb-h34t0f34-1-h34teo9p-de

SFSBの削除を防ぐ方法、または少なくともそれをより適切に処理する方法についてのアイデアはありますか?

これが私のバッキングBeanです:

package com.ray.named;

import java.io.Serializable;
import java.util.List;

import javax.annotation.PostConstruct;
import javax.ejb.EJBTransactionRolledbackException;
import javax.ejb.Stateful;
import javax.ejb.TransactionAttribute;
import javax.inject.Named;
import javax.persistence.EntityManager;
import javax.persistence.PersistenceContext;
import javax.persistence.PersistenceContextType;

import org.apache.myfaces.extensions.cdi.core.api.scope.conversation.ViewAccessScoped;

import com.ray.model.Person;

@Named
@Stateful
@ViewAccessScoped
@TransactionAttribute(javax.ejb.TransactionAttributeType.NEVER)
public class ConversationBean implements Serializable {
  private static final long serialVersionUID = 1L;
  //properties
  private List<Person> people;
  private String name;
  private Boolean editMode;

  @PersistenceContext(type=PersistenceContextType.EXTENDED)
  private EntityManager em;

  @PostConstruct
  public void init() {
    people = em.createNamedQuery("Person.findAll", Person.class).getResultList();
    setEditMode(false);
  }

  //event listeners
  public void beginEdits() {
    setEditMode(true);
  }

  public void addPerson() {
    Person p = new Person(name);
    em.persist(p);
    people.add(p);
    name = null;
  }

  public void removePerson(Person p) {
    people.remove(people.indexOf(p));
    em.remove(p);
  }

  //this method flushes the persistence context to the database
  @TransactionAttribute(javax.ejb.TransactionAttributeType.REQUIRES_NEW)
  public void saveEdits() {
    setEditMode(false);
  }

  //getters/setters
  public List<Person> getPeople() {
    return people;
  }

  public String getName() {
    return name;
  }

  public void setName(String name) {
    this.name = name;
  }

  public Boolean getEditMode() {
    return editMode;
  }

  public void setEditMode(Boolean editMode) {
    this.editMode = editMode;
  }
}

PersonエンティティBeanは次のとおりです。

package com.ray.model;

import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import javax.persistence.NamedQueries;
import javax.persistence.NamedQuery;
import javax.persistence.Version;

@Entity
@NamedQueries({
  @NamedQuery(name="Person.findAll",
              query="SELECT p FROM Person p")
})
public class Person {
  @Id @GeneratedValue(strategy=GenerationType.IDENTITY) 
  private Integer id;
  private String name;
  @Version
  private int version;

  public Person() { }

  public Person(String name) {
    setName(name);
  }

  public boolean equals(Object o) {
    if (!(o instanceof Person)) {
      return false;
    }
    return id == ((Person)o).id;
  }

  //getters/setters
  public String getName() {
    return name;
  }
  public void setName(String name) {
    this.name = name;
  }

  public Integer getId() {
    return id;
  }

  public int getVersion() {
    return version;
  }

  public void setVersion(int version) {
    this.version = version;
  }
}

ビューは次のとおりです。

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml"
      xmlns:h="http://java.sun.com/jsf/html"
      xmlns:f="http://java.sun.com/jsf/core"
      xmlns:ui="http://java.sun.com/jsf/facelets">
<h:head>
  <script src="http://code.jquery.com/jquery-latest.min.js"></script>
  <script>
  $(document).ready(function() {
    setInterval(function() {
      $.get("#{request.contextPath}/poll");
    }, #{(session.maxInactiveInterval - 10) * 1000});
  });
  </script>
  <title>Conversation Test</title>
</h:head>
<h:body>
  <h:form>
    <h:commandLink value="Begin Edits" rendered="#{!conversationBean.editMode}">
      <f:ajax render="@form" listener="#{conversationBean.beginEdits}"/>
    </h:commandLink>
    <h:commandLink value="Save" rendered="#{conversationBean.editMode}">
      <f:ajax render="@form" listener="#{conversationBean.saveEdits}"/>
    </h:commandLink>
    <h:dataTable id="peopleTable" value="#{conversationBean.people}" var="person">
      <h:column>
        <f:facet name="header">Name</f:facet>
        <h:panelGroup>
          <h:inputText value="#{person.name}" disabled="#{!conversationBean.editMode}">
            <f:ajax/>
          </h:inputText>
          <h:commandLink value="X" disabled="#{!conversationBean.editMode}">
            <f:ajax render="@form" listener="#{conversationBean.removePerson(person)}"/>
          </h:commandLink>
        </h:panelGroup>
      </h:column>
    </h:dataTable>
    <h:panelGrid columns="2">
      <h:outputLabel for="name">Name:</h:outputLabel>
      <h:inputText id="name" value="#{conversationBean.name}" disabled="#{!conversationBean.editMode}"/>
    </h:panelGrid>
    <h:commandButton value="Add" disabled="#{!conversationBean.editMode}">
      <f:ajax execute="@form" render="@form" listener="#{conversationBean.addPerson}"/>
    </h:commandButton>
  </h:form>
</h:body>
</html>

セッションを存続させるために使用されるサーブレットは次のとおりです(セッションが期限切れになる10秒前にjQuery ajax getリクエストによって呼び出されます)。

package com.ray.web;

import java.io.IOException;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

public class PollServlet extends HttpServlet {
  private static final long serialVersionUID = 1L;

  public void init() throws ServletException {
  }

  public String getServletInfo() {
    return null;
  }

  public void doGet(HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException {
    request.getSession(); //Keep session alive
  }

  public void doPost(HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException {
  }

  public void destroy() {
  }
}
4

2 に答える 2

1

SFSBの削除を防ぐ方法、または少なくともそれをより適切に処理する方法についてのアイデアはありますか?

さらに調査するには、パッシベーション用のEJBライフサイクルフックを調べて、そこにデバッグ出力を追加することをお勧めします。

それが問題の原因である場合は、パッシベーションを構成/非アクティブ化できますが、スケーラビリティが問題になる可能性があります。

正直なところ、このシナリオは私には非常に珍しいようです。一般に、リクエスト/会話/セッションはデフォルトの境界で多かれ少なかれ機能することを期待します-これを回避するコードを書いていることに気付いた場合、RESTful /ステートレスアプローチの方が良いかもしれません...?

可能な場合は、さらに詳しい情報で質問を更新してください。

于 2012-06-07T06:53:56.443 に答える
0

あなたはすでに問題を解決していると思います。それ以外の場合は、このJBoss wikiページが役立ちます(将来の読者にも役立ちます...)。

https://community.jboss.org/wiki/Ejb3DisableSfsbPassivation

乾杯、ルイージ

于 2012-06-28T15:58:42.493 に答える