>PHP 프레임워크 >Laravel >laravel 모델 바인딩 실패의 이유와 해결 방법을 설명하는 기사

laravel 모델 바인딩 실패의 이유와 해결 방법을 설명하는 기사

PHPz
PHPz원래의
2023-04-08 23:30:021044검색

Laravel 모델 바인딩은 Laravel 프레임워크의 중요한 기능으로, 개발자가 수동으로 데이터베이스를 쿼리하는 지루한 프로세스를 방지하면서 컨트롤러 메서드에서 지정된 모델에 URL 매개변수를 자동으로 삽입하는 기능을 제공합니다. 그러나 실제로 개발자는 모델 바인딩 실패로 인해 프로그램 예외가 발생하는 경우가 있습니다. 이 글은 Laravel 모델 바인딩에 발생할 수 있는 문제와 해결책을 설명하기 위해 실제 사례부터 시작하겠습니다.

예:

제품 목록 페이지가 있다고 가정합니다. URL은 제품 카테고리를 나타내는 카테고리 매개변수를 허용합니다. 컨트롤러는 모델 바인딩을 사용하여 지정된 카테고리 아래의 제품을 쿼리하고 뷰를 반환합니다. 다음 코드를 통해 구현합니다.

public function index(Category $category)
{
    $products = $category->products;
    return view('products.index', ['products' => $products]);
}

보시다시피 index()의 <code>$category 매개변수를 통해 Category 모델을 바인딩합니다. code> 메소드를 사용하면 이 카테고리의 모든 제품을 가져옵니다. 그러나 브라우저 주소 표시줄에 존재하지 않는 카테고리를 입력하면 아래와 같이 프로그램에서 예외가 발생합니다. index() 方法的参数 $category 来绑定 Category 模型,然后获取该分类下的所有产品。然而,当我们在浏览器地址栏中输入一个不存在的分类时,程序就会抛出异常,如下所示:

Illuminate\Database\Eloquent\ModelNotFoundException

No query results for model [App\Category].

我们发现,程序抛出的异常信息是 ModelNotFoundException,也就是说 Laravel 框架无法找到对应的模型。那么,出现这种问题的原因是什么呢?

问题原因:

模型绑定能够正常工作的前提是,URL 参数必须与数据库中对应的模型匹配。如果 URL 参数无法匹配,则会出现模型绑定失败的情况。在上面的示例中,我们在地址栏中输入了一个不存在的分类 ID,这就导致了程序无法找到对应的模型。

另外,如果我们在控制器方法中手动注入了模型,例如:

public function show(Product $product)
{
    return view('products.show', ['product' => $product]);
}

则还需要注意以下问题:

  1. 参数名必须与路由中的占位符名称相同。
  2. 参数类型必须是一个 Eloquent 模型类。
  3. 如果无法找到对应的模型,则会抛出 ModelNotFoundException 异常。

解决方案:

那么,我们该如何解决模型绑定失败的问题呢?下面是两种解决方案。

  1. 定义全局范围的解析器

我们可以在 boot() 方法中定义全局范围的解析器,以捕获所有的 ModelNotFoundException 异常,然后将其转换为自定义的异常。以下是实现方式:

(1)定义自定义异常:

namespace App\Exceptions;

use Exception;

class NotFoundException extends Exception
{
    public function render($request)
    {
        return response()->view('errors.404');
    }
}

该异常类在抛出异常时,将返回 resources/views/errors/404.blade.php 视图。

(2)在 AppServiceProvider.php 中注册解析器:

use App\Exceptions\NotFoundException;
use Illuminate\Database\Eloquent\ModelNotFoundException;
use Illuminate\Support\ServiceProvider;

class AppServiceProvider extends ServiceProvider
{
    public function boot()
    {
        $this->app->bind(ModelNotFoundException::class, function ($e) {
            throw new NotFoundException;
        });
    }
}

boot() 方法中,我们绑定了 ModelNotFoundException 异常到自定义异常 NotFoundException,以便程序能够正确渲染 404 页面。

  1. 手动抛出异常

在控制器方法中,我们可以手动抛出自定义异常,例如:

public function show($id)
{
    $product = Product::findOrFail($id);

    if ($product->category->id !== $this->category->id) {
        throw new NotFoundException;
    }

    return view('products.show', ['product' => $product]);
}

在该方法中,我们手动查询产品模型,然后根据分类 ID 判断产品是否属于当前分类,如果不属于,则手动抛出自定义异常 NotFoundExceptionrrreee

프로그램에서 발생한 예외 정보가 ModelNotFoundException인 것을 확인했습니다. 이는 Laravel 프레임워크가 해당 모델을 찾을 수 없음을 의미합니다. 그렇다면 이 문제의 원인은 무엇입니까?

문제 원인:

모델 바인딩이 제대로 작동하려면 URL 매개변수가 데이터베이스의 해당 모델과 일치해야 합니다. URL 매개변수가 일치할 수 없으면 모델 바인딩이 실패합니다. 위의 예에서는 주소 표시줄에 존재하지 않는 카테고리 ID를 입력하여 프로그램이 해당 모델을 찾을 수 없게 했습니다. 🎜🎜또한 컨트롤러 메소드에 모델을 수동으로 주입하는 경우(예: 🎜rrreee🎜) 다음 문제에도 주의해야 합니다. 🎜
  1. 매개변수 이름은 매개변수 이름과 동일해야 합니다. 경로의 자리 표시자 이름입니다.
  2. 매개변수 유형은 Eloquent 모델 클래스여야 합니다.
  3. 해당 모델을 찾을 수 없으면 ModelNotFoundException 예외가 발생합니다.
🎜해결책:🎜🎜그렇다면 모델 바인딩 실패 문제를 어떻게 해결할 수 있을까요? 아래에는 두 가지 솔루션이 있습니다. 🎜
  1. 전역 범위 파서 정의
🎜 boot() 메서드에서 전역 범위 파서를 정의하여 모든 ModelNotFoundExceptionModelNotFoundException을 캡처할 수 있습니다. code> 예외를 선택한 다음 이를 사용자 정의 예외로 변환합니다. 구현 방법은 다음과 같습니다. 🎜🎜(1) 사용자 정의 예외 정의: 🎜rrreee🎜이 예외 클래스에서 예외가 발생하면 resources/views/errors/404.blade.php를 반환합니다. 코드>보기. 🎜🎜(2) <code>AppServiceProvider.php에 파서를 등록합니다. 🎜rrreee🎜 boot() 메서드에서 ModelNotFoundException 예외를 프로그램이 404 페이지를 올바르게 렌더링할 수 있도록 사용자 정의 예외 NotFoundException 🎜
  1. 수동으로 예외 발생
🎜컨트롤러 메소드에서 사용자 정의 예외를 수동으로 발생시킬 수 있습니다. 예: 🎜rrreee🎜이 메소드에서는 수동으로 쿼리합니다. 그런 다음 카테고리 ID를 기반으로 제품이 현재 카테고리에 속하는지 여부를 확인합니다. 그렇지 않은 경우 사용자 지정 예외 NotFoundException가 수동으로 발생합니다. 🎜🎜결론:🎜🎜Laravel 모델 바인딩은 Laravel 프레임워크의 중요한 기능으로, 데이터베이스 쿼리와 컨트롤러 코드를 크게 단순화할 수 있습니다. 그러나 실제로는 존재하지 않는 모델을 수동으로 주입하는 것을 방지하고, 예외를 처리하기 위해 전역 파서를 정의하는 등 일부 세부 사항에도 주의를 기울여야 합니다. 이 글을 통해 모든 분들이 Laravel 모델 바인딩을 더 잘 이해하고 불필요한 문제를 피할 수 있기를 바랍니다. 🎜

위 내용은 laravel 모델 바인딩 실패의 이유와 해결 방법을 설명하는 기사의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.