27/07/2019, 16:52

Tập 7: Service provider Laravel

Rất vui được gặp lại các bạn và cảm ơn đã đồng hành cùng với mình trong các tập trước. Và ngày hôm nay, mình sẽ nói về "Service provider", đây cũng là một trong những khái niệm cấu trúc của Laravel. Nếu bạn nào đã hiểu rõ về "Service container" ở tập trước thì mình tin chắc rằng ...

Rất vui được gặp lại các bạn và cảm ơn đã đồng hành cùng với mình trong các tập trước. Và ngày hôm nay, mình sẽ nói về "Service provider", đây cũng là một trong những khái niệm cấu trúc của Laravel. Nếu bạn nào đã hiểu rõ về "Service container" ở tập trước thì mình tin chắc rằng nội dung ngày hôm nay sẽ không làm khó được bạn. Trong tập này bạn sẽ học được cách làm thể nào để viết cũng như đăng ký một service provider cho ứng dụng.

Service provider là trung tâm của tất cả các bootstrapping trong ứng dụng Laravel. Các service của riêng bạn hay tất cả các service cốt lõi của Laravel đều được bootstrap thông qua nó. Nói "bootstrap" mấy bữa giờ nhưng chúng ta vẫn chưa biết nó thực thi gì cả. Trong Laravel, "bootstrap" có nghĩa là đăng ký mọi thứ, bao gồm đăng ký các service container binding, envent listener, middleware và route... Service provider còn được biết là trung tâm cấu hình cho ứng dụng của bạn.

Nếu bạn mở file config/app.php thì bạn sẽ thấy một mảng providers, tại đây chứa tất cả các lớp service provider sẽ được load cho ứng dụng. Chú ý rằng một trong số đó là các "deferred" provider (provider "hoãn lại"), điều này có nghĩa các provider này sẽ không được load trong mỗi request mà chỉ khi nào cần đến chúng.

Tất cả các service provider đều kế thừa class IlluminationSupportServiceProvider và chứa hai method register và boot. Hai method này chính là hai chốt chặn "Register service providers" và "Bootstrap service providers" mà mình đã nói ở trong tập 5.

Để tạo một provider mới với là TestServiceProvider chẳng hạn, bạn chỉ cần việc chạy lệnh Artisan:

php artisan make:provider TestServiceProvider

1. Phương thức "register" (The "register" method)

Trong method register, chúng ta chỉ nên thực hiện công việc là bind đến container. Các bạn không nên cố đăng ký bất kỳ một event listener, route hoặc bất kỳ chức năng nào khác. Bởi vì trong quá trình thực thi một chức năng nào đó ngoài việc binding thì bạn có thể vô tình sử dụng một service provider nào đó mà chưa được load.

Trong bất cứ method của service provider, bạn luôn có quyền truy cập đến $app.

public function register()
{
    $this->app->singleton('AppUser', function() {
        //
    });
}

Về binding service container thì mình sẽ không đề cập đến nữa vì đã nói rất chi tiết ở tập trước rồi.

2. Thuộc tính "bindings" và "singletons" (The "bindings" and "singletons" properties)

Nếu trong method register có quá nhiều simple binding, bạn có thể sử dụng thuộc tính bindings và singletons để thay thế cho việc bind hay singleton thủ công.

Ví dụ như bạn có đoạn binding service container như sau:

public function register()
{
    $this->app->bind(User:class, function() {
        return new AppUser;
    });

    $this->app->singleton(Post:class, function() {
        return new AppPost;
    });
}

Thì bạn có thể làm như sau:

// ...

use IlluminateSupportServiceProvider;

class TestServiceProvider extends ServiceProvider
{
    public $bindings = [
        User::class => AppUser::class,
        //
    ];
    
    public $singletons = [
        Post::class => AppPost::class,
        //
    ];
    
    public function register()
    {
        //
    }
    
// ...

3. Phương thức "boot" (The "boot" method)

Trong method boot này, bạn đã có thể gọi tất cả các service provider (ngay cả ở những file provider khác) đã đăng ký trước đó.

public function boot()
{
    $this->app->make('AppUser');
}

Ngoài ra ở method boot, bạn có thể inject dependency, container sẽ tự động inject bất kỳ class nào mà bạn khai báo.

use AppUser;

public function boot(User $user)
{
    //
}

Tất cả các provider đều được đăng ký ở file config/app.php, cụ thể hơn là tại mảng providers. Các provider cốt lõi của Laravel mặc định đã được liệt kê ở đây sẵn. Để đăng ký provider do bạn tạo ra, bạn có thể liệt kê nó như thế này:

'providers' => [
    // ...

    // Other Service Providers
    AppProvidersTestServiceContainer::class,
],

Như đã nói ở trên, deferred provider sẽ không được load sau mỗi request mà chỉ khi nào có yêu cầu resolve thì mới load chúng.

Để defer một provider, bạn chỉ cần implement class IlluminateContractsSupportDeferrableProvider cho provider và khởi tạo method providers. Method này trả về một mảng chứa các class cần defer.

// ...
use IlluminateSupportServiceProvider;
use IlluminateContractsSupportDeferrableProvider;

class TestServiceProvider extends ServiceProvider implements DeferrableProvider
{
    public function register()
    {
        $this->app->singleton('User', function() {
            return new AppUser;
        });
    }
    
    public function providers()
    {
        return [
            User::class,
            //
        ];      
    }
    
// ...

Cảm ơn các bạn đã quan tâm theo dõi. Cùng đồng hành với mình qua những tập tiếp theo tại series "Hành trình chinh phục Laravel Framework" nhé! Chúc may mắn và hẹn gặp lại.

0