使用场景
大部分情况都需要多个文件才可以完成业务,这就会出现类名冲突的情况发生,命名空间可以解决这类问题。
在操作系统中目录用来将文件管理,它就扮演了命名空间的角色。例如文件 foo.txt 可以同时在目录 /home/grep
和 /home/other
中存在,但在同一个目录中不能存在两个 foot.txt 文件。另外,在目录 /home/grep
外访问 foo.txt 文件时,我们必须将目录名以及目录分隔符放在文件名之前得到 /home/grep/foo.txt
。这个原理应用到程序设计领域就是命名空间的概念。
基本使用
默认情况下常量、类和函数名都放在全局空间下。命名空间通过 namespace 来声明。
namespace 必须定义在文件头部,并在declare(strict_types=1)
语句下面。
helper.php
function sum()
{
return 'helper sum';
}
test.php
namespace App;
include 'helper.php';
function sum()
{
return 'laot sum';
}
# 使用当前命名空间中的sum
echo sum();
# 使用 helper.php 中的sum
echo \sum();
- 不指定命名空间时将使用当前命名空间
- 如果命名空间中的函数或常量未定义,将会使用全局命名空间
子命名空间
PHP 命名空间也允许指定层次化的命名空间的名称,如 App\Controller
形式,一般情况下我们将层次与目录结构匹配。
namespace App\Controller;
class Bootstrap
{
public function make()
{
return __METHOD__;
}
}
引入方式
非限定名称
调用类时没有指定命名空间时,将使用当前命名空间。
下面在使用 User
时没有指定命名空间将使用当前命名空间。
namespace App;
class User
{
public function make()
{
return __METHOD__;
}
}
$user = new User;
echo $user->make();
限定名称
限定名称类似于文件系统中的相对路径。
Order.php
namespace App\Controller;
class Order
{
public static function make()
{
return __METHOD__;
}
}
Test.php
namespace App;
include 'Order.php';
class User
{
public static function make()
{
return __METHOD__;
}
}
echo Controller\Order::make();
使用限定名称调用 Order
类,系统会使用当前命名空间加上 Controller
做为 Order
类的命名空间。
完全限定名称
类似于文件系统中的绝对路径,以下面的代码为例,调用 Order 类时可以使用以下的完全限定方式。
echo \App\Controller\Order::make();
常量
常量 __NAMESPACE__
的值时包含当前命名空间名称的字符串。
<?php
namespace App;
function factory($class){
return __NAMESPACE__ . '\\' . $class;
}
class User{
public static function make(){
return __METHOD__;
}
}
class Order{
public static function make(){
return __METHOD__;
}
}
echo factory('User')::make();
使用 namespace
关键字可用来显式访问当前命名空间或子命名空间中的元素。它等价于类中的 self 操作符。
<?php
namespace App;
class Order
{
public static function make()
{
return __METHOD__;
}
}
echo namespace\Order::make();
别名引入
通过操作符 use 来为类或命名空间使用别名。
基本使用
本实例使用的示例目录结构如下:
App
--Controller
-- Comment.php
-- User.php
boot.php
boot.php
namespace Laot;
use App\Controller\User;
include 'App/Controller/User.php';
echo User::make();
如果使用 use App\Controller
,在引入 User
类时方法如下:
Controller\User::make()
别名
在引入多个同名类时会有冲突情况发生,可以通过起别名的方式处理。
namespace Laot;
use App\Controller\User as Member;
include 'App/Controller/User.php';
echo Member::make();
多个引入
可以使用多行 use
或用逗号分隔多个类(空间)形式处理多个导入的情况。
namespace Laot;
use App\Controller\User as Member;
use App\Controller\Comment;
include 'App/Controller/User.php';
include 'App/Controller/Comment.php';
echo Member::make();
使用逗号分隔
<?php
...
use App\Controller\User as Member, App\Controller\Comment;
...
自动加载
注册加载
以往的 __autoload
自动加载函数以及在 php7.2 废弃使用,所以使用 spl_autoload_register
函数完成自动加载处理。
<?php
namespace App;
use App\Controller\User;
spl_autoload_register(function ($class){
$file = str_replace('\\', '/', $class).'.php';
if(!is_file($file)){
throw new \Exception("file don't exists");
}
require $file;
});
echo User::make();
使用类方法实现
<?php
namespace App;
use App\Controller\User;
class Bootstrap
{
public function autoload($class)
{
$file = str_replace('\\', '/', $class) . '.php';
if (!is_file($file)) {
throw new \Exception("file don't exists");
}
require $file;
}
}
spl_autoload_register([new Bootstrap, 'autoload']);
echo User::make();
composer
使用 composer 是现在开发建议使用的方式,composer 相关文档教程也会陆续更新,下面简略介绍使用方法。
项目初始
$ composer init
执行上面的命令并一直回车下去。
修改配置文件
个性 composer.json
配置文件,其中 autoload
配置项是自动加载设置。
- files 是自动加载文件列表,适合于加载函数
- psr-4 自动加载命名空间与目录对应关系
{
"name": "14778/htdocs",
"authors": [
{
"name": "Laot",
"email": "1477821088@qq.com"
}
],
"autoload": {
"files": [
"App/helper.php"
],
"psr-4": {
"App\\": "App"
}
},
"require": {}
}
安装与更新
修改配置文件后执行 composer install
生成vendor,如果修改过配置文件需要执行 composer update
。
配置
在项目中使用以下代码即可完成自动加载。
include 'vendor/autoload.php';