| Oracle® Fusion Middleware Oracle Application Development Framework Fusion開発者ガイド 11g リリース2(11.1.2.3.0) B69399-02 |
|
![]() 前 |
![]() 次 |
この章では、ADFページ・ライフサイクルとそのフェーズ、Fusion Webアプリケーション内でのライフサイクルの最適な使用方法について説明します。
この章の内容は次のとおりです。
ページが送信され、新しいページがリクエストされると、標準のJSFリクエスト・ライフサイクルを拡張するADF Facesページ・ライフサイクル、およびADFページ・ライフサイクルの両方がアプリケーションによって起動されます。ADF Facesライフサイクルでは、ページ上の値の送信、コンポーネント値の検証、ページ間の移動、結果ページへのコンポーネントの表示、および状態の保存とリストアが処理されます。JSFライフサイクル・フェーズでは、UIコンポーネント・ツリーを使用してfacesコンポーネントの表示を管理します。このツリーはJSFページのランタイム表現で、ページ内の各UIコンポーネント・タグがツリーのUIコンポーネント・インスタンスに対応します。JSFアプリケーションのリクエスト処理ライフサイクルは、FacesServletサーブレットによって管理されます。FacesServletは、リクエスト処理に必要な情報を含むFacesContextと呼ばれるオブジェクトを作成し、ライフサイクルを実行するオブジェクトを起動します。
拡張JSFライフサイクルの詳細は、『Oracle Fusion Middleware Oracle Application Development Framework Webユーザー・インタフェース開発者ガイド』の「JSFおよびADF Facesのライフサイクルの理解」を参照してください。
ADFページ・ライフサイクルは、データ・モデルの準備と更新、モデル・レイヤーでのデータの検証、ビジネス・レイヤーでのメソッドの実行を処理します。ADFページ・ライフサイクルでは、バインディング・コンテナを使用して、現在のページ・リクエスト時にページによる簡単なデータ参照を可能にします。
組み合せたJSFページ・ライフサイクルとADFページ・ライフサイクルは、アプリケーション・サーバーにHTTPリクエストが届いたときに始まってページがクライアントに戻されるまで続く、より大規模なイベント・シーケンスに含まれる1つのシーケンスにすぎません。このイベント・シーケンス全体をWebページ・ライフサイクルと呼びます。このライフサイクルは、MVCアーキテクチャで定義されているようにモデル、ビュー、コントローラのレイヤーを通して処理をたどっていきます。ページ・ライフサイクルは厳格に定義されたイベント・セットではなく、一般的なユースケースで使用されるイベント・セットです。図25-1に、JSFとOracle ADFを併用した場合のWebページ・リクエストのライフサイクルのシーケンス図を示します。
JSFおよびOracle ADFを使用したWebページ・リクエストの処理の基本的なフローは、次のようになります。
http://yourserver/yourapp/faces/some.jspに対するWebリクエストが、クライアントからアプリケーション・サーバーに到着します。
ADFBindingFilterオブジェクトが、HTTPセッション内でADFバインディング・コンテキストを検索し、それがまだ存在しない場合はその初期化を初めて実行します。ADFBindingFilterの機能には、バインディング・コンテキスト・メタデータ・ファイルの名前の検索、各データ・コントロールの検索と構築などがあります。
ADFBindingFilterオブジェクトが、該当するリクエストに関与する各データ・コントロールに対してbeginRequest()メソッドを起動します。このメソッドにより、データ・コントロールは各リクエストの開始時に通知を受けるため、必要な設定を実行できるようになります。
JSF Lifecycleオブジェクト(各リクエストの標準的な処理フェーズの調整を行う)が、ライフサイクルの各フェーズでADFPhaseListenerクラスに通知し、JSFライフサイクルとADF Modelデータ・バインディング・レイヤーを調整するためのカスタムな処理が実行できるようにします。JSFページ・ライフサイクルとADFページ・ライフサイクルのフェーズの詳細は、25.2項「JSFページ・ライフサイクルとADFページ・ライフサイクルについて」を参照してください。
|
注意:
|
ADFPhaseListenerオブジェクトが、各リクエストを処理するためのADF PageLifecycleオブジェクトを作成し、該当するフェーズ前後のメソッドをADF PageLifecycleクラス内の対応メソッドに委譲します。ページのバインディング・コンテナは、ユーザーのセッション中、以前に使用されていない場合は作成されます。
アプリケーション・モジュール・データ・コントロールが、リクエスト時に初めて参照されると、アプリケーション・モジュール・プールからアプリケーション・モジュールのインスタンスを取得します。
JSF Lifecycleオブジェクトが、レンダリング対象のページにコントロールを移動します。
該当するページ上の一連のUIコンポーネントが、該当ページのバインディング・コンテナ内の値バインディングおよびイテレータ・バインディングにアクセスし、フォーマット設定された出力をレンダリングしてブラウザに表示します。
ADFBindingFilterオブジェクトが、該当するリクエストに関与する各データ・コントロールに対してendRequest()メソッドを起動します。このメソッドにより、データ・コントロールは各リクエストの終了時に通知を受けるため、必要なリソースのクリーンアップを実行できるようになります。
アプリケーション・モジュール・データ・コントロールがendRequestの通知を使用して、アプリケーション・モジュール・プールにアプリケーション・モジュールのインスタンスを解放して戻します。
得られたページが、ユーザーのブラウザに表示されます。
ADFページ・ライフサイクルには、対応するJSFフェーズの実行前後にADFページ・ライフサイクル・リスナーに通知するためのみに定義されたフェーズも含まれています(これらのフェーズは実装されていません)。これらのフェーズにより、カスタム・リスナーを作成してJSFページ・ライフサイクルとADFページ・ライフサイクルの両方のフェーズに登録できるため、必要に応じてグローバルに、またはページ・レベルでADFページ・ライフサイクルをカスタマイズできるようになります。
図25-2は、JSFページ・ライフサイクルとADFページ・ライフサイクルを組み合せた概要を示します。
たとえば、テキストが表示された入力テキスト・コンポーネント、およびコマンド・ボタンからなるページがあるとします。このページが初めてレンダリングされるとき、ビューのリストア・フェーズ中にコンポーネント・ツリーが構築され、コンポーネントのレンダリング中、ライフサイクルはレスポンスのレンダリング・フェーズまで進みますユーザーがボタンをクリックすると、完全なライフサイクルが起動されます。リクエスト値の適用フェーズおよび検証処理フェーズ中、コンポーネント・ツリーが再構築され、入力テキスト・コンポーネントではすべての新規値が抽出されます。エラーが発生した場合は(検証によるエラーなど)、ライフサイクルはレスポンスのレンダリング・フェーズにまで進みます。それ以外の場合は、モデルの更新フェーズ中にモデルが新規値によって更新され、アプリケーションの起動フェーズ中、コマンド・ボタンに関連付けられたすべてのアプリケーション処理(ナビゲーションなど)が実行されます。
図25-3は、JSF、ADF Faces、ADFモデル・フェーズがページ・リクエストのライフサイクルの中で統合されるしくみの詳細を示しています。
ADF Modelレイヤーを使用するJSFアプリケーションでは、ページ・ライフサイクルのフェーズは次のようになります。
ビューのリストア: リクエストされたページのURLがbindingContextオブジェクトに渡され、URLと一致するページ定義が検索されます。リクエストされたページのコンポーネント・ツリーが、新しく作成されるかリストアされます。送信されたページ上のすべてのコンポーネント・タグ、イベント・ハンドラ、コンバータおよびバリデータは、FacesContextインスタンスにアクセスできます。コンポーネント・ツリーが空の場合(つまり、送信されたページからのデータがない)は、ページ・ライフサイクルは直接レスポンスのレンダリング・フェーズに進みます。
リクエスト状態とサーバー側の状態の間で不一致が検出された場合、エラーがスローされ、ページ・ライフサイクルはレスポンスのレンダリング・フェーズに進みます。
JSFのビューのリストア: ビューのリストア・フェーズに対して、フェーズ前後のイベントを提供します。リスナーを作成してこのフェーズの前または後のイベントに登録すると、リスナーをビューのリストア・フェーズに登録したときのように、アプリケーションが動作します。ADFモデル・ページ・ライフサイクルのコンテキストの初期化フェーズでは、after(JSF Restore View)イベントをリスニングして実行します。ADF Controllerは、このフェーズの前後のイベントのリスナーを使用して、サーバー側の状態をリクエストと同期化させます。たとえば、ブラウザの「戻る」ボタンの検出およびブックマーク参照などは、このフェーズで処理されます。前後のリスナーの実行後、ページ・フロー・スコープが使用可能になします。
コンテキストの初期化: リクエストされたページのページ定義ファイルの実行時表現であるbindingContainerオブジェクトが、ページ定義ファイルを使用して作成されます。ADFページ・ライフサイクルの全フェーズで情報を保持するためのLifecycleContextクラスがインスタンス化され、関連付けられているリクエスト、バインディング・コンテナおよびライフサイクルの値で初期化されます。
モデルの準備: ADFページ・ライフサイクルは、BindingContainer.refresh(PREPARE_MODEL)メソッドをコールすることにより、モデルの準備フェーズを開始します。モデルの準備フェーズでは、BindingContainerページ・パラメータが準備された後、評価されます。タスク・フローにパラメータが存在する場合、パラメータはフローに渡されます。
次に、RefreshプロパティがprepareModelに設定された実行可能ファイルが、ページ定義ファイルの<executables>セクション内のエントリの順序と、それらのRefreshConditionプロパティ(存在する場合)の評価に基づいてリフレッシュされます。実行可能ファイルがイテレータ・バインディング・リフレッシュに到達すると、対応するデータ・コントロールが実行され、それによってサービス・オブジェクト内の1つ以上のコレクションが実行されます。イテレータ・バインディングがリフレッシュに失敗すると、JBO例外がスローされ、データを表示できなくなります。詳細は、25.2.2項「Refreshプロパティの正しい使用について」を参照してください。
受信リクエストにPOSTデータまたは問合せパラメータが含まれていない場合、ライフサイクルはレスポンスのレンダリング・フェーズに進みます。
ページがテンプレートを使用して作成されており、そのテンプレートにADF MODELレイヤーを使用したバインディングが含まれている場合、テンプレートのページ定義ファイルを使用してテンプレートのバインディング・コンテナが作成されます。コンテナがバインディング・コンテキストに追加されます。
taskFlow実行可能バインディングが存在する(ページにリージョンが含まれているなど)場合、taskFlowバインディングによりADFコントローラのViewPortContextオブジェクトがタスク・フローに作成された後、フローにあるページのネストされたバインディング・コンテナが実行されます。
リクエスト値の適用: ツリー内の各コンポーネントにより、リクエスト・パラメータから(そのデコード・メソッドを使用して)新しい値が抽出され、ローカルに格納されます。関連付けられているほとんどのイベントは、後の処理のためにキューされます。コンポーネントのimmediate属性がtrueに設定されている場合は、コンポーネントに関連付けられている検証、変換およびイベントがこのフェーズで処理され、ライフサイクルは検証処理フェーズ、モデル値の更新フェーズ、アプリケーションの起動フェーズをスキップします。さらに、関連イテレータが起動されます。ADF Facesの検証と変換の詳細は、『Oracle Fusion Middleware Oracle Application Development Framework Webユーザー・インタフェース開発者ガイド』の「入力の検証および変換」を参照してください。
JSFのリクエスト値の適用: リクエスト値の適用フェーズに対して、フェーズ前後のイベントを提供します。リスナーを作成してこのフェーズの前または後のイベントに登録すると、リスナーをリクエスト値の適用フェーズに登録したときのように、アプリケーションが動作します。
処理の検証: クライアントでコンポーネントのローカルの値が変換され、検証されます。エラーが発生した場合は、ライフサイクルがレスポンスのレンダリング・フェーズに進みます。このフェーズの最後に、コンポーネントの新しい値が設定され、検証または変換のエラー・メッセージおよびイベントがFacesContextでキューされて、すべての値変更イベントが配信されます。例外もバインディング・コンテナにより捕捉されてキャッシュされます。
JSFの検証処理: 検証処理フェーズに対して、フェーズ前後のイベントを提供します。リスナーを作成してこのフェーズの前または後のイベントに登録すると、リスナーを検証処理フェーズに登録したときのように、アプリケーションが動作します。
モデル値の更新: コンポーネントの検証済のローカル値がモデルに移され、ローカル・コピーが破棄されます。更新可能なコンポーネント(inputTextコンポーネントなど)については、リフレッシュ条件がデフォルト(deferred)に設定され、リフレッシュ条件(該当する場合)がtrueに評価されている場合、対応するイテレータがリフレッシュされます。
JSFページのバッキングBeanを使用してUIコンポーネントを管理する場合、バッキングBeanプロパティにバインドされているUI属性もこのフェーズでリフレッシュされます。
JSFのモデル値の更新: モデル値の更新フェーズに対して、フェーズ前後のイベントを提供します。リスナーを作成してこのフェーズの前または後のイベントに登録すると、リスナーをモデル値の更新フェーズに登録したときのように、アプリケーションが動作します。
モデル更新の検証: モデルに設定された検証ルーチンと比較して、更新されたモデルが検証されます。例外がバインディング・コンテナにより捕捉されてキャッシュされます。
アプリケーションの起動: コマンド・コンポーネントまたはイベントのアクション・バインディングが起動されます。
JSFのアプリケーションの起動: アプリケーションの起動フェーズに対して、フェーズ前後のイベントを提供します。リスナーを作成してこのフェーズの前または後のイベントに登録すると、リスナーをアプリケーションの起動フェーズに登録したときのように、アプリケーションが動作します。
メタデータのコミット: 実行時メタデータへの変更がコミットされます。このフェーズでは、メタデータ・サービス(MDS)を使用して、実行時にアプリケーションに対して行われるあらゆる変更が格納されます。MDSを使用した実行時の変更の保持に関する詳細は、第39章「MDSによるアプリケーションのカスタマイズ」を参照してください。
内容の初期化(アプリケーションの起動ライフサイクルでナビゲーションが発生した場合のみ): 実行されるbeforeJSFRenderResponseイベントをリスニングします。次のページのページ定義ファイルが初期化されます。
モデルの準備(アプリケーションの起動ライフサイクルでナビゲーションが発生した場合のみ): 次のページの定義に含まれるすべてのページ・パラメータが設定されます。
レンダリングの準備: バインディング・コンテナがリフレッシュされ、リクエスト値の適用フェーズまたは検証フェーズで発生したすべての変更が反映されます。読取り専用コンポーネント(outputTextコンポーネントなど)に対応するイテレータがリフレッシュされます。必要な場合は、すべての動的リージョンが切り替わります。afterJSFRenderResponseイベントのように、登録されたすべてのリスナーにprepareRenderイベントが送信されます。
|
注意: 選択する有効フェーズとして、JDeveloperでは |
レスポンスのレンダリング: Java EE Webコンテナがページ内のタグを横断するにつれて、ツリー内のコンポーネントがレンダリングされます。後続のリクエストおよびビューのリストア・フェーズ用に、状態の情報が保存されます。
ページと関連データの両方を表示するのに必要な待機時間を短くするため、特定のADF Facesコンポートネント(tableコンポーネントなど)では、最初のリクエストにデータ・ストリーミングを使用します。ページにこれらのコンポーネントが1つ以上含まれる場合、ページは通常のライフサイクルをたどります。しかし、そのリクエスト時にデータをフェッチするかわりに、特別な別のリクエストが実行されます。ページは単にレンダリングされるだけなので、データ・ストリーミングを使用するコンポーネントに対してレスポンスのレンダリング・フェーズのみが実行され、対応するデータがフェッチされ、表示されます。ユーザーのアクション(表でのスクロールなど)によって次のデータのフェッチが発生する場合は、別のリクエストが実行されます。表、ツリー、ツリー表およびデータ視覚化のコンポーネントはすべて、データ・ストリーミングを使用します。
JSFのレスポンスのレンダリング: レスポンスのレンダリング・フェーズに対して、フェーズ前後のイベントを提供します。リスナーを作成してこのフェーズの前または後のイベントに登録すると、リスナーをレスポンスのレンダリング・フェーズに登録したときのように、アプリケーションが動作します。
ADF Facesには、ページ上の特定のコンポーネント(通常は値が変更されたコンポーネント)にのみページ・リクエスト・ライフサイクル(変換と検証を含む)を実行する場合に使用できる、最適化されたライフサイクルが用意されています。
最適化されたライフサイクルを使用する方法の1つは、一方のコンポーネントのイベントが、もう一方のコンポーネント(ターゲット)のトリガーとして機能するように、依存性を手動で設定する方法です。トリガー・コンポーネントでイベントが発生すると、ターゲット・コンポーネントと、トリガーおよびターゲットの両方の子コンポーネントでライフサイクルが実行され、これらのコンポーネントのみがレンダリングされます。これは、部分ページ・レンダリング(PPR)とみなされます。
例25-1は、トリガーの役割を果たすラジオ・ボタンと、ターゲットとなる出力テキストを保持するpanelGroupLayoutコンポーネントを示します(outputTextコンポーネントは常時レンダリングされるわけではないため、panelGroupLayoutコンポーネントをターゲットとする必要があります)。
例25-1 部分ページ・レンダリングの例
<af:form> <af:inputText label="Required Field" required="true"/> <af:selectBooleanRadio id="show" autoSubmit="true" text="Show" value="#{validate.show}"/> <af:selectBooleanRadio id="hide" autoSubmit="true" text="Hide" value="#{validate.hide}"/> <af:panelGroupLayout partialTriggers="show hide" id="panel"> <af:outputText value="You can see me!" rendered="#{validate.show}"/> </af:panelGroupLayout> </af:form>
ラジオ・ボタンのautoSubmit属性がtrueに設定されているため、選択されると、SelectionEventが起動されます。panelGroupLayoutコンポーネントが両方のラジオ・コンポーネントのターゲットとして設定されているため、このイベントが起動されると、selectOneRadio(トリガー)、panelGroupLayoutコンポーネント(トリガーのターゲット)とその子コンポーネント(outputTextコンポーネント)のみ、ライフサイクルの処理が行われます。outputTextコンポーネントは表示ラジオ・ボタンの選択時にのみレンダリングされるよう構成されているため、ユーザーは、ラジオ・ボタンの上の必須入力フィールドにテキストを入力しなくても、ラジオ・ボタンを選択して出力テキストを表示できます。
部分トリガーとターゲットをコード内で手動で設定するかわりに、ADF Facesフレームワークには自動PPRが用意されています。自動PPRは、イベントに対応するイベント・ルート・コンポーネントが存在する場合、またはコンポーネント自体がイベント・ルートである場合に実行されます。イベント・ルート・コンポーネントにより、PPRの境界が決定されます。たとえば、attributeChangeEventはイベントを起動させるinputTextコンポーネントを、そのイベント・ルートであるとみなします。したがって、前述の例では、inputTextコンポーネントの値が変更されると、これがPPRの最初の境界であるため、inputTextコンポーネントとそのすべての子コンポーネント上のみでライフサイクルが実行されます。
イベント・ルートのほか、Fusion Webアプリケーションではデフォルトで、同一のイテレータ・バインディングに関連付けられたすべてのUIコンポーネントに対しても自動PPRが実行されます。関連付けられたすべてのコンポーネントは、そのいずれかで値変更イベントが発生するたびに、すべてリフレッシュされます。この機能は、イテレータのchangeEventPolicy属性をpprに設定することで制御します。デフォルトでは、これはグローバルに設定されています。
たとえば、「データ・コントロール」パネルを使用して多数のinputTextコンポーネントを作成するとします(フォームを作成する場合など)。これらのinputTextコンポーネントのいずれかでattributeChangeEventが発生すると、他のinputTextコンポーネントも同一のイテレータに関連付けられているため、これらすべてのinputTextコンポーネントがリフレッシュされます。
もう1つの例として、panelSplitterコンポーネントを含むページがあるとします。スプリッタの左側は新規顧客を作成するためのフォーム(CustomerInfoVO1コレクションを使用してフォームをドロップして作成)であり、スプリッタの右側にはフォームの用法の説明文が表示されています。フォームを構成する各inputTextコンポーネントのバインディングは、すべてCustomerInfoVO1Iteratorイテレータに関連付けられているため、フォームが送信され、いずれかのinputTextコンポーネントの値が変更されるたびに、CustomerInfoVO1Iteratorに関連付けられているinputTextコンポーネントのみがライフサイクル処理されます。スプリッタ右側の説明文を含め、ページの残りの部分はリフレッシュされません。
|
注意: 自動PPRが実行されるには、イベントおよびイベント・ルート・コンポーネントが、『Oracle Fusion Middleware Oracle Application Development Framework Webユーザー・インタフェース開発者ガイド』のイベントおよび部分ページ・レンダリングに関する項の表5-1内にリストされている必要があります。リストされていない場合は、同ガイドの、部分ページ・レンダリングの宣言的な有効化に関する項を参照して、PPRを手動で設定する必要があります。 たとえば、同一のデータ・コントロール・コレクションに基づきフォームと表を作成するとします。フォームにはデータ行の詳細が表示され、表にはコレクション内のすべての行が表示されます。フォーム内で現在表示されている行は、表内でも選択された状態で表示されます。フォームの「次へ」ボタンがクリックされたときに、新たな行が選択された状態に表を更新させる必要があります。選択イベントは自動PPRによってサポートされないため、表とフォームは同一のイテレータを使用しているにもかかわらず、表はリフレッシュされません。ターゲットの表に対するPPRのトリガーとして、「次へ」および「前へ」ボタンを設定する必要があります。 |
|
注意: 構成済イテレータに関連付けられたページ定義ファイル内でコントロール・バインディングが定義されているUIコンポーネントのみが、リフレッシュされます。 マネージドBeanからイテレータに直接アクセスし、このBeanを介してイテレータの値をUIコンポーネントに公開する場合は、これらのUIコンポーネントに対して自動PPRは実行されません。この場合は、PPRを手動で構成する必要があります。詳細は、『Oracle Fusion Middleware Oracle Application Development Framework Webユーザー・インタフェース開発者ガイド』の部分ページ・レンダリングの宣言的な有効化に関する項を参照してください。 |
|
注意: ADF Facesの問合せコンポーネントは、自動PPRをサポートしません。 |
デフォルトでは、すべての新規アプリケーションではグローバルPPRが使用されます。設定する必要はありません。ただし、ページに対するページ定義ファイル内で、この設定をオーバーライドできます。
作業を始める前に、次のようにします。
部分ページ・レンダリングに関する知識が役立つ場合があります。詳細は、25.2.1項「部分ページ・レンダリングおよびイテレータ・バインディングについて」を参照してください。
PPRを使用するようにイテレータ・バインディングを設定するには:
特定のイテレータ・バインディングに対し、PPRを使用するように、または使用しないように設定するには、関連付けられたページ定義ファイルを開き、イテレータを選択して、「プロパティ・インスペクタ」内でChangeEventPolicyを設定します。
すべてのイテレータ・バインディングがPPRを使用するように設定するには:
「アプリケーション・ナビゲータ」で「アプリケーション・リソース」パネルを開き、「ディスクリプタ」および「ADF META-INF」ノードを開いて、adf-config.xmlをダブルクリックします。
概要エディタで「モデル」タブを選択し、「デフォルトの変更イベント・ポリシーはPPRです」を選択します。
ADF FacesフレームワークでPPRが使用される方法の詳細は、『Oracle Fusion Middleware Oracle Application Development Framework Webユーザー・インタフェース開発者ガイド』の「部分ページ・コンテンツのレンダリング」を参照してください。
スケーラビリティ上の理由から、バインディング・コンテナ内のイテレータ・バインディングは、実行時に、行セット・イテレータへの参照を各リクエストの終了時に解放します。次のリクエスト中に、各イテレータ・バインディングにより、データ・コレクションの現在の行を追跡している有効な行セット・イテレータへのそれぞれの再バインドが行われます。ADFページ・ライフサイクルの中でADFイテレータ・バインディングを再バインドする処理は、イテレータのリフレッシュと呼ばれます。デフォルトでは、ライフサイクルの中でクライアント・レイヤーが初めてイテレータにアクセスしたときの1回のみ、必要に応じて発生します。この、イテレータへの初回のアクセスは、ページのレンダリング処理の中で、そのイテレータまたはイテレータに関係するコントロール・バインディングを参照するページ内のEL式が評価される際に発生します。もしくは、コードを記述して、イテレータへの初回のアクセスがレンダリングの準備フェーズの前に発生するようにプログラミングできます。イテレータ・バインディングによる行セット・イテレータのリフレッシュを、ライフサイクルのより早い段階で強制的に発生させるには、イテレータ・バインディングのrefresh属性をデフォルト以外の値に設定します。
|
ヒント: イテレータ・バインディングがリフレッシュされるたびに、問合せが強制的に再実行されることはありません。特定のユーザーのセッション中にビュー・オブジェクト・インスタンスの行セット・イテレータが初めてアクセスされる際、そのビュー・オブジェクトの問合せがまだ実行されていない場合には、問合せが暗黙的に実行されます(ただし、問合せの実行前にユーザーが検索条件を入力できるようにするための準備として、そのイテレータに関係するページ定義内の検索バインディングにより、その行セットが消去されていない場合にかぎられます)。 同じ論理作業ユニットの一部であるページ・リクエストでそのビュー・オブジェクト・インスタンスに関連するイテレータ・バインディングをこの後でリフレッシュすると、行セット・イテレータへの再アクセスのみが行われ、問合せの再実行は強制されません。問合せを再実行してそのデータをリフレッシュするには、 |
refreshおよびrefreshCondition属性は、イテレータ・バインディングなどの実行可能ファイルとinvokeActionのどちらをどのタイミングで起動するかを決めるために使用します。refresh属性の値によって、実行可能ファイルを起動するライフサイクル・フェーズが決定し、refreshCondition属性の値によって提供されるオプションのブール条件の結果で、指定のライフサイクル・フェーズでリフレッシュが発生するかどうかが決まります。デフォルトでは、(操作の1つをコマンド・コンポーネントとしてドロップするなどして)ページ定義に実行可能ファイルが追加されると、その実行可能バインディングのrefresh属性が、バインディングへの初回アクセス時には常に実行を強制するdeferredに設定されます。refreshCondition値が存在しない場合は、実行可能ファイルが起動します。refreshConditionの値が存在する場合は、値がEL式として評価され、評価の戻り値がtrueの場合は、実行可能ファイルが起動します。値がfalseと評価された場合は、実行可能ファイルは起動しません。refresh属性の詳細は、A.8.1項「PageDef.xmlの構文」を参照してください。
Fusion Webアプリケーションでは、イテレータ・バインディングのrefreshまたはrefreshConditionの値の変更が必要になることはほとんどありません。これらの値は、正しいデータの表示を確実にするために設定されます。
invokeAction実行可能ファイルは、以前のリリースからの上位互換性のためにサポートが継続されますが、Oracle ADF 11gでは、タスク・フローのメソッド・アクティビティを使用して、ページがレンダリングされる前にアプリケーション動作を実行するアクション・バインディング(または任意のバッキングBeanメソッド)をコールします。たとえば、オブジェクトの作成に使用するページには、CreateInsert操作をコールするメソッド・アクティビティで始まるタスク・フローを使用できます。そのタスク・フローは次に、ユーザーがデータを入力するページのビュー・アクティビティに進みます。この動作を適切なメソッド・コール・アクティビティとしてモデル化すると、アプリケーションが自己文書化される上に管理しやすくなり、アプリケーション・ロジックとデータバインディングがより明確に分離されます。詳細は、26.6項「入力フォームの作成」を参照してください。
ただし、完全性を期すために、イテレータ・バインディングまたはinvokeActionのrefresh属性を変更する状況が発生した場合は(たとえば、モデルの準備フェーズ後に実行されるプログラム・コードがあり、イテレータにプログラムでアクセスする必要がある場合)、有効な値に関する次の情報を確認した上で、変更を行う必要があります。
deferred (デフォルト): 要求時。
|
ヒント: ページ定義ファイルの |
prepareModel: モデルの準備フェーズ中。
renderModel: レンダリングの準備フェーズ中。
|
ヒント: 図25-3からわかるとおり、モデルの準備フェーズとレンダリングの準備フェーズの主な違いは、一方がJSFのアプリケーションの起動フェーズの前にあり、他方が後にあることです。JSFのアプリケーション起動フェーズではアクション・リスナーが起動されるため、これらのアクション・リスナーによってそれぞれの処理が実行された後に、イテレータのリフレッシュ、または |
ifNeeded: モデルの準備フェーズおよびレンダリングの準備フェーズ中(必要な場合のみ)。イテレータに関しては、バインディングがまだリフレッシュされていない場合に、リフレッシュが必要であるとみなされます。invokeActionの実行が必要かどうかを判断するため、フレームワークは、評価されたパラメータ値の現在のセットを、前回のメソッド・アクション・バインディングの起動に使用されたセットと比較します。現在の起動に対するパラメータ値が前回使用した値と完全に同じ場合、invokeActionは、バインドされているメソッド・アクション・バインディングを起動しません。この設定を使用して、invokeAction実行可能ファイルがパラメータを受け入れるメソッド・アクション・バインディングにバインドされるかどうかを判断します。
|
ヒント: パラメータを取らないメソッドにバインドされている |
prepareModelIfNeededおよびrenderModelIfNeeded: ifNeededと同様ですが、名前付きフェーズ中に実行されます。
never: invokeAction実行可能ファイルでは無効です。イテレータの場合、イテレータがリフレッシュされることはありません。独自のコードがイテレータ・バインディングでgetRowSetIterator()をコールする場合に使用します。
always: invokeAction実行可能ファイルでは無効です。イテレータの場合、イテレータは常に、モデルの更新フェーズ中に加えて、モデルの準備とレンダリングの準備の両フェーズ中にリフレッシュされます(複数回の場合もあります)。
refreshAfter: 実行可能ファイル間の依存性の処理に使用します。たとえば、別の実行可能ファイルの後にこの実行可能ファイルがリフレッシュされるように条件を設定できます。
|
ヒント: 実行可能ファイルの起動順序は、 |
タスク・フローは、(ページに関連付けられている)親バインディング・コンテナのリフレッシュ時に初めてリフレッシュされます。これはモデルの準備フェーズで発生します。後続のリクエストでは、レンダリングの準備フェーズにおいて、refreshおよびrefreshCondition属性とそのパラメータ値に基づいてタスク・フローがリフレッシュされます。
|
注意: 子ページ・フラグメントのページ定義は、引き続き子ページ・フラグメントのバインディングのリフレッシュを処理します。 |
|
ヒント: 最初は公開されないリージョン(ポップアップ・ダイアログなど)がページ上にある場合、リージョンが表示されなくても、親ページのレンダリング時にパラメータは使用可能である必要があります。リージョンにパラメータが必要であるにもかかわらず、これらのパラメータ値を親ページのレンダリング時に使用できない場合は、動的リージョンを使用する必要があります。パラメータがNULLの場合、リージョンのパラメータの準備が完了してリージョンを表示できるようになるまで、空のタスク・フローを使用できます。空のタスク・フローに切り替えるには、動的リージョンの |
EL式をrefreshCondition属性の値として設定した場合、ライフサイクルのレンダリングの準備フェーズ中に評価されます。式がtrueに評価されると、タスク・フローは再度リフレッシュされます。refreshConditionがfalseに評価されると、refreshConditionが指定されていない場合と動作は同じです。
|
注意: EL式で |
タスク・フロー実行可能ファイルのrefreshプロパティの有効な値は次のとおりです。
default: 親ページが最初に表示されるときに、リージョンは1回のみリフレッシュされます。
ifNeeded: taskFlowバインディングのパラメータ値に変更があった場合のみ、リージョンがリフレッシュされます。taskFlowバインディングにパラメータがない場合、ifNeededはデフォルトと同じです。ifNeededを使用した場合、refreshCondition属性は考慮されません。
|
注意:
|
taskFlowバインディングによる唯一の処理はそのパラメータのリフレッシュであり、Refreshをalwaysに設定する意味はありません。taskFlowバインディングのパラメータが変更されないかぎり、ADFリージョンをリフレッシュする理由はありません。
子ページ・フラグメントのページ定義は、引き続き子ページ・フラグメントのバインディングのリフレッシュを処理します。
実行時には、バインディング・コンテナおよびマネージドBeanなどのADFオブジェクトはインスタンス化されています。これらのオブジェクトにはそれぞれ、そのスコープ属性により設定された存続期間が定義されています。RequestContext APIからjava.util.Mapとしてスコープにアクセスできます。たとえば、リクエスト・スコープのfooというオブジェクトにアクセスするには、式#{requestScope.foo}を使用します。
Fusion Webアプリケーションには次の6種類のスコープがあります。
アプリケーション・スコープ: オブジェクトは、アプリケーションが継続する間使用できます。
セッション・スコープ: オブジェクトは、セッションが継続する間使用できます。
|
注意: セッション・スコープにはウィンドウの一意性はなく、セッション内のすべてのウィンドウは、同じセッション・スコープ・インスタンスを共有します。複数のウィンドウが同じオブジェクトにアクセスできることが不適切な場合(たとえば、ウィンドウ間でマネージドBeanが競合しないようにする場合)は、ページ・フロー・スコープやビュー・スコープなどのウィンドウ固有のスコープを使用する必要があります。 |
ページ・フロー・スコープ: オブジェクトは、バインド・タスク・フローが継続する間使用できます。
|
注意: これは標準JSFスコープではないため、Beanを参照するためのスコープをEL式に明示的に含める必要があります。たとえば、 |
リクエスト・スコープ: オブジェクトは、HTTPリクエストが作成された時点からレスポンスがクライアントに戻されるまで使用できます。
バッキングBeanスコープ: ページ・フラグメントおよび宣言コンポーネントのみのマネージドBeanに使用されます。このオブジェクトは、HTTPリクエストが発生してからレスポンスがクライアントに返送されるまでの間、利用できます。1つのページに複数のページ・フラグメントまたは宣言コンポーネントが含まれる場合があり、競合を避けるには別々のスコープ・インスタンスに値を保持する必要があるため、フラグメントおよび宣言コンポーネントにはこのスコープが必要です。したがって、すべてのページ・フラグメントまたは宣言コンポーネントのマネージドBeanで、バッキングBeanスコープを使用する必要があります。
|
注意: これは標準JSFスコープではないため、Beanを参照するためのスコープをEL式に明示的に含める必要があります。たとえば、バッキングBeanスコープから |
ビュー・スコープ: このオブジェクトは、現在のビュー・アクティビティのビューIDが変更されるまで利用できます。このスコープは、所定のページの値を保持するために使用できます。ただし、1つのページから次のページに必要な値を格納するために使用できるリクエスト・スコープとは異なり、ビュー・スコープに格納されたものはビューIDの変更とともに失われます。
|
注意: JSFおよびADF Facesはいずれも、ビュー・スコープを実装しています。ADF Facesのビュー・スコープのみが、ページのリダイレクトおよびリフレッシュの後も維持されます。Fusion Webアプリケーションでは、式で このビュー・スコープは標準JSFスコープではないため、Beanを参照するためのスコープをEL式に明示的に含める必要があります。たとえば、ビュー・スコープから |
|
注意: スコープを定義する必要があるオブジェクト(マネージドBeanなど)を作成する場合、スコープを |
オブジェクト・スコープは、プログラミング言語におけるグローバル変数スコープおよびローカル変数スコープに類似しています。スコープの幅が広くなるほど、オブジェクトの可用性が高まります。オブジェクトの存続期間中、これらのオブジェクトによってあるインタフェースが公開され、情報が保持され、あるいは変数およびパラメータが他のオブジェクトに渡されます。たとえば、セッション・スコープで定義されたマネージドBeanは、ページ・リクエストが複数存在する場合に使用できます。ただし、リクエスト・スコープで定義されたマネージドBeanは、1つのページ・リクエストが継続する間のみ使用できます。
デフォルトでは、バインディング・コンテナおよびそれに含まれるバインディング・オブジェクトは、セッション・スコープで定義されます。ただし、値バインディングおよびイテレータ・バインディングによって参照される値はリクエスト間で定義されず、スケーラビリティ上の理由により、セッション・スコープに残りません。このため、バインディング・オブジェクトが参照する値は、そのバインディング・コンテナがADFライフサイクルによって準備されているリクエストの間のみ有効です。セッション・スコープには、バインディング・コンテナおよびバインディング・オブジェクト自体のみ残されます。
図25-4に、各タイプのスコープが有効な期間を示します。
マネージドBeanをどのスコープに登録するかを決定するときは常に、できるかぎり狭いスコープを使用するようにします。セッション・スコープは、セッション全体に関連する情報(ユーザー情報やコンテキスト情報など)にのみ使用します。タスク・フロー間で値を渡すためにセッション・スコープを使用しないでください。ページ・フラグメントまたは宣言コンポーネントのマネージドBeanを作成するときは、バッキングBeanスコープを使用する必要があります。
マネージドBeanは、adfc-config.xmlまたは特定のタスク・フローの構成ファイルに登録できます。FusionアプリケーションでのマネージドBeanの使用に関する詳細は、24.4項「Fusion WebアプリケーションでのマネージドBeanの使用」を参照してください。
|
注意: Fusion Webアプリケーションでは、 |
タスク・フロー内の変数にどのスコープを使用するかを決定するときは、アプリケーション・スコープまたはセッション・スコープ以外のスコープ・オプションを使用する必要があります。これら2つのスコープは、タスク・フローの存続期間を超えてオブジェクトをメモリーに保存するため、タスク・フローのカプセル化および再利用で障害が発生します。さらに、アプリケーション・スコープおよびセッション・スコープでは、オブジェクトがメモリーに必要以上に長い時間保持されるため、不要なオーバーヘッドが発生します。
タスク・フロー内のアクティビティ間でデータ値を渡す必要がある場合は、ページ・フロー・スコープを使用する必要があります。現在のビュー・アクティビティ内でのみ必要とされ、ビュー・アクティビティ間にまたがっていない変数の場合、ビュー・スコープがお薦めされます。現在のリクエストより長い期間スコープが存続する必要がない場合には、リクエスト・スコープを使用する必要があります。これは、UIコンポーネント情報の格納に使用する唯一のスコープです。最後に、タスク・フローが同じページの2つのリージョン・コンポーネントまたは宣言コンポーネントに表示される可能性があり、リージョン・インスタンスを分離する場合、タスク・フロー内のマネージドBeanには、バッキングBeanスコープを使用する必要があります。
ADFライフサイクルには明確に定義されたフェーズが含まれており、このフェーズによって対応するJSFフェーズの実行前後にADFライフサイクル・リスナーに通知が行われます。このライフサイクルは、必要なコードを起動するカスタム・フェーズ・リスナーを作成した後、これらのフェーズのいずれかで実行されるようにライフサイクルに登録することによってカスタマイズできます。
たとえば、アプリケーション・モジュールでデフォルトの管理状態解放レベルを使用しない場合は、カスタムADFページ・フェーズ・リスナー・クラスを作成することで、ADFライフサイクルのafter-prepareRenderフェーズから解放レベルを設定できます(詳細は43.4.5項「ADF PagePhaseListenerでの解放レベルの設定方法」を参照)。
|
注意: 1つのアプリケーションでは、複数のフェーズ・リスナー・インスタンスを保持できません。デフォルトでは、最初の 警告メッセージ「ADFc: ADFページ・ライフサイクル実装を''新しいリスナーのクラス名''と置換しています。」により、インスタンスがより新しいものに置き換えられたことが通知されます。 |
カスタム・フェーズ・リスナーを作成するには、PagePhaseListenerインタフェースを実装するリスナー・クラスを作成する必要があります。次に、コードの実行が必要なフェーズの前または後にコードを実行するメソッドを追加します。
|
注意: JSF |
例25-2に、カスタム・フェーズ・リスナーを作成するために変更できるテンプレートを示します。JDeveloperでクラスを作成する方法の詳細は、4.13.1項「カスタム・クラスの生成方法」を参照してください。
例25-2 カスタム・フェーズ・リスナーの例
package mypackage.listeners;
import oracle.adf.controller.v2.lifecycle.ADFLifecycle;
import javax.faces.context.FacesContext;
import oracle.adf.controller.v2.lifecycle.PagePhaseEvent;
import oracle.adf.controller.v2.lifecycle.PagePhaseListener;
import oracle.fodemo.storefront.adfextensions.FODApplicationModuleImpl;
public class MyPagePhaseListener implements PagePhaseListener
{
public void afterPhase(PagePhaseEvent event)
{
System.out.println("In afterPhase " + event.getPhaseId());
}
public void beforePhase(PagePhaseEvent event)
{
System.out.println("In beforePhase " + event.getPhaseId());
}
}
カスタム・リスナー・クラスを作成したら、クラスを起動する必要があるフェーズに登録する必要があります。アプリケーション全体で使用できるようにグローバルに登録するか、1つのページに対してのみ登録することができます。
ADFライフサイクルをグローバルにカスタマイズするには、adf-settings.xml構成ファイルを編集してカスタム・フェーズ・リスナーを登録します。adf-settings.xmlファイルは、ADFコントローラなど複数のADFコンポーネントにより共有され、構成情報を格納します。
adf-settings.xmlにリスナーを登録する手順:
adf-setting.xmlファイルを開きます。アプリケーション開発の状況によっては、このファイルは「アプリケーション・リソース」パネルの「ディスクリプタ」→「META-INF」ノードに置かれている可能性があります。ここに存在しない場合は、これを手動で開く必要があります。これはview_project/src/META-INFディレクトリ内にあります。
エディタ・ウィンドウで「ソース」タブをクリックします。
ソース・エディタで、まで下にスクロールします。
<adfc-controller-config xmlns=
"http://xmlns.oracle.com/adf/controller/config">
このエントリが存在しない場合は、これをファイルに追加します。
イタリックで示されている残りの要素を入力します。
<?xml version="1.0" encoding="US-ASCII" ?> <adf-config xmlns="http://xmlns.oracle.com/adf/config">
.
.
.
<adfc-controller-config xmlns="http://xmlns.oracle.com/adf/controller/config">
<lifecycle>
<phase-listener>
<listener-id>MyPagePhaseListener</listener-id>
<class>mypackage.MyPagePhaseListener</class>
</phase-listener>
</lifecycle>
</adfc-controller-config>
.
.
.
</adf-config>
次の要素の値を入力します。
<listener-id>: リスナーの一意の識別子(完全修飾クラス名を使用可能)
<class>: リスナーのクラス名
adf-settings.xmlでは複数のフェーズ・リスナーを指定できるほか、これらの相対的なコール順序も任意で指定できます。このファイルに新しいリスナーを登録する際には、次の2つのパラメータを使用して、リスナー・リストでの配置を指定します。
beforeIdSet: そのリスナーが、beforeIdSetで指定されているリスナーよりも先にコールされます。
afterIdSet: そのリスナーが、afterIdSetで指定されているリスナーよりも後にコールされます。
例25-3に、アプリケーション用に複数のリスナーが登録された構成ファイルの例を示します。
例25-3 複数のリスナー登録のあるadf-settings.xml構成ファイル
<lifecycle>
<phase-listener>
<listener-id>MyPhaseListener</listener-id>
<class>view.myPhaseListener</class>
<after-id-set>
<listener-id>ListenerA</listener-id>
<listener-id>ListenerC</listener-id>
</after-id-set>
<before-id-set>
<listener-id>ListenerB</listener-id>
<listener-id>ListenerM</listener-id>
<listener-id>ListenerY</listener-id>
</before-id-set>
</phase-listener>
</lifecycle>
例では、MyPhaseListenerは登録されたリスナーであり、リスナーAおよびCの後かつ、リスナーB、MおよびYより先に実行されます。リスナーBの後にMyPhaseListenerを実行するには、リスナーBの<listener-id>要素を<after-id-set>要素の下に移動します。
単一ページのライフサイクルをカスタマイズするには、ページ定義ファイルでControllerClass属性を設定します。このリスナーが有効なのは、ページ定義によって記述された特定ページのライフサイクルに対してのみです。ページ定義ファイルとFusion Webアプリケーションにおけるその役割の詳細は、13.7項「ページ定義ファイルでの作業」を参照してください。
標準JSFページまたはページ・フラグメントのどちらに対するものかに応じて、異なるコントローラ・クラスを指定します。
単一ページまたはページ・フラグメント用にADFライフサイクルをカスタマイズする手順:
アプリケーション・ナビゲータで、ページまたはページ・フラグメントを右クリックし、「ページ定義に移動」を選択します。
構造ウィンドウで、ページ定義ノードを選択します。
「プロパティ・インスペクタ」で「ControllerClass」の隣のドロップダウン・メニューをクリックし、「編集」を選択します。
「階層」タブをクリックし、ページまたはページ・フラグメントの目的のコントローラ・クラスに移動します。各種ページに使用するコントローラ・クラスを次に示します。
標準JSFページの場合: oracle.adf.controller.v2.lifecycle.PageControllerを指定します。
afterPhase/beforePhaseイベントを受信する必要がある場合は、oracle.adf.controller.v2.lifecycle.PagePhaseListenerを指定します。
ページ・フラグメント: 次を指定します。
oracle.adf.model.RegionController
|
ヒント: 完全修飾クラス名としてページ定義の
|
リージョンとして使用されるページ・フラグメント内のバインディングは、RegionControllerインタフェースのrefreshRegionおよびvalidateRegionイベントを介してリフレッシュされます。これらのイベントは、25.4.4項「単一ページのライフサイクル・リスナーの登録方法」の説明に従ってControllerClassフィールドにoracle.adf.model.RegionControllerを指定した場合に使用できます。
例25-4に示すように、refreshRegionイベントを使用して、リージョンがリフレッシュされる前に実行するカスタム・コードを追加できます。たとえば、リージョン内のページ・フラグメントにより使用されるバインディングをリフレッシュして、リフレッシュされたバインディング値が内部バインディング・コンテナに伝播されるようにすることができます。
そのためには、RegionControllerインタフェースを実装する新規のクラスを作成します。その後、モデルの準備フェーズの前に実行するカスタム・コードも含めて、次のrefreshRegionメソッドを記述します。
例25-4 regionRefreshメソッド
public boolean refreshRegion(RegionContext regionCtx)
{
int refreshFlag = regionCtx.getRefreshFlag();
if (refreshFlag == RegionBinding.PREPARE_MODEL)
{
// Execute some code before
}
// Propagate the refresh to the inner binding container
regionCtx.getRegionBinding().refresh(refreshFlag);
return false;
}
public boolean validateRegion(RegionContext regionCtx)
{
// Propagate the validate to the inner binding container
regionCtx.getRegionBinding().validate();
return false;
}
例25-4に示すように、リフレッシュ・フラグの値は次のようになります。
RegionBinding.PREPARE_MODEL: ADFライフサイクルprepareModelフェーズ時に発生するイベントに対応
RegionBinding.RENDER_MODEL: ADFライフサイクルprepareRenderフェーズ時に発生するイベントに対応