ZetCode

PHP Namespaces

last modified March 13, 2025

Namespaces in PHP are used to organize code and avoid naming conflicts between classes, functions, and constants. They allow you to group related code under a unique name, making it easier to manage large projects. This tutorial covers the basics of PHP namespaces with practical examples.

Basic Namespace Usage

Namespaces are declared using the namespace keyword. They help prevent conflicts between classes or functions with the same name.

basic_namespace.php
<?php

namespace Blog\Entities;

class Post {
    public string $title;

    public function __construct(string $title) {
        $this->title = $title;
        echo "Post '{$this->title}' created in Blog\Entities.\n";
    }
}

$post = new Post("PHP Basics");

This example demonstrates a practical use of namespaces in a blogging system. The Blog\Entities namespace groups entities related to blog content. Here, a Post class is defined with a $title property.

When instantiated with new Post("PHP Basics"), the constructor sets the title and outputs a confirmation message. Without namespaces, a Post class might conflict with another Post class in a different context (e.g., a forum system). The namespace ensures uniqueness.

Using Multiple Namespaces

You can define multiple namespaces in a single file, but it is not recommended due to potential confusion. Instead, use separate files for each namespace.

multiple_namespaces.php
<?php

namespace Ecommerce\Products;

class Product {
    public function getDescription(): string {
        return "Product from Ecommerce\Products namespace.";
    }
}

namespace Blog\Entities;

class Product {
    public function getDescription(): string {
        return "Product from Blog\Entities namespace.";
    }
}

$shopItem = new \Ecommerce\Products\Product();
$blogItem = new \Blog\Entities\Product();

echo $shopItem->getDescription() . "\n";
echo $blogItem->getDescription() . "\n";

This example shows two namespaces in one file for demonstration: Ecommerce\Products and Blog\Entities. Each defines a Product class with a getDescription method, simulating products in an online store and blog merchandise.

Using fully qualified names (\Ecommerce\Products\Product and \Blog\Entities\Product), we instantiate objects from each namespace. The output distinguishes the two, showing how namespaces prevent naming collisions. In practice, these would be in separate files (e.g., Product.php under respective directories).

Namespace Aliasing

You can use the use keyword to create aliases for namespaces or classes, making it easier to reference them.

namespace_alias.php
<?php
namespace Shop;

class Order {
    public int $id;

    public function __construct(int $id) {
        $this->id = $id;
        echo "Order #{$this->id} created in Shop namespace.\n";
    }
}

namespace Checkout;

use Shop\Order as ShopOrder;

$order = new ShopOrder(12345);

This example simulates an e-commerce checkout process. The Shop namespace contains an Order class with an $id property. The Checkout namespace uses this class via an alias.

The use Shop\Order as ShopOrder statement creates a shorthand ShopOrder for Shop\Order. When we instantiate new ShopOrder(12345), it creates an order from the Shop namespace. Aliasing simplifies code in files that frequently reference external namespaces, improving readability.

Global Namespace

Code that is not enclosed in a namespace belongs to the global namespace. You can reference global classes or functions using a backslash (\).

global_namespace.php
<?php

namespace Blog\Utilities;

class Logger {
    public function log(string $message): void {
        echo "Logged: $message\n";
    }
}

$logger = new Logger();
$logger->log("Starting blog process");

$date = new \DateTime();
echo $date->format("Y-m-d H:i:s") . "\n";

This example uses a Blog\Utilities namespace for a Logger class, which might log events in a blog application. The class is instantiated locally within the namespace without needing a fully qualified name since it's in the current scope.

To access the global DateTime class, we use new \DateTime() with a leading backslash, indicating the global namespace. This is necessary because, within a namespace, PHP assumes unqualified class names belong to the current namespace unless specified otherwise. The output shows a log message and the current date-time.

Nested Namespaces

Namespaces can be nested to create a hierarchical structure, which is useful for organizing large projects.

nested_namespace.php
<?php
namespace Framework\Database\Drivers;

class MySQLDriver {
    public function connect(): string {
        return "Connected to MySQL database.";
    }
}

$driver = new \Framework\Database\Drivers\MySQLDriver();
echo $driver->connect() . "\n";

This example models a database layer in a framework. The nested namespace Framework\Database\Drivers organizes database-related drivers, here defining a MySQLDriver class with a connect method. Nested namespaces reflect a logical hierarchy: framework → database → drivers.

The fully qualified name \Framework\Database\Drivers\MySQLDriver is used to instantiate the class from outside its namespace. This structure is common in large applications, allowing developers to group related functionality (e.g., all database drivers) under a parent namespace, enhancing organization and clarity.

Autoloading with Namespaces

PHP's autoloading feature allows you to automatically load classes from namespaces without manually including files. This is done using the spl_autoload_register function.

autoload.php
<?php
spl_autoload_register(function (string $class): void {
    $file = str_replace("\\", "/", $class) . ".php";
    if (file_exists($file)) {
        include $file;
    }
});

use Ecommerce\Entities\Customer;

$customer = new Customer("Jane Doe", "jane@example.com");
echo $customer->getDetails() . "\n";
Ecommerce/Entities/Customer.php
<?php

namespace Ecommerce\Entities;

class Customer {
    private string $name;
    private string $email;

    public function __construct(string $name, string $email) {
        $this->name = $name;
        $this->email = $email;
    }

    public function getDetails(): string {
        return "Customer: {$this->name}, Email: {$this->email}";
    }
}

This example implements autoloading for an e-commerce application. The spl_autoload_register function registers a callback that converts namespace separators (\) to directory separators (/) and appends ".php" to locate class files (e.g., Ecommerce\Entities\Customer becomes Ecommerce/Entities/Customer.php).

The Customer class, stored in a separate file, is autoloaded when instantiated via new Customer() after being imported with use. This eliminates the need for manual include statements, making code cleaner and scalable. The output displays customer details, showing the practical use of autoloading in real projects.

Namespaces and Constants

Constants can also be defined within namespaces. Use the const keyword to define constants in a namespace.

namespace_constants.php
<?php
namespace Config;

const API_KEY = "xyz123";
const MAX_USERS = 100;

echo "API Key: " . API_KEY . "\n";
echo "Max Users: " . MAX_USERS . "\n";

This example defines configuration constants within a Config namespace, suitable for an application's settings. The constants API_KEY and MAX_USERS are scoped to Config, preventing conflicts with similar constants elsewhere (e.g., a different API_KEY in another module).

When accessed within the same namespace, they're referenced directly (e.g., API_KEY). From outside, you'd use \Config\API_KEY. The output displays both values, illustrating how namespaces organize constants in a practical configuration context.

Namespaces and Functions

Functions can also be organized within namespaces to avoid naming conflicts.

namespace_functions.php
<?php
namespace Utils\Math;

function calculateArea(float $radius): float {
    return pi() * $radius * $radius;
}

echo "Circle area: " . \Utils\Math\calculateArea(5) . "\n";

This example places a calculateArea function in a Utils\Math namespace, useful for a utility library. It calculates the area of a circle given a radius, leveraging PHP's built-in pi() function.

The function is called with \Utils\Math\calculateArea(5) from outside its namespace, returning approximately 78.54. Namespacing ensures this calculateArea doesn't conflict with another function of the same name (e.g., in a geometry or physics module), making it reusable across projects.

Namespaces with Traits

Traits can be defined within namespaces to encapsulate reusable behavior.

namespace_traits.php
<?php
namespace Blog\Traits;

trait Timestampable {
    public function getTimestamp(): string {
        return (new \DateTime())->format("Y-m-d H:i:s");
    }
}

namespace Blog\Entities;

use Blog\Traits\Timestampable;

class Article {
    use Timestampable;

    public function publish(): string {
        return "Published at: " . $this->getTimestamp();
    }
}

$article = new Article();
echo $article->publish() . "\n";

This example introduces a Timestampable trait in the Blog\Traits namespace, providing a method to get the current timestamp. Traits are reusable code blocks, and namespacing them keeps them organized within a blog application.

The Article class in Blog\Entities uses this trait via use Blog\Traits\Timestampable. When publish() is called, it leverages the trait's method to include the timestamp. This shows how namespaces enable modular, reusable functionality in larger systems.

Namespace Conflict Resolution

Namespaces resolve conflicts when integrating third-party libraries with similar class names.

namespace_conflict.php
<?php

namespace Vendor\Logger;

class Logger {
    public function log(string $msg): void {
        echo "Vendor log: $msg\n";
    }
}

namespace App\Logger;

class Logger {
    public function log(string $msg): void {
        echo "App log: $msg\n";
    }
}

namespace App;

use Vendor\Logger as VendorLogger;
use App\Logger as AppLogger;

$vendorLog = new VendorLogger();
$appLog = new AppLogger();

$vendorLog->log("Vendor error");
$appLog->log("App event");

This example simulates integrating a third-party logging library (Vendor\Logger) with an application's own logging system (App\Logger). Both define a Logger class, which would conflict without namespaces.

In the App namespace, aliases (VendorLogger and AppLogger) distinguish the two classes. Instantiating and calling log() on each produces distinct outputs, demonstrating how namespaces and aliases resolve naming conflicts in real-world scenarios.

Namespaces in MVC Structure

Namespaces organize code in an MVC (Model-View-Controller) architecture.

namespace_mvc.php
<?php

namespace App\Models;

class User {
    public function getName(): string {
        return "John Doe";
    }
}

namespace App\Controllers;

use App\Models\User;

class UserController {
    private User $user;

    public function __construct() {
        $this->user = new User();
    }

    public function show(): string {
        return "User: " . $this->user->getName();
    }
}

namespace App;

use App\Controllers\UserController;

$controller = new UserController();
echo $controller->show() . "\n";

This example applies namespaces to an MVC structure for a simple app. The App\Models namespace holds the User model, while App\Controllers contains the UserController. This mirrors a typical web application layout.

The controller uses the model via use App\Models\User and instantiates it to display a user's name. The main App namespace ties it together, showing how namespaces separate concerns (models, controllers) while enabling interaction, a common pattern in frameworks like Laravel.

Best Practices for Namespaces

Source

PHP Namespaces Documentation

In this tutorial, we explored how to use namespaces in PHP to organize code, avoid naming conflicts, and improve code readability. Namespaces are essential for managing large projects and maintaining clean, modular code.

Author

My name is Jan Bodnar, and I am a passionate programmer with many years of programming experience. I have been writing programming articles since 2007. So far, I have written over 1400 articles and 8 e-books. I have over eight years of experience in teaching programming.

List all PHP tutorials.