Home >PHP Framework >Laravel >An article explaining the reasons and solutions for laravel model binding failure

An article explaining the reasons and solutions for laravel model binding failure

PHPz
PHPzOriginal
2023-04-08 23:30:021028browse

Laravel model binding is an important feature of the Laravel framework. It provides a function to automatically inject URL parameters into the specified model in the controller method, avoiding the tedious process of developers manually querying the database. However, in practice, developers sometimes encounter model binding failures, causing program exceptions. This article will start with actual cases to explain possible problems and solutions to Laravel model binding.

Example:

Suppose we have a product list page. The URL accepts a category parameter to represent the product category. The controller uses model binding to query the products under the specified category and returns the view. We implement it through the following code:

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

As you can see, we bind the Category model through the parameter $category of the index() method , and then get all products under this category. However, when we enter a non-existent category in the browser address bar, the program will throw an exception, as shown below:

Illuminate\Database\Eloquent\ModelNotFoundException

No query results for model [App\Category].

We found that the exception information thrown by the program is ModelNotFoundException , which means that the Laravel framework cannot find the corresponding model. So, what is the reason for this problem?

Cause of the problem:

The premise for model binding to work properly is that the URL parameters must match the corresponding model in the database. If the URL parameters cannot be matched, model binding will fail. In the above example, we entered a non-existent category ID in the address bar, which caused the program to be unable to find the corresponding model.

In addition, if we manually inject the model in the controller method, for example:

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

, we also need to pay attention to the following issues:

  1. The parameter name must be the same as the route The placeholder names in are the same.
  2. The parameter type must be an Eloquent model class.
  3. If the corresponding model cannot be found, a ModelNotFoundException exception will be thrown.

Solution:

So, how do we solve the problem of model binding failure? Below are two solutions.

  1. Define a global scope parser

We can define a global scope parser in the boot() method to capture all ModelNotFoundException exception and then convert it into a custom exception. The following is the implementation method:

(1) Define a custom exception:

namespace App\Exceptions;

use Exception;

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

When this exception class throws an exception, it will return resources/views/errors/404.blade. php view.

(2) Register the parser in 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;
        });
    }
}

In the boot() method, we bind ModelNotFoundException Exception to custom exception NotFoundException so that the program can render the 404 page correctly.

  1. Manually throw exceptions

In the controller method, we can manually throw a custom exception, for example:

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

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

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

In this method , we manually query the product model, and then determine whether the product belongs to the current category based on the category ID. If it does not, we manually throw a custom exception NotFoundException.

Conclusion:

Laravel model binding is an important feature in the Laravel framework, which can greatly simplify database queries and controller code. However, in practice, we also need to pay attention to some details, such as avoiding manually injecting non-existent models, defining global parsers to handle exceptions, etc. I hope that through this article, everyone can better understand Laravel model binding and avoid unnecessary problems.

The above is the detailed content of An article explaining the reasons and solutions for laravel model binding failure. For more information, please follow other related articles on the PHP Chinese website!

Statement:
The content of this article is voluntarily contributed by netizens, and the copyright belongs to the original author. This site does not assume corresponding legal responsibility. If you find any content suspected of plagiarism or infringement, please contact admin@php.cn