検索
ホームページPHPフレームワークLaravelLaravel コア分析の例外処理 (コード)

この記事の内容は、Laravel コア解析の例外処理 (コード) に関するものです。必要な方は参考にしていただければ幸いです。

例外処理は、プログラムの実行時エラーを処理するメカニズムを開発者に提供する非常に重要ですが、プログラム自体の詳細は次のとおりです。ユーザーに提供され、開発者に完全なエラー トレースバック スタックを提供すると同時に、プログラムの堅牢性も向上します。

この記事では、Laravel で提供される例外処理機能を簡単に確認し、その後、開発での例外処理の使用方法、カスタム例外の使用方法、Laravel の例外処理機能を拡張する方法について説明します。

例外ハンドラーの登録

ここでは、ブートストラップ ステージで何度も述べたように、カーネルがリクエストを処理する前にブートストラップ ステージに戻る必要があります。 Foundation\Bootstrap\ HandleExceptions セクションで、Laravel はシステム例外処理動作を設定し、グローバル例外ハンドラーを登録します。

class HandleExceptions
{
    public function bootstrap(Application $app)
    {
        $this->app = $app;

        error_reporting(-1);

        set_error_handler([$this, 'handleError']);

        set_exception_handler([$this, 'handleException']);

        register_shutdown_function([$this, 'handleShutdown']);

        if (! $app->environment('testing')) {
            ini_set('display_errors', 'Off');
        }
    }
    
    
    public function handleError($level, $message, $file = '', $line = 0, $context = [])
    {
        if (error_reporting() & $level) {
            throw new ErrorException($message, 0, $level, $file, $line);
        }
    }
}

set_Exception_handler([$this, 'handleException']) は、HandleExceptions の handleException メソッドをグローバルとして登録します。プログラムのハンドラー メソッド:

public function handleException($e)
{
    if (! $e instanceof Exception) {
        $e = new FatalThrowableError($e);
    }

    $this->getExceptionHandler()->report($e);

    if ($this->app->runningInConsole()) {
        $this->renderForConsole($e);
    } else {
        $this->renderHttpResponse($e);
    }
}

protected function getExceptionHandler()
{
    return $this->app->make(ExceptionHandler::class);
}

// 渲染CLI请求的异常响应
protected function renderForConsole(Exception $e)
{
    $this->getExceptionHandler()->renderForConsole(new ConsoleOutput, $e);
}

// 渲染HTTP请求的异常响应
protected function renderHttpResponse(Exception $e)
{
    $this->getExceptionHandler()->render($this->app['request'], $e)->send();
}

プロセッサでは、例外は主に ExceptionHandler のレポート メソッドを通じて報告されます。ここでは、例外は storage/laravel.log ファイルに記録され、それに応じて例外応答がレンダリングされます。リクエスト タイプに変更して、クライアントへの出力を生成します。ここでの ExceptionHandler は \App\Exceptions\Handler クラスのインスタンスであり、プロジェクトの先頭でサービス コンテナに登録されます。

// bootstrap/app.php

/*
|--------------------------------------------------------------------------
| Create The Application
|--------------------------------------------------------------------------
*/

$app = new Illuminate\Foundation\Application(
    realpath(__DIR__.'/../')
);

/*
|--------------------------------------------------------------------------
| Bind Important Interfaces
|--------------------------------------------------------------------------
*/
......

$app->singleton(
    Illuminate\Contracts\Debug\ExceptionHandler::class,
    App\Exceptions\Handler::class
);

ちなみに、ここでは set_error_handler 関数を使用して登録します。エラー ハンドラー。一部の古いコードやクラス ライブラリでは、例外ハンドラーは例外のみを処理でき、エラーは処理できないため、通常は set_error_handler が使用されます。グローバル エラー ハンドラー メソッドを登録し、メソッド内でエラーをキャッチした後、エラーを例外に変換して再スローすることで、プロジェクト内のすべてのコードが正しく実行されなかった場合に例外インスタンスをスローできるようになります。

/**
 * Convert PHP errors to ErrorException instances.
 *
 * @param  int  $level
 * @param  string  $message
 * @param  string  $file
 * @param  int  $line
 * @param  array  $context
 * @return void
 *
 * @throws \ErrorException
 */
public function handleError($level, $message, $file = '', $line = 0, $context = [])
{
    if (error_reporting() & $level) {
        throw new ErrorException($message, 0, $level, $file, $line);
    }
}

一般的に使用される Laravel 例外の例

Laravel は、一般的なプログラム例外に対応する例外インスタンスをスローします。これにより、開発者はこれらのランタイム例外をキャッチし、次に従ってフォローアップ処理を行うことができます。あなた自身のニーズ (例: catch で別の修復メソッドを呼び出す、例外をログ ファイルに記録する、アラーム メールやテキスト メッセージを送信する)

ここでは、開発中によく発生するいくつかの例外をリストし、その内容について説明します。通常のコーディングでは、プログラム内でこれらの例外をキャッチすることに注意し、それらを適切に処理してプログラムをより堅牢にする必要があります。

Illuminate\Database\QueryException この例外は、Laravel で SQL ステートメントを実行するときにエラーが発生したときにスローされます。これは最も一般的に使用される例外でもあり、たとえば、多くの人が SQL 実行エラーをキャプチャするために使用されます。 Update ステートメントの実行時に SQL を判断するのと同様に、実行後に変更された行の数が判断され、UPDATE が成功したかどうかが判断されます。ただし、一部のシナリオでは、実行された UPDATE ステートメントはレコード値を変更しません。また、トランザクションの実行中に QueryException がキャッチされた場合、トランザクションはキャッチ コード ブロックでロールバックされる可能性があります。

Illuminate\Database\Eloquent\ModelNotFoundException この例外は、モデルの findOrFail メソッドと firstOrFail メソッドで 1 つのレコードが見つからない場合にスローされます (データが見つからない場合、find と first は NULL を返します)。

Illuminate\Validation\ValidationException この例外は、リクエストが Laravel の FormValidator 検証に合格しない場合にスローされます。

Illuminate\Auth\Access\AuthorizationException この例外は、ユーザーリクエストが Laravel のポリシー (ポリシー) 検証に合格しない場合にスローされます。

Symfony\Component\Routing\Exception\MethodNotAllowedException ルーティングをリクエストするとき、 HTTP メソッドが正しくありません

Illuminate\Http\Exceptions\HttpResponseException Laravel は、HTTP リクエストの処理が失敗したときにこの例外をスローします

Laravel の例外ハンドラーを拡張します

# 前述したように、Laravel は \App\Exceptions\Handler をグローバル例外ハンドラーとして正常に登録しました。コード内でキャッチされなかった例外は、最終的には \App\Exceptions\Handler によってキャッチされます。次に、例外応答をレンダリングし、クライアントに応答を送信します。ただし、組み込みの例外ハンドラー メソッドは使いにくいことがよくあります。次の例では、Sentry システムに例外を報告します。これは非常に優れたエラー収集サービスです。使用:

public function report(Exception $exception)
{
    if (app()->bound('sentry') && $this->shouldReport($exception)) {
        app('sentry')->captureException($exception);
    }

    parent::report($exception);
}
とデフォルトのレンダリング メソッド。フォーム検証中に生成される応答の JSON 形式は、プロジェクトの統一された

JOSN 形式とは異なることが多いため、レンダリングをカスタマイズする必要があります。方法。

public function render($request, Exception $exception)
{
    //如果客户端预期的是JSON响应,  在API请求未通过Validator验证抛出ValidationException后
    //这里来定制返回给客户端的响应.
    if ($exception instanceof ValidationException && $request->expectsJson()) {
        return $this->error(422, $exception->errors());
    }

    if ($exception instanceof ModelNotFoundException && $request->expectsJson()) {
        //捕获路由模型绑定在数据库中找不到模型后抛出的NotFoundHttpException
        return $this->error(424, 'resource not found.');
    }


    if ($exception instanceof AuthorizationException) {
        //捕获不符合权限时抛出的 AuthorizationException
        return $this->error(403, "Permission does not exist.");
    }

    return parent::render($request, $exception);
}

自定义后,在请求未通过FormValidator验证时会抛出ValidationException, 之后异常处理器捕获到异常后会把错误提示格式化为项目统一的JSON响应格式并输出给客户端。这样在我们的控制器中就完全省略了判断表单验证是否通过如果不通过再输出错误响应给客户端的逻辑了,将这部分逻辑交给了统一的异常处理器来执行能让控制器方法瘦身不少。

使用自定义异常

这部分内容其实不是针对Laravel框架自定义异常,在任何项目中都可以应用我这里说的自定义异常。

我见过很多人在Repository或者Service类的方法中会根据不同错误返回不同的数组,里面包含着响应的错误码和错误信息,这么做当然是可以满足开发需求的,但是并不能记录发生异常时的应用的运行时上下文,发生错误时没办法记录到上下文信息就非常不利于开发者进行问题定位。

下面的是一个自定义的异常类

namespace App\Exceptions\;

use RuntimeException;
use Throwable;

class UserManageException extends RuntimeException
{
    /**
     * The primitive arguments that triggered this exception
     *
     * @var array
     */
    public $primitives;
    /**
     * QueueManageException constructor.
     * @param array $primitives
     * @param string $message
     * @param int $code
     * @param Throwable|null $previous
     */
    public function __construct(array $primitives, $message = "", $code = 0, Throwable $previous = null)
    {
        parent::__construct($message, $code, $previous);
        $this->primitives = $primitives;
    }

    /**
     * get the primitive arguments that triggered this exception
     */
    public function getPrimitives()
    {
        return $this->primitives;
    }
}

定义完异常类我们就能在代码逻辑中抛出异常实例了

class UserRepository
{
  
    public function updateUserFavorites(User $user, $favoriteData)
    {
        ......
        if (!$executionOne) {
            throw new UserManageException(func_get_args(), 'Update user favorites error', '501');
        }
        
        ......
        if (!$executionTwo) {
            throw new UserManageException(func_get_args(), 'Another Error', '502');
        }
        
        return true;
    }
}

class UserController extends ...
{
    public function updateFavorites(User $user, Request $request)
    {
        .......
        $favoriteData = $request->input('favorites');
        try {
            $this->userRepo->updateUserFavorites($user, $favoritesData);
        } catch (UserManageException $ex) {
            .......
        }
    }
}

除了上面Repository列出的情况更多的时候我们是在捕获到上面列举的通用异常后在catch代码块中抛出与业务相关的更细化的异常实例方便开发者定位问题,我们将上面的updateUserFavorites 按照这种策略修改一下

public function updateUserFavorites(User $user, $favoriteData)
{
    try {
        // database execution
        
        // database execution
    } catch (QueryException $queryException) {
        throw new UserManageException(func_get_args(), 'Error Message', '501' , $queryException);
    }

    return true;
}

在上面定义UserMangeException类的时候第四个参数$previous是一个实现了Throwable接口类实例,在这种情景下我们因为捕获到了QueryException的异常实例而抛出了UserManagerException的实例,然后通过这个参数将QueryException实例传递给PHP异常的堆栈,这提供给我们回溯整个异常的能力来获取更多上下文信息,而不是仅仅只是当前抛出的异常实例的上下文信息, 在错误收集系统可以使用类似下面的代码来获取所有异常的信息。

while($e instanceof \Exception) {
    echo $e->getMessage();
    $e = $e->getPrevious();
}

异常处理是PHP非常重要但又容易让开发者忽略的功能,这篇文章简单解释了Laravel内部异常处理的机制以及扩展Laravel异常处理的方式方法。更多的篇幅着重分享了一些异常处理的编程实践,这些正是我希望每个读者都能看明白并实践下去的一些编程习惯,包括之前分享的Interface的应用也是一样。

以上がLaravel コア分析の例外処理 (コード)の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

声明
この記事はsegmentfaultで複製されています。侵害がある場合は、admin@php.cn までご連絡ください。
LARAVELバージョン:移行チュートリアルLARAVELバージョン:移行チュートリアルMay 14, 2025 am 12:17 AM

Laravelの移行システムは、最新バージョンでどのような新機能とベストプラクティスを提供していますか? 1。多型関係にnullablemorphs()を追加しました。 2。列()メソッドが導入され、列の順序が指定されます。 3.孤立した記録を避けるために、外国のキー制約の処理を強調します。 4.インデックスを適切に追加するなど、パフォーマンスを最適化することをお勧めします。 5.移行の実装と記述名の使用を提唱します。

Laravelの最新のLTSバージョンは何ですか?Laravelの最新のLTSバージョンは何ですか?May 14, 2025 am 12:14 AM

laravel10、relietinginfebruary2023、isthelatestltsversion、supportedforthreeyears.itrequiresphp8.1、拡張、forfeatureflagsを拡張し、エラーハンドリング、洗練されたもの、および実質的な形成、特にineLoquentormを改善します。

更新を維持:最新のLaravelバージョンの最新機能更新を維持:最新のLaravelバージョンの最新機能May 14, 2025 am 12:10 AM

Laravelの最新バージョンでは、複数の新機能を紹介します。1。Laravelpennantは、機能フラグを管理するために使用され、新しい機能を段階的にリリースできるようにします。 2。LARAVELREVERBは、リアルタイムコメントなどのリアルタイム機能の実装を簡素化します。 3. Laravelviteは、フロントエンドの建設プロセスを加速します。 4.新しいモデル工場システムは、テストデータの作成を強化します。 5.エラー処理メカニズムを改善し、より柔軟なエラーページのカスタマイズオプションを提供します。

Laravelでソフト削除の実装:ステップバイステップチュートリアルLaravelでソフト削除の実装:ステップバイステップチュートリアルMay 14, 2025 am 12:02 AM

softleteinelelavelisling -memptry -bracechortsdevetus -teedeecetovedlydeveledteecetetecedelave

現在のLaravelバージョン:最新リリースと更新を確認してください現在のLaravelバージョン:最新リリースと更新を確認してくださいMay 14, 2025 am 12:01 AM

laravel10.xisthecurrentversion、newfeatureslikeNumsuportineloquentModelsEndimprovedeModelbindingwithenums.theseupdatesenhanceCodereadabilityandsecurity、butrequirecarefulplanningandinningandincrementarementalementalementation forasucesupgrade。

Laravelの移行の使用方法:ステップバイステップのチュートリアルLaravelの移行の使用方法:ステップバイステップのチュートリアルMay 13, 2025 am 12:15 AM

laravelMigrationSstreamLedinedAtabaseManagementionbyAllowingsCheMachAngESTOBEDEDINPHPCODE

最新のLaravelバージョンを見つける:迅速で簡単なガイド最新のLaravelバージョンを見つける:迅速で簡単なガイドMay 13, 2025 am 12:13 AM

Laravelの最新バージョンを見つけるには、公式Webサイトlaravel.comにアクセスして、右上隅の「ドキュメント」ボタンをクリックするか、Composersコマンド「Composershowlaravel/Framework | Grepversions」を使用できます。更新され続けると、プロジェクトのセキュリティとパフォーマンスの向上に役立ちますが、既存のプロジェクトへの影響を考慮する必要があります。

Laravelで更新されたまま:最新バージョンを使用することの利点Laravelで更新されたまま:最新バージョンを使用することの利点May 13, 2025 am 12:08 AM

Youは、loredupdateTotheTothESTLARAVERVERSIONFORPERFORMANCEIMPROVEMENTS、強化セキュリティ、NewFeatures、BetterCommunitySupport、およびLong-Termmantenance.1)パフォーマンス:laravel9'seloquentormizationsenhanceapplicationspeed.2)laravel8introducedbetter

See all articles

ホットAIツール

Undresser.AI Undress

Undresser.AI Undress

リアルなヌード写真を作成する AI 搭載アプリ

AI Clothes Remover

AI Clothes Remover

写真から衣服を削除するオンライン AI ツール。

Undress AI Tool

Undress AI Tool

脱衣画像を無料で

Clothoff.io

Clothoff.io

AI衣類リムーバー

Video Face Swap

Video Face Swap

完全無料の AI 顔交換ツールを使用して、あらゆるビデオの顔を簡単に交換できます。

ホットツール

EditPlus 中国語クラック版

EditPlus 中国語クラック版

サイズが小さく、構文の強調表示、コード プロンプト機能はサポートされていません

SublimeText3 英語版

SublimeText3 英語版

推奨: Win バージョン、コードプロンプトをサポート!

MantisBT

MantisBT

Mantis は、製品の欠陥追跡を支援するために設計された、導入が簡単な Web ベースの欠陥追跡ツールです。 PHP、MySQL、Web サーバーが必要です。デモおよびホスティング サービスをチェックしてください。

SublimeText3 Linux 新バージョン

SublimeText3 Linux 新バージョン

SublimeText3 Linux 最新バージョン

SecLists

SecLists

SecLists は、セキュリティ テスターの究極の相棒です。これは、セキュリティ評価中に頻繁に使用されるさまざまな種類のリストを 1 か所にまとめたものです。 SecLists は、セキュリティ テスターが必要とする可能性のあるすべてのリストを便利に提供することで、セキュリティ テストをより効率的かつ生産的にするのに役立ちます。リストの種類には、ユーザー名、パスワード、URL、ファジング ペイロード、機密データ パターン、Web シェルなどが含まれます。テスターはこのリポジトリを新しいテスト マシンにプルするだけで、必要なあらゆる種類のリストにアクセスできるようになります。