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

9 Mẹo lập trình Web “ẩn mình” giúp tiết kiệm hàng giờ đồng hồ

Hầu hết các lập trình viên (kể cả những người giỏi) đều tốn thời gian x

Can GPT-4o Generate Images? All You Need to Know about GPT-4o-image

OpenAI‘s GPT-4o, introduced on March 25, 2025, has revolutionized the way we create visual con

Khi nào nên dùng main, section, article, header, footer, và aside trong HTML5

HTML5 đã giới thiệu các thẻ ngữ nghĩa giúp cấu trúc nội dung web một cách có

So sánh Webhook và API: Khi nào nên sử dụng?

Trong lĩnh vực công nghệ thông tin và phát triển phần mềm, Webhook và API là hai th