ページの先頭行へ戻る
Interstage Application Server Java EE運用ガイド(Java EE 6編)
Interstage

7.2.1 永続性コンテキストキャッシュ

エンティティマネージャを使用してEntityをデータベースから取得する場合、または作成したEntityを永続性コンテキストにマージする場合、Entityインスタンスが永続性コンテキストキャッシュに格納されます。このキャッシュのライフサイクルは永続性コンテキストと同じです。

以下に永続性コンテキストキャッシュの特徴を説明します。

findメソッド

エンティティマネージャのfindメソッドでEntityを検索すると、まずは永続性コンテキストキャッシュから探します。
キャッシュにある場合は、データベースに問い合わせずにキャッシュされたインスタンスを返却します。キャッシュにない場合は、「7.2.2 永続性ユニットの共有キャッシュ」を探します。共有キャッシュにもない場合、SQL文を発行しデータベースに問い合わせします。検索が成功した場合、Entityインスタンスを永続性コンテキストキャッシュに格納し、アプリケーションに返却します。

クエリメソッド

エンティティマネージャを使用してJPQLクエリメソッドを実行すると、SQL文を発行しデータベースから検索します。ただ、レコードを見つけた場合、レコードと同じプライマリキーのEntityがすでに永続性コンテキストキャッシュに存在しているかどうか確認します。
存在している場合は、現在の永続性コンテキストで行った更新やEntity間のリレーション関係の整合性を保つため、キャッシュしたインスタンスを返却します。
存在していない場合は、永続性ユニットの共有キャッシュを探します。共有キャッシュにもない場合、SQL文を発行しデータベースに問い合わせします。検索が成功した場合、Entityインスタンスを永続性コンテキストキャッシュに格納し、アプリケーションに返却します。

Entityインスタンスの更新、マージ、削除

管理されているEntityインスタンスの更新、削除、または非管理状態のEntityインスタンスの永続性コンテキストへのマージは永続性コンテキストキャッシュに反映されます。

データベースのレコードとの不整合について

エンティティマネージャを使用し各操作を実行する際、同一トランザクションや別トランザクションにおいて、そのエンティティマネージャ以外のアクセス方法でデータベースの同一レコードを更新すると、永続性コンテキスト内で管理されたEntityインスタンスとデータベースのレコード間で不整合が発生する可能性があります。
これを回避するためには、エンティティマネージャの各操作に対するフラッシュやEntityデータの最新化を必要に応じて明示的に実行してください。
また、Entityデータを最新化する方法については以下に説明します。

Entityデータの最新化

Entityインスタンスの永続化フィールドの最新データをデータベースから取得するため、以下の方法があります。

エンティティマネージャのrefreshメソッドを実行

Entityインスタンスをエンティティマネージャのrefreshメソッドに指定し本メソッドを実行すると、SQL文が発行され、レコードの最新データが取得されます。最新データがEntityインスタンスに反映され永続性コンテキストキャッシュも更新されます。

JPQLクエリメソッドにInterstage永続性プロバイダのクエリヒントを使用

JPQLクエリーにリフレッシュのヒントを指定しクエリメソッドを実行すると、永続性コンテキストキャッシュと共有キャッシュを検索せずにデータベースに問い合わせます。成功した場合、レコードを取得してEntityインスタンスのコピーを作成して、コピーを共有キャッシュに格納し、もとのEntityインスタンスをアプリケーションに返却します。永続性コンテキストキャッシュにそのEntityがすでに存在した場合、既存のオブジェクトが非管理状態になり、まだフラッシュしなかった修正がデータベースに更新されません。共有キャッシュにEntityがすでに存在した場合、既存のオブジェクトを新しいオブジェクトに上書きします。


注意

データの最新化の各機能における注意事項を以下に説明します。

  • エンティティマネージャのrefreshメソッドを実行する場合

    • コンテナ管理のエンティティマネージャを使用した場合、このメソッドをトランザクション中しか実行できないため、データ更新のない操作でもトランザクションを開始する必要があります。

    • refreshメソッドに指定するEntityインスタンスを取得した時にもSQL文を発行するので、データベースへ2回アクセスします。

  • Interstage永続性プロバイダのクエリヒントを使用する場合

    • 永続性コンテキストですでに行った更新やEntity間のリレーション関係の整合性を保てません。

    • Interstage永続性プロバイダの独自機能であるため、その他の永続性プロバイダで該当するヒントがない可能性があり、移植性に影響があります。

リフレッシュ機能は以下のクエリヒントで指定します。

クエリヒント名

値 (太字:省略値)

説明

QueryHints.REFRESH

HintValues.TRUE

キャッシュに問い合わせせずにデータベースを検索します。検索したEntityのコピーを作成しコピーを共有キャッシュに格納します。もとのEntityインスタンスをアプリケーションに返却します。

HintValues.FALSE

データベースからの検索結果をキャッシュ内のEntityと比較し、すでに存在した場合、キャッシュされたEntityインスタンスをアプリケーションに返却します。

QueryHints.REFRESH_CASCADE

CascadePolicy.CascadeAllParts

検索したEntityをリフレッシュします。このEntityとリレーションが持つEntityが検索結果に含まれている場合、そのEntityもリフレッシュします。

CascadePolicy.CascadeByMapping

検索したEntityをリフレッシュします。リレーションのcascade定義にCascadeType.REFRESHまたはCascadeType.ALLを指定したEntityもリフレッシュします。その他のリレーションのEntityインスタンスをリフレッシュしません。

CascadePolicy.NoCascading

検索したEntityをリフレッシュしますが、このEntityインスタンスとリレーションが持つEntityインスタンスはキャッシュ内に存在した場合、データをリフレッシュせずにキャッシュしたオブジェクトを返却します。

QueryHints.REFRESH_CASCADEヒントの設定を有効にするためには、QueryHints.REFRESHヒントをHintValues.TRUEに指定する必要があります。

クエリヒントはマッピング定義ファイルdeployment descriptor (orm.xml)のquery-hintタグ、アノテーション(@QueryHint)、またはQuery.setHintメソッドで指定します。

リフレッシュのヒントの定義例

@NamedQuery(name="Order.findOrdersByPK",
  query="SELECT o FROM Order o WHERE o.id = :id",
  hints={@QueryHint(name=QueryHints.REFRESH,
                    value=HintValues.TRUE),
         @QueryHint(name=QueryHints.REFRESH_CASCADE,
                    value=CascadePolicy.CascadeAllParts)})