web-dev-qa-db-ja.com

Beanが@ViewScopedであるにもかかわらず、@ PostConstructコールバックが毎回起動するのはなぜですか? JSF

ページのデータテーブルを使用し、バインディング属性を使用してバッキングBeanにバインドしています。これは私のコードです:-

<?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:p="http://primefaces.prime.com.tr/ui">
    <h:head>
        <title>Facelet Title</title>
    </h:head>
    <h:body>
            <h:form prependId="false">

                <h:dataTable var="item" value="#{testBean.stringCollection}" binding="#{testBean.dataTable}">
                    <h:column>
                        <h:outputText value="#{item}"/>
                    </h:column>
                    <h:column>
                        <h:commandButton value="Click" actionListener="#{testBean.action}"/>
                    </h:column>
                </h:dataTable>

            </h:form>

    </h:body>
</html>

これは私の豆です:-

package managedBeans;

import Java.io.Serializable;
import Java.util.ArrayList;
import Java.util.List;
import javax.annotation.PostConstruct;
import javax.faces.bean.ManagedBean;
import javax.faces.bean.ViewScoped;
import javax.faces.component.html.HtmlDataTable;

@ManagedBean(name="testBean")
@ViewScoped
public class testBean implements Serializable {

    private List<String> stringCollection;

    public List<String> getStringCollection() {
        return stringCollection;
    }

    public void setStringCollection(List<String> stringCollection) {
        this.stringCollection = stringCollection;
    }

    private HtmlDataTable dataTable;

    public HtmlDataTable getDataTable() {
        return dataTable;
    }

    public void setDataTable(HtmlDataTable dataTable) {
        this.dataTable = dataTable;
    }

    @PostConstruct
    public void init(){
        System.out.println("Post Construct fired!!");
        stringCollection = new ArrayList<String>();
        stringCollection.add("a");
        stringCollection.add("b");
        stringCollection.add("c");

    }

    public void action(){
        System.out.println("Clicked!!");

    }
}

@PostConstructがボタンをクリックするたびに毎回起動する理由を教えてください。私のBeanが@ViewScopedであるため、同じページにいる限り、1回だけ起動する必要があります。さらに、バインディング属性を削除すると、すべてが正常に機能し、@ PostConstructコールバックが1回だけ発生します。では、なぜ毎回バインディング属性を使用するのですか?バインディング属性が必要で、Webサービスからのデータのフェッチなどの初期化タスクを1回だけ実行したい。私は何をすべきか?初期化タスクはどこに書くべきですか?

32
TCM

興味深いことに、ビュースコープBeanでコンポーネントバインディングを使用すると、ビュースコープが壊れます。

それがJSF2のバグかどうかはわかりませんが、最初にJSF2仕様全体を読む必要があります。これまでのところ、最善の策は、今のところコンポーネントバインディングを削除し、選択した項目を新しいEL 2.2メソッド引数構文を介して渡すことです。

<h:dataTable var="item" value="#{testBean.stringCollection}">
    <h:column>
        <h:outputText value="#{item}"/>
    </h:column>
    <h:column>
        <h:commandButton value="Click" action="#{testBean.action(item)}"/>
    </h:column>
</h:dataTable>

以下も参照してください。


更新(2012年12月):これは確かにJSF2のバグです。鶏卵問題です。ビュースコープのBeanはJSFビューステートに格納されます。したがって、ビュースコープのBeanは、ビューの復元フェーズの後でのみ使用できます。ただし、binding属性はビューの復元フェーズ中に実行されますが、ビュースコープのBeanはまだ使用できません。これにより、新しいビュースコープBeanインスタンスが作成され、後で、復元されたJSFビューステートに格納された実際のビュースコープBeanに置き換えられます。

これは JSF issue 1492 および JSF spec isssue 787 として報告され、JSF 2.2で修正されます。それまでは、リクエストスコープBeanでbindingを排他的に使用するか、特定の機能要件の代替方法を探すことが最善の策です。


更新(2015年3月):JSF 2.2修正がMojarra 2.1.18にバックポートされました。したがって、JSF 2.0/2.1をまだ使用している場合は、少なくともそのバージョンにアップグレードすることをお勧めします。 a.oも参照してください。 JSFのコンポーネントバインディングとは何ですか?いつ使用するのが望ましいですか? および JSF2 FaceletsのJSTLは理にかなっていますか?

33
BalusC

他の人が言ったように、私はコンポーネントのバインディングを削除することをお勧めします(ここでは必要ありません)。

しかし、次のように、アクションパラメータを使用して、よりオブジェクト指向の方法で実行しようとしているのと同じことを実現できることを付け加えておきます。

<h:commandButton value="Click" action="#{testBean.action(item)}"/>

...そしてJavaコード:

  public void action(Item item){
    System.out.println("Clicked!!" + item);
}
4
ymajoros

Viewscoped Beanがあり、フォームに入力された値を保持したい場合、またはpostconstructを起動したくない場合は、アクションメソッドからnullを返す必要があります。

何らかの結果(無効など)を返し、faces-config.xmlを使用して無効な結果を同じページにポイントすると、viewscoped Beanが再作成されるため、postconstructが再び起動します。

0
Pramod Kankure

Baluscの答えは私を大いに助けました、私はmojarraバージョン2.1.7でそのバグがあったと言いたいです、私は現在、January-2015でリリースされた2.1.29-01を使用しています、そしてこのバグは修正されました、私の問題はviewscoped Beanへのtabview。このバージョンでは、そのバグはなく、バインディングとpostconstructは正常に動作しています。私はJboss 5.2を使用していて、mojarra 2.1.xを使用する必要があるので、この答えが同じ状況の他の人々を助けることを願っています。

http://mvnrepository.com/artifact/com.Sun.faces/jsf-api/2.1.29-01http://mvnrepository.com/artifact/com.Sun。 faces/jsf-impl/2.1.29-01

0
Luis Vidal

その他の解決策:

  • HtmlDataTableをリクエストスコープBeanにバインドする。
  • このリクエストスコープBeanをビュースコープBeanに挿入します。

JBoss Seamは、JSFコンポーネントを対話スコープコンポーネントにバインドするためにこのソリューションを使用します。