PHP thuần: Tạo Facade giống laravel thật đơn giản

Đối với những ai đã từng sử dụng framework laravel, ắt hẳn các bạn đã biết tới Facades. Mặc dù có nhiều ý kiến cho rằng nó là một Anti-pattern. Facades đóng vai trò là “proxy tĩnh” đối với các lớp cơ bản trong vùng chứa dịch vụ, mang lại lợi ích của cú pháp

Đối với những ai đã từng sử dụng framework laravel, ắt hẳn các bạn đã biết tới Facades.
Mặc dù có nhiều ý kiến cho rằng nó là một Anti-pattern.

Facades đóng vai trò là “proxy tĩnh” đối với các lớp cơ bản trong vùng chứa dịch vụ, mang lại lợi ích của cú pháp ngắn gọn, biểu cảm trong khi vẫn duy trì khả năng kiểm tra và tính linh hoạt hơn các phương thức tĩnh truyền thống.

Tuy nhiên, không phải lúc nào các dự án của chúng ta đều được phép sử dụng framework.
Vì khi sử dụng framework, một khi có lỗi xảy ra thì chúng ta rất khó để control.
Trong bài viết này, mình xin chia sẻ cách tạo một Facades giống laravel trong PHP thuần.

Đặt vấn đề

Giả sử các bạn có một class như sau:

<?phpnamespaceCoreHttp;classResponse{/**
     * response json.
     *
     * @param  mixed  $content
     * @param  int    $options
     * @return string
     */publicfunctionjson($content,int$options=0){header('Content-type: application/json');echojson_encode($content,$options);}}

Khi đó nếu bạn muốn gọi hàm json, bạn phải khởi tạo instance của class Response, sau đó mới gọi đến hàm json

$instance=newResponse;$instance->json(['message'=>'success']);

Thật rắc rối. Với Facade, chúng ta chỉ làm đơn giản như sau:

Response::json(['message'=>'success']);

Let’s go 😃)

Khởi tạo abstract class Facade

<?phpnamespaceSupportFacades;useRuntimeException;abstractclassFacade{//}

Đặc điểm của abstract là các bạn chỉ có thể extends mà không thể tạo instance của nó.
Abstract class này sẽ là khuôn mẫu cho các Facade mà chúng ta sẽ khởi tạo sau này.

Khởi tạo hàm getFacadeAccessor

/**
 * Get the registered name of the component.
 *
 * @return string
 *
 * @throws RuntimeException
 */protectedstaticfunctiongetFacadeAccessor(){thrownewRuntimeException('Facade does not implement getFacadeAccessor method.');}

Nếu bạn không khai báo hàm getFacadeAccessor ở class con, khi hàm getFacadeAccessor được gọi, một RuntimeException sẽ được bắn ra.
Điều đó có nghĩa là hàm getFacadeAccessor là bắt buộc ở class con.
Class con sẽ phải khai báo và trả về một string là class name cần khởi tạo.

Xử lý các lệnh gọi tĩnh(static) đến đối tượng

Khi bạn gọi đến một phương thức static không có trong facade, magic method __callStatic sẽ được gọi.
Chúng ta sẽ thực hiện khởi tạo instance và gọi tới hàm cần thiết.

/**
     * Handle dynamic, static calls to the object.
     *
     * @param  string  $method
     * @param  array  $arguments
     * @return mixed
     *
     * @throws RuntimeException
     */publicstaticfunction__callStatic($method,$arguments){$instance=static::resolvedInstance(static::getFacadeAccessor());if(!$instance){thrownewRuntimeException('A facade root has not been set.');}return$instance->$method(...$arguments);}

Hàm resolvedInstance có nhiệm vụ kiểm tra instance đã được tạo trước đó hay chưa.
Nếu chưa thì thực hiện khởi tạo, lưu vào static property $resolvedInstance và return instance.

/**
     * The resolved object instances.
     *
     * @var array
     */protectedstatic$resolvedInstance;/**
     * Resolve the facade root instance.
     *
     * @param  string  $name
     * @return mixed
     */protectedstaticfunctionresolvedInstance(string$accessor){if(isset(static::$resolvedInstance[$accessor])){returnstatic::$resolvedInstance[$accessor];}returnstatic::$resolvedInstance[$accessor]=new$accessor;}

Tạo Response Facade

Response Facade sẽ extend từ abstract class Facade chúng ta đã khởi tạo ở trên

<?phpnamespaceSupportFacades;classResponse{/**
     * Get the registered name of the component.
     *
     * @return string
     */protectedstaticfunctiongetFacadeAccessor(){returnCoreHttpResponse::class;}}

Hàm getFacadeAccessor sẽ return class name của CoreHttpResponse cho phép facades khởi tạo instance từ nó.

Sử dụng

Bây giờ chỉ cần gọi SupportFacadesResponse thay vì CoreHttpResponse;

<?phpuseSupportFacadesResponse;Response::json(['message'=>'success']);

Thật là đơn giản phải không 😃).

Các bạn có thể xem file full tại đây

Nếu thấy hay các bạn cho mình xin một vote một star nhe.

Thank for watching ❤️

Nguồn: viblo.asia

Bài viết liên quan

WebP là gì? Hướng dẫn cách để chuyển hình ảnh jpg, png qua webp

WebP là gì? WebP là một định dạng ảnh hiện đại, được phát triển bởi Google

Điểm khác biệt giữa IPv4 và IPv6 là gì?

IPv4 và IPv6 là hai phiên bản của hệ thống địa chỉ Giao thức Internet (IP). IP l

Check nameservers của tên miền xem website trỏ đúng chưa

Tìm hiểu cách check nameservers của tên miền để xác định tên miền đó đang dùn

Mình đang dùng Google Domains để check tên miền hàng ngày

Từ khi thông báo dịch vụ Google Domains bỏ mác Beta, mình mới để ý và bắt đầ