Home >Backend Development >PHP Tutorial >PHP Design Patterns: Front Controller

PHP Design Patterns: Front Controller

Barbara Streisand
Barbara StreisandOriginal
2024-12-23 20:07:11660browse

PHP Design Patterns: Front Controller

The Front Controller is a design pattern used in web application development to centralize request handling. Instead of having multiple entry points for different parts of the system, all requests are routed through a single central controller, responsible for directing them to the appropriate component or module.

How It Works

  1. Single Entry Point: All HTTP requests are redirected to a single file (typically index.php) using web server configuration (e.g., .htaccess for Apache or routing rules in Nginx).
  2. Routing: The Front Controller analyzes the URL and determines which part of the code should execute. This can be implemented manually or with routing libraries/frameworks.
  3. Delegation: Based on the route, the Front Controller delegates the request to the appropriate controller (class or method), which processes the data and returns a response.
  4. Response: The controller generates a response (usually HTML or JSON) sent back to the browser or client.

Advantages

  • Centralization: All incoming application flows are handled through a single point, making it easier to manage and track requests.
  • Flexibility: Easily integrates global features like authentication, permission control, logging, or error handling.
  • Reusability: Common logic can be centralized in the Front Controller, reducing duplication.
  • Maintainability: Centralization simplifies updates, such as adding new routes or controllers.

Example

Directory Structure

/app
    /Controllers
        HomeController.php
        ProductController.php
    /Core
        Entrypoint.php
/config
    routes.php
/public
    index.php
/vendor
composer.json

Autoload

To implement PSR-4, create a composer.json file in the project root:

{
    "autoload": {
        "psr-4": {
            "App\": "app/"
        }
    }
}

Run the following command to generate the autoloader:

composer dump-autoload

Redirect requests

apache(.htaccess)

RewriteEngine On
RewriteCond %{REQUEST_FILENAME} !-f
RewriteCond %{REQUEST_FILENAME} !-d
RewriteRule ^ index.php [L]

nginx

server {
    listen 80;
    server_name example.com;

    root /path/to/your/project/public;
    index index.php;

    location / {
        try_files $uri $uri/ /index.php?$query_string;
    }

    location ~ \.php$ {
        include snippets/fastcgi-php.conf;
        fastcgi_pass unix:/var/run/php/php8.2-fpm.sock; # Adjust for your PHP version
        fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name;
        include fastcgi_params;
    }

    location ~* \.(css|js|jpg|jpeg|png|gif|ico|woff|woff2|ttf|svg|eot|ttc|otf|webp|avif)$ {
        expires max;
        log_not_found off;
    }

    location ~ /\.ht {
        deny all;
    }
}

After saving the configuration file, restart Nginx to apply the changes

sudo systemctl reload nginx

Controllers

HomeController

namespace App\Controllers;

class HomeController {
    public function index(): void {
        echo "Welcome to the home page!";
    }
}

ProductController

namespace App\Controllers;

class ProductController
{
    public function list(): void
    {
        echo "Product list.";
    }
}

Core

Entrypoint

namespace App\Core;

class Entrypoint {

    public function __construct(private array $routes) {
    }

    public function handleRequest(): void {
        $uri = parse_url($_SERVER['REQUEST_URI'], PHP_URL_PATH);

        if (isset($this->routes[$uri])) {
            $route = $this->routes[$uri];
            $controller = new $route['controller'];
            $method = $route['method'];

            if (method_exists($controller, $method)) {
                $controller->$method();
            } else {
                $this->sendResponse(500, "Method not found.");
            }
        } else {
            $this->sendResponse(404, "Page not found.");
        }
    }

    private function sendResponse(int $statusCode, string $message): void {
        http_response_code($statusCode);
        echo $message;
    }
}

Config

routes

$routes = [
    '/' => [
        'controller' => 'HomeController',
        'method' => 'index'
    ],
    '/products' => [
        'controller' => 'ProductController',
        'method' => 'list'
    ]
];

index (Front Controller)

require_once __DIR__ . '/../vendor/autoload.php';

use App\Core\Entrypoint;

$routes = require __DIR__ . '/../config/routes.php';

$entrypoint = new Entrypoint($routes);
$entrypoint->handleRequest();

Result

This implementation:

  • Centralizes request handling using the Front Controller pattern.
  • Encapsulates routing logic within the Entrypoint class.
  • Adopts PSR-4 for autoloading and better code organization.
  • Uses Nginx configuration for a seamless setup.

The above is the detailed content of PHP Design Patterns: Front Controller. 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