PHPでの正しいエラー処理

適切な処理が意味すること:




ワンストップソリューション


これを行うために、exceptionHandlerClassクラスを作成します。 exceptionHandlerClassは、設定と静的メソッド(エラー、例外、およびアサーションハンドラー)を格納します。 setupHandlersメソッドとrestoreHandlersメソッドも必要です。 最初の方法は、エラートラップを構成します。 エラーおよびアサーションハンドラーはErrorExceptionをスローします。 例外ハンドラーは未処理の例外を処理し、設定に応じて適切な応答を出力します。 restoreHandlersはすべてのハンドラーを元の状態に戻します-これは、既存のエラー処理メカニズムを備えたコードにクラスを埋め込むときに役立ちます。 接続は次のようになります。
  1. require 'exceptionHandler / exceptionHandlerClass.php' ;
  2. exceptionHandlerClass :: setupHandlers ;

デバッグモードを有効にする(デフォルトではオフ):
  1. exceptionHandlerClass :: $ debug = true ;

出力形式


例で説明する方が簡単です。Webページにトレースを出力するには、preタグでラップしてhtmlspecialchars()を使用します。一方、コンソールに出力するときに同じトレースを読むのは不便です。 エラーをSapServerの応答として出力する必要がある場合は、適切に形成されたXMLドキュメント(SoapFault)である必要があります。 スクリプトが画像などのバイナリデータを表示する場合、WildFireを介してエラーを表示する方が便利です。 これらのすべての状況で、異なる出力形式を適用するだけです。
さまざまな形式に対して、さまざまなクラスを作成します。 まずは、exceptionHandlerOutputWeb(Web用)とexceptionHandlerOutputCli(コマンドライン用)の2つの出力形式を実装します。 また、ファクトリクラス(exceptionHandlerOutputFactory)も必要です。このクラスでは、ロジックがカプセル化され、どの出力形式をいつ使用するかが決まります。
  1. パブリック 関数 getExceptionHandlerOutput {
  2. if php_sapi_name == 'cli' {
  3. 新しい exceptionHandlerOutputCli )を 返し ます。
  4. }
  5. 新しい exceptionHandlerOutputWeb )を 返し ます。
  6. }

setupHandlersを呼び出すとき、exceptionHandlerOutput *またはexceptionHandlerOutputFactory *クラスのインスタンスを渡すことにより、出力形式を設定できます。
  1. exceptionHandlerClass :: setupHandlers 新しい exceptionHandlerOutputAjax ;

このアーキテクチャのおかげで、フォーマットを簡単に拡張できます。 新しい形式を作成するには、抽象クラスexceptionHandlerOutputから継承し、1つのメソッド(出力)を実装するクラスを作成するだけで十分です。
  1. class exceptionHandlerOutputAjax exceptionHandlerOutput {
  2. public function output $ exception $ debug {
  3. header 'HTTP / 1.0 500 Internal Server Error' true 500 ;
  4. header 'Status:500 Internal Server Error' true 500 ;
  5. $応答 = 配列
  6. 'エラー' => true
  7. 'メッセージ' => ''
  8. ;
  9. if $ debug {
  10. $ response [ 'message' ] = $ exception- > getMessage ;
  11. } else {
  12. $ response [ 'message' ] = self :: $ productionMessage ;
  13. }
  14. exit json_encode $ response ;
  15. }
  16. }

出力形式を自動的に選択するより複雑なロジックが必要な場合は、exceptionHandlerOutputFactoryを継承するクラスを作成し、getExceptionHandlerOutputメソッドを実装する必要があります。
  1. class exceptionHandlerOutputAjaxFactory exceptionHandlerOutputDefaultFactoryを拡張します{
  2. パブリック 関数 getExceptionHandlerOutput {
  3. if self :: detect {
  4. 新しい exceptionHandlerOutputAjax )を 返し ます。
  5. }
  6. :: getExceptionHandlerOutput ;
  7. }
  8. public static function detect {
  9. return empty $ _SERVER [ 'HTTP_X_REQUESTED_WITH' ]
  10. && strtolower $ _SERVER [ 'HTTP_X_REQUESTED_WITH' ] == 'xmlhttprequest' ;
  11. }
  12. }
  13. exceptionHandlerClass :: setupHandlers 新しい exceptionHandlerOutputAjaxFactory ;

ロギング


上で述べたように、ログは必要に応じて有効にできます。 これを行うために、exceptionHandlerメソッドでexceptionLogメソッドが作成されました
  1. public static function exceptionLog $ exception $ logPriority = null {
  2. if is_null self :: $ exceptionHandlerLog {
  3. self :: $ exceptionHandlerLog- > log $ exception $ logPriority ;
  4. }
  5. }

ロギングを有効にする必要がある場合は、次を実行します。
  1. exceptionHandlerClass :: $ exceptionHandlerLog = new exceptionHandlerSimpleLog ;

ロギングクラスは、抽象exceptionHandlerLogを継承し、logメソッドを実装する必要があります
  1. class exceptionHandlerSimpleLog exceptionHandlerLogを拡張します{
  2. public function log $ exception $ logType {
  3. switch $ logType {
  4. ケース self :: uncaughtException
  5. error_log $ exception- > getMessage ;
  6. 休憩 ;
  7. }
  8. }
  9. }

logTypeは、exceptionHandlerLogで宣言された定数の1つです。
  1. const uncaughtException = 0 ; //未処理の例外
  2. const caughtException = 1 ; //エラーハンドラの外部でロギングメソッドを呼び出します
  3. const ignoreError = 2 ; // screamオプションが無効な場合、@でマスクされたエラーがログに記録されます
  4. const lowPriorityError = 3 ; //例外にならないエラー
  5. const assertion = 4 ; //アサーション

logTypeと例外を使用して、開発者はどの例外とどのようにログを記録するかを自分で決定できます。 たとえば、uncaughtExceptionをメールで送信したり、重大度がE_ERRORのignoreErrorをファイルに記録したりできます。

トレース


トレースの出力では、例外のタイプ、メッセージ、および実際のトレース自体を確認します。 各呼び出しのトレースでは、どの関数が呼び出されたか、「短い」パラメーターのリスト、ファイルと呼び出しが発生した行を表示する必要があります。 例として「短い」パラメーターが何であるかを説明します:関数が1000文字の文字列で呼び出された場合、トレースにこの文字列が存在しても問題の解決には役立ちません。読み取りトレースが複雑になるだけです。同じことが大きなネストを持つ配列に適​​用されます。 画面へのトレース出力は、単にどこを見るべきかを伝えるだけです。 何が起こっているのかを正確に把握するには、xdebugまたはプリミティブvar_dump()およびdie()を使用して、必要に応じてデバッグする必要があります。
トレースの例:
	 [ErrorException]:E_WARNING-mysql_connect(): 'localhost'上のMySQLサーバーに接続できません(10061)
	 #0:mysql_connect()
	     D:\ projects1 \ d \ index.php:19
	 #1:testClass :: test1( "長い文字列... eeeeeery長い文字列"(56))
	     D:\ projects1 \ d \ index.php:22
	 #2:testClass-> test2(testClass()、-∞、i:iTest、c:testClass、fa:testClass :: test2)
	     D:\ projects1 \ d \ index.php:27
	 #3:testAll(r:ストリーム、fs:testClass :: test1)
	     D:\ projects1 \ d \ index.php:30
凡例

そして、最も便利な...トレースから直接IDEでファイルを開くためのリンク。

IDEのリンクをクリックすると、対応する行の対応するファイルが開きます。
コンソールモード(NetBeansコンソール)の場合:
NetBeansコンソール
  1. exceptionHandlerOutputCli :: setFileLinkFormat ':in%f on line%l' ;
Webモード(TextMate)の場合:
  1. exceptionHandlerOutputWeb :: setFileLinkFormat 'txmt:// open /?file://%f&line =%l' ;

NetbBeans(または別のIDE)に実装できます。 このために必要なこと:プロトコルを登録します。 このプロトコルのハンドラーを作成します(最も単純なのはbatファイルです)。 ハンドラーで、コマンドラインを使用して、対応するファイルと行でNetBeansを呼び出します。 しかし、これは次の記事のトピックです。
コードは2日で作成されたため、小さな欠陥が発生する可能性があります。 ダウンロード (リポジトリに置く時間がありませんでした)。

UPD: PHPブログに移植
UPD2: PHPで例外を処理するトピックの継続


Source: https://habr.com/ru/post/J99431/


All Articles