OCI8
PHP Manual

接続のハンドリング

接続関数

oci8 拡張モジュールは Oracle に接続するための 3 つの異なる関数を提供しています。標準の接続関数は oci_connect() です。これは Oracle データベースへの接続を作成し、 それ以降のデータベースで使うリソースを返します。

Oracle サーバへの接続は、完了まで要する時間という点から見ると、 かなりコストのかかる操作です。oci_pconnect() 関数は、 異なるスクリプトリクエスト間で接続の再利用が可能な 持続的キャッシュを使用します。 これは、PHP プロセス (もしくは Apache の子プロセス) 毎の接続に関するオーバーヘッドを一度のみ負うということを意味しています。

もしアプリケーションが信用された異なる Web ユーザー毎に Oracle に接続する場合、oci_pconnect() による持続的キャッシュは、 同時ユーザー数の増加と共に有効ではなくなるでしょう。 これは、多くのアイドル状態の接続が維持されることが原因で、 Oracle サーバ全体のパフォーマンスに不利な影響を与え始めるためです。 もしアプリケーションがこの方法で構成されている場合、 oci8.max_persistentoci8.persistent_timeout (持続的接続のキャッシュサイズや生存期間の制御が可能になります) を使用してアプリケーションをチューニングする、もしくは代わりに oci_connect() を使用することが推奨されます。

oci_connect()oci_pconnect() の両者とも接続キャッシュを使用します。もし、同一パラメータと共に oci_connect() を複数回コールする場合、 2 番目以降は既存の接続ハンドルを返します。oci_connect() によって使用されるキャッシュは、スクリプト実行終了時、 もしくは明示的に接続ハンドルを閉じた時にクリアされます。 oci_pconnect() も同様の動作をしますが、 キャッシュは独立して維持され、リクエスト間で残存します。

このキャッシュ機能は忘れてはならないほど重要です。 それは、2 つのハンドルがトランザクション的に独立していない (実際には同じ接続なので、どのような種類の独立もありません) ためです。もしアプリケーションが 2 つの別々でトランザクション的に独立した接続を必要とする場合、 oci_new_connect() を使用すべきです。

PHP プロセス終了時に oci_pconnect() キャッシュは消去され、 データベース接続は全て閉じられます。 そのため、持続的接続を効果的に使用するには、 PHP は Apache のモジュールであるか、または FCGI によって使用されるか、 または同様のものでなければいけません。 PHP が CGI によって、またはコマンドラインを介して使用される場合、 持続的接続には oci_connect() 以上に全く利益がありません。

oci_new_connect() は、他の既存の接続が存在したとしても 常に Oracle サーバへの新規接続を生成します。 特にアプリケーションの最も負荷が高い部分など、 高トラフィックな Web アプリケーションに対しては oci_new_connect() の使用を避けてください。

DRCP 接続プーリング

PHP 5.3 (PECL OCI8 1.3) では、 Oracle 11g のデータベース常駐接続プーリング (DRCP) をサポートします。 DRCP によりデータベースマシンのメモリをより効率的に使用し、 高い拡張性が得られます。 DRCP を使うためにアプリケーションを変更する必要はないか、または 最小限です。

DRCP は、ごく少数のデータベーススキーマを使用し、データベース接続を短時間オープン状態に 保つアプリケーションに適しています。 その他のアプリケーションは、 Oracle のデフォルトの Dedicated データベースサーバプロセスか、または Shared サーバを使用しなければいけません。

DRCP は3つの接続機能全てに有益ですが、 oci_pconnect() で接続を作成すると最高の拡張性が得られます。

OCI8 で DRCP を利用可能にするには、 PHP で使用する Oracle クライアントライブラリ、 及び Oracle データベースのバージョンが共に 11g でなければいけません。

DRCP についてのドキュメントはいくつかの Oracle マニュアルに見つかります。 例えば、使用法の情報のために、 Oracle ドキュメントで » データベース常駐接続プーリングの構成 をご覧ください。 » DRCP ホワイトペーパー には、 DRCP についての 予備知識となる情報が含まれています。

DRCP を使用するには、 OCI8 1.3 エクステンション及び Oracle 11g ライブラリと共に PHP をビルドし、これらのステップを続けます。

注意:

持続的接続のパフォーマンスを必要とする Oracle 10g を使うアプリケーションでは、 Oracle Shared サーバ(マルチスレッドサーバとして既知)を使用することにより、 必要なデータベース・メモリー量を減らせます。 詳細は Oracle ドキュメントを参照してください。

DRCP 推奨事項と既知の制約

Changing a password over DRCP connections will fail with the error ORA-56609: Usage not supported with DRCP. This is a documented restriction of Oracle Database 11g.

With the OCI8 1.3 extension, 持続的接続 can now be closed by the user, allowing greater control over connection resource usage. 持続的接続 will now also be closed automatically when there is no PHP variable referencing them, such as at the end of scope of a PHP user function. This will rollback any uncommitted transaction. These changes to persistent connections make them behave similarly to non-persistent connections, simplifying the interface, allowing for greater application consistency and predictability. Use oci8.old_oci_close_semantics set to On to retain the historical behavior.

If the Oracle Database is version 11.1.0.6, then the Oracle database patch for Oracle bug 6474441 must be applied to use DRCP. Without this patch, errors such as ORA-01000: maximum open cursors exceeded, ORA-01001 invalid cursor or ORA-01002 fetch out of sequence may occur. This bug was fixed in Oracle 11.1.0.7 onwards.

If the Oracle 11.1.0.6 database patch cannot be applied, one of the following three workarounds can be used instead:

Oracle Database 11.1.0.7 and the Oracle Database 11.1.0.6 patch for Oracle bug 6474441 allow PHP applications with DRCP connection to use a database LOGON trigger to set session properties at the time of session creation. Examples of such settings are the NLS language and the date format.

If the Oracle 11.1.0.6 database patch cannot be applied, one of the following workarounds can be used instead of using LOGON triggers:

The automatic re-establishment of PHP 持続的接続 after an Apache or FGCI process respawns means LOGON triggers in PHP are only recommended for setting session attributes and not for per-application user connection requests. This is even more so with DRCP due to the automatic pool sizing and with the way LOGON triggers fire with DRCP authentication.

Fast Application Notification (FAN) サポート

FAN サポートにより、高速な接続フェールオーバが得られます。これは高可用性フィーチャーです。 これにより、データベース・マシンまたはデータベース・インスタンスが利用できなくなると、 PHP OCI8 スクリプトに通知されます。 FAN 無しでは、 TCP のタイムアウトが起きてエラーが返されるまで OCI8 はハングする場合があります。 それは数分間かもしれません。 OCI8 で FAN を有効にすると、 web のユーザーが機能不全を気づかないように、 アプリケーションでエラーを検出して利用可能なデータベース・インスタンスに再接続できます。

FAN サポートは、 PHP がリンクする Oracle クライアント・ライブラリーと Oracle データベースがバージョン 10gR2 かまたは 11g の場合に利用可能です。

FAN は、 Oracle の クラスタリング・テクノロジー (RAC) の利用者に有益です。 なぜなら、生き残ったデータベース・インスタンスへの接続を直ちに作成できるからです。 ブローカーを持つ Oracle の Data Guard の利用者は、 スタンバイ・データベースがオンラインになる際に生成される FAN イベントに気づくでしょう。 スタンドアロンのデータベースは、データベース再起動の際に FAN イベントを送信します。

アクティブな接続に関して、マシンまたはデータベース・インスタンスが利用できなくなると、 現在コールされている OCI エクステンションの関数によって、 接続失敗エラーが返されます。 以降の PHP スクリプト再接続で、 生き残ったデータベース・インスタンスへの接続が確立されます。 データベースマシンやインスタンスの失敗で影響を受けたアイドル状態の接続を OCI8 エクステンションも全て透過的にクリーンアップします。 こうして、スクリプトでサービスの中断に気づくことなく、 PHP 接続コールにより新たな接続が確立されます。

oci8.eventsOn の場合、 疎通確認を無効にするために oci8.ping_interval を -1 に設定することを提案します。 なぜなら、 FAN イベントを有効にすると、アイドル状態の接続の能動的な接続管理を サービスの中断によって無効にしてしまうからです。

PHP で FAN サポートを有効にするには、 Oracle 10gR2 または 11g ライブラリと共にPHP をビルドし、これらのステップを続けます。


OCI8
PHP Manual