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

Thay đổi Package Name của Android Studio dể dàng với plugin APR

Nếu bạn đang gặp khó khăn hoặc bế tắc trong việc thay đổi package name trong And

Lỗi không Update Meta_Value Khi thay thế hình ảnh cũ bằng hình ảnh mới trong WordPress

Mã dưới đây hoạt động tốt có 1 lỗi không update được postmeta ” meta_key=

Bài 1 – React Native DevOps các khái niệm và các cài đặt căn bản

Hướng dẫn setup jenkins agent để bắt đầu build mobile bằng jenkins cho devloper an t

Chuyển đổi từ monolith sang microservices qua ví dụ

1. Why microservices? Microservices là kiến trúc hệ thống phần mềm hướng dịch vụ,