PHPでは、 SessionHandlerは、セッションデータの保存、読み取り、更新、および破棄方法を定義する抽象クラスです。 sessionhandler :: closeは、PHPセッションが終了したときに自動的に呼び出されるライフサイクル方法であるため、シャットダウン操作が実行され、リソースが解放されます。デフォルトでは、セッションファイルを閉じながら、セッションデータの書き込みと保存を閉じます。
ただし、開発者がセッションストレージメカニズムをカスタマイズすると、 SessionHandler :: closeの動作が影響を受ける可能性があり、データが適切に保存されないか、データの損失さえもなります。
session_write_close()は早すぎると呼ばれます
session_write_close()がコードで呼び出された場合、セッションデータをストレージメディア(ファイル、データベースなど)に書き込むなど、現在のセッションの操作が強制されます。この時点で、 SessionHandler :: closeメソッドは事前にトリガーされる場合があり、その結果、データは正しく記述されません。
たとえば、データストレージまたは変更操作を実行した直後にsession_write_close()に電話すると、他のタスクが完了するのを待つことなく、データの永続性プロセスが中断され、セッションデータの一部またはすべてが失われる可能性があります。
不適切なカスタムセッションストレージメカニズム
特にデータが保存されて読み取られた場合、カスタムセッションハンドラークラスを実装する場合、緊密なメソッドが適切に実装されていない場合、セッションデータは完全に記述されない場合があります。たとえば、閉じる方法は書き込み操作を正しく呼び出さないか、実行中にエラーが発生し、データが保存されません。
データベースストレージメカニズムをカスタマイズし、閉じるメソッドでデータベース接続をコミットまたはクリーンアップするのを忘れた場合、データはデータベースに保存されない場合があります。
SESSION_SAVE_PATH設定エラー
SESSION_SAVE_PATHは、PHPが使用する構成アイテムであり、セッションデータストレージの場所を指定します。パスが正しく設定されていない場合、またはPHPにディレクトリに書き込むのに十分な権限がない場合、 sessionhandler :: close :: closeは、書き込み操作を実行するときに故障する可能性があり、セッションデータが失われます。
この場合、閉じる方法は書き込みタスクを正しく完了しない可能性があり、エラーがない場合でもデータは失われます。
sessionhandler :: closeを呼び出した後、セッションデータを変更します
SessionHandler :: Closeを呼び出した後もセッションデータを変更し続けると、これらの変更は保存されません。セッションは、閉鎖方法を呼び出した後、終了としてマークされているため、セッションデータのさらなる操作は無視されます。
session_write_close()を正しく呼び出します
スクリプトの早い段階でセッションを終了する必要がある場合は、最後の操作が完了したらsession_write_close()に電話することができます。すべてのデータが保存される前に、この方法を呼び出すことは避けてください。通常、 session_write_close()はスクリプト実行の最後に配置され、セッションを閉じる前にすべての操作が完了するようにします。
カスタムセッションストレージを慎重に実装します
SessionHandlerをカスタマイズするときは、 Closeメソッドですべてのデータを書き込む操作が正しく実行されていることを確認してください。ファイルストレージの場合、 file_put_contentsまたはflockメソッドを使用して、ファイルの書き込みが完了していることを確認できます。データベースストレージの場合、セッションの終了前にコミット操作が完了することを確認する必要があります。
サンプルコード:
<span><span><span class="hljs-class"><span class="hljs-keyword">class</span></span></span><span> </span><span><span class="hljs-title">MySessionHandler</span></span><span> </span><span><span class="hljs-keyword">extends</span></span><span> </span><span><span class="hljs-title">SessionHandler</span></span><span> {
</span><span><span class="hljs-keyword">public</span></span><span> </span><span><span class="hljs-function"><span class="hljs-keyword">function</span></span></span><span> </span><span><span class="hljs-title">close</span></span><span>(</span><span><span class="hljs-params"></span></span><span>) {
</span><span><span class="hljs-comment">// 書き込み操作を確認してください</span></span><span>
</span><span><span class="hljs-keyword">if</span></span><span> (</span><span><span class="hljs-variable language_">$this</span></span><span>-></span><span><span class="hljs-title function_ invoke__">saveDataToStorage</span></span><span>()) {
</span><span><span class="hljs-built_in">parent</span></span><span>::</span><span><span class="hljs-title function_ invoke__">close</span></span><span>(); </span><span><span class="hljs-comment">// 親クラスを呼び出します close 方法</span></span><span>
}
}
</span><span><span class="hljs-keyword">private</span></span><span> </span><span><span class="hljs-function"><span class="hljs-keyword">function</span></span></span><span> </span><span><span class="hljs-title">saveDataToStorage</span></span><span>(</span><span><span class="hljs-params"></span></span><span>) {
</span><span><span class="hljs-comment">// カスタムストレージ操作</span></span><span>
</span><span><span class="hljs-keyword">return</span></span><span> </span><span><span class="hljs-literal">true</span></span><span>; </span><span><span class="hljs-comment">// 成功を想定します</span></span><span>
}
}
</span></span>
session_save_pathの構成を確認してください
session_save_pathが正しく構成されており、PHPプロセスがディレクトリに書き込む許可があることを確認してください。カスタムストレージメソッド(データベースストレージなど)を使用する場合は、関連する接続とアクセス許可が正しく構成されていることを確認してください。
セッションストレージパスは、次のコードで検証できます。
<span><span><span class="hljs-keyword">echo</span></span><span> </span><span><span class="hljs-title function_ invoke__">session_save_path</span></span><span>(); </span><span><span class="hljs-comment">// 電流を印刷します session ストレージパス</span></span><span>
</span></span>
終了後にセッションデータの変更を避けてください
sessionhandler :: closeまたはsession_write_close()が呼び出されたら、セッションデータを変更しないでください。セッションがすべてのタスクを処理した後、セッションを閉じるのが最善です。
例:
<span><span><span class="hljs-title function_ invoke__">session_start</span></span><span>();
</span><span><span class="hljs-variable">$_SESSION</span></span><span>[</span><span><span class="hljs-string">'username'</span></span><span>] = </span><span><span class="hljs-string">'example'</span></span><span>;
</span><span><span class="hljs-comment">// ここに電話しないでください session_write_close</span></span><span>
</span><span><span class="hljs-title function_ invoke__">session_write_close</span></span><span>(); </span><span><span class="hljs-comment">// すべての操作の後は必ず電話してください</span></span><span>
</span></span>