什么是命名空间?从广义上来说,命名空间是一种封装事物的方法。
php 命名空间(namespace)是在php 5.3中加入的,如果你学过c#和java,那命名空间就不算什么新事物。 不过在php当中还是有着相当重要的意义。
php 命名空间可以解决以下两类问题:
-
用户编写的代码与php内部的类/函数/常量或第三方类/函数/常量之间的名字冲突。
-
为很长的标识符名称(通常是为了缓解第一类问题而定义的)创建一个别名(或简短)的名称,提高源代码的可读性。
php 命名空间提供了一种将相关的类、函数和常量组合到一起的途径。接下来,我们一起来看看 php 命名空间的相关知识!
定义命名空间
默认情况下,所有常量、类和函数名都放在全局空间下,就和php支持命名空间之前一样。
命名空间通过关键字 namespace 来声明。如果一个文件中包含命名空间,它必须在其它所有代码之前声明命名空间。语法格式如下;
< ?php
// 定义代码在 'myproject' 命名空间中
namespace myproject;
// ... 代码 ...
你也可以在同一个文件中定义不同的命名空间代码,如:
< ?php
namespace myproject1;
// myproject1 命名空间中的php代码
namespace myproject2;
// myproject2 命名空间中的php代码
// 另一种语法
namespace myproject3 {
// myproject3 命名空间中的php代码
}
?>
不建议使用这种语法在单个文件中定义多个命名空间。建议使用下面的大括号形式的语法。
将全局的非命名空间中的代码与命名空间中的代码组合在一起,只能使用大括号形式的语法。全局代码必须用一个不带名称的 namespace 语句加上大括号括起来,例如:
在声明命名空间之前唯一合法的代码是用于定义源文件编码方式的 declare 语句。所有非 php 代码包括空白符都不能出现在命名空间的声明之前。
const connect_ok = 1;
class connection { /* ... */ }
function connect() { /* ... */ }
}
namespace { // 全局代码
session_start();
$a = myproject\connect();
echo myproject\connection::start();
}
?>
以下代码会出现语法错误:
namespace myproject; // 命名空间前出现了“” 会致命错误 - 命名空间必须是程序脚本的第一条语句
?>
子命名空间
与目录和文件的关系很象,php 命名空间也允许指定层次化的命名空间的名称。因此,命名空间的名字可以使用分层次的方式定义:
namespace myproject\sub\level; //声明分层次的单个命名空间
const connect_ok = 1;
class connection { /* ... */ }
function connect() { /* ... */ }
?>
上面的例子创建了常量 myproject\sub\level\connect_ok,类 myproject\sub\level\connection 和函数 myproject\sub\level\connect。
命名空间使用
php 命名空间中的类名可以通过三种方式引用:
-
非限定名称,或不包含前缀的类名称,例如 $a=new foo(); 或 foo::staticmethod();。如果当前命名空间是 currentnamespace,foo 将被解析为 currentnamespace\foo。如果使用 foo 的代码是全局的,不包含在任何命名空间中的代码,则 foo 会被解析为foo。 警告:如果命名空间中的函数或常量未定义,则该非限定的函数名称或常量名称会被解析为全局函数名称或常量名称。
-
限定名称,或包含前缀的名称,例如 $a = new subnamespace\foo(); 或 subnamespace\foo::staticmethod();。如果当前的命名空间是 currentnamespace,则 foo 会被解析为 currentnamespace\subnamespace\foo。如果使用 foo 的代码是全局的,不包含在任何命名空间中的代码,foo 会被解析为subnamespace\foo。
-
完全限定名称,或包含了全局前缀操作符的名称,例如, $a = new \currentnamespace\foo(); 或 \currentnamespace\foo::staticmethod();。在这种情况下,foo 总是被解析为代码中的文字名(literal name)currentnamespace\foo。
下面是一个使用这三种方式的实例:
file1.php 文件代码
file2.php 文件代码
注意访问任意全局类、函数或常量,都可以使用完全限定名称,例如 \strlen() 或 \exception 或 \ini_all。
在命名空间内部访问全局类、函数和常量:
命名空间和动态语言特征
php 命名空间的实现受到其语言自身的动态特征的影响。因此,如果要将下面的代码转换到命名空间中,动态访问元素。
example1.php 文件代码:
必须使用完全限定名称(包括命名空间前缀的类名称)。注意因为在动态的类名称、函数名称或常量名称中,限定名称和完全限定名称没有区别,因此其前导的反斜杠是不必要的。
动态访问命名空间的元素
namespace关键字和__namespace__常量
php支持两种抽象的访问当前命名空间内部元素的方法,__namespace__ 魔术常量和 namespace 关键字。
常量 __namespace__ 的值是包含当前命名空间名称的字符串。在全局的,不包括在任何命名空间中的代码,它包含一个空的字符串。
__namespace__ 示例, 在命名空间中的代码
__namespace__ 示例,全局代码
常量 __namespace__ 在动态创建名称时很有用,例如:
使用 __namespace__ 动态创建名称
关键字 namespace 可用来显式访问当前命名空间或子命名空间中的元素。它等价于类中的 self 操作符。
amespace 操作符,命名空间中的代码
namespace 操作符, 全局代码
使用命名空间:别名/导入
php 命名空间支持 有两种使用别名或导入方式:为类名称使用别名,或为命名空间名称使用别名。注意php不支持导入函数或常量。
在php中,别名是通过操作符 use 来实现的. 下面是一个使用所有可能的三种导入方式的例子:
1、使用 use 操作符导入/使用别名
2、 一行中包含多个 use 语句
导入操作是在编译执行的,但动态的类名称、函数名称或常量名称则不是。
3、导入和动态名称
另外,导入操作只影响非限定名称和限定名称。完全限定名称由于是确定的,故不受导入的影响。
4、导入和完全限定名称
使用命名空间:后备全局函数/常量
在一个命名空间中,当 php 遇到一个非限定的类、函数或常量名称时,它使用不同的优先策略来解析该名称。类名称总是解析到当前命名空间中的名称。因此在访问系统内部或不包含在命名空间中的类名称时,必须使用完全限定名称,例如:
1、在命名空间中访问全局类
对于函数和常量来说,如果当前命名空间中不存在该函数或常量,php 会退而使用全局空间中的函数或常量。
2、 命名空间中后备的全局函数/常量
全局空间
如果没有定义任何命名空间,所有的类与函数的定义都是在全局空间,与 php 引入命名空间概念前一样。在名称前加上前缀 \ 表示该名称是全局空间中的名称,即使该名称位于其它的命名空间中时也是如此。
使用全局空间说明
命名空间的顺序
自从有了命名空间之后,最容易出错的该是使用类的时候,这个类的寻找路径是什么样的了。
名称解析遵循下列规则:
-
对完全限定名称的函数,类和常量的调用在编译时解析。例如 new \a\b 解析为类 a\b。
-
所有的非限定名称和限定名称(非完全限定名称)根据当前的导入规则在编译时进行转换。例如,如果命名空间 a\b\c 被导入为 c,那么对 c\d\e() 的调用就会被转换为 a\b\c\d\e()。
-
在命名空间内部,所有的没有根据导入规则转换的限定名称均会在其前面加上当前的命名空间名称。例如,在命名空间 a\b 内部调用 c\d\e(),则 c\d\e() 会被转换为 a\b\c\d\e() 。
-
非限定类名根据当前的导入规则在编译时转换(用全名代替短的导入名称)。例如,如果命名空间 a\b\c 导入为c,则 new c() 被转换为 new a\b\c() 。
-
在命名空间内部(例如a\b),对非限定名称的函数调用是在运行时解析的。例如对函数 foo() 的调用是这样解析的:
-
在当前命名空间中查找名为 a\b\foo() 的函数
-
尝试查找并调用 全局(global) 空间中的函数 foo()。
-
在命名空间(例如a\b)内部对非限定名称或限定名称类(非完全限定名称)的调用是在运行时解析的。下面是调用new c() 及 new d\e() 的解析过程:
new c()的解析:- 在当前命名空间中查找a\b\c类。
- 尝试自动装载类a\b\c。
new d\e() 的解析:
- 在类名称前面加上当前命名空间名称变成:a\b\d\e,然后查找该类。
- 尝试自动装载类 a\b\d\e。
为了引用全局命名空间中的全局类,必须使用完全限定名称 new \c()。