在另一个PHP文件中包含此文件。 myapp.php 1 2 3 4 5 6 7 ?php header( 'Content-type: text/plain' ); require_once ( 'lib1.php' ); echo AppLib1MYCONST . "n" ; echo AppLib1MyFunction() . "n" ; echo AppLib1MyClass::WhoAmI() . "n" ; ?
在另一个PHP文件中包含此文件。
myapp.php
3
6 7 |
echo AppLib1MyClass::WhoAmI() . "n" ;
?>
|
1 2 3 4 5 6 7 8 9 10 11 12 13 |
|
1 2 3 4 5 6 7 8 9 10 11 12 13 |
<?php
// application library 1
namespace AppLib1;
const MYCONST = 'AppLib1MYCONST' ;
function MyFunction() {
return __FUNCTION__ ;
}
class MyClass {
static function WhoAmI() {
return __METHOD__ ;
}
}
?>
|
1 2 3 4 5 6 7 8 9 10 11 12 13 |
<?php
// application library 2
namespace AppLib2;
const MYCONST = 'AppLib2MYCONST' ;
function MyFunction() {
return __FUNCTION__ ;
}
class MyClass {
static function WhoAmI() {
return __METHOD__ ;
}
}
?>
|
先来了解一些PHP术语:
完全限定名
任何PHP代码都具有完全限定名——一个以反斜线起始的标识符。例如:AppLib1MYCONST, AppLib2MyFunction()。
完全限定名不会产生任何歧义。首个反斜线与文件系统路径类似,都指明了“根”全局空间。如果在全局空间内实现了一个MyFunction()函数,可在lib1.php和lib2.php中使用“MyFunction()”对其进行调用。
完全限定名对一次性函数调用或对象初始化有帮助。然而,它们不适用于大量调用的情况。就像接下来要看到的,PHP为命名空间输入提供了其他一些选择。
部分限定名
至少有一个命名空间分隔符的标识符。例如:Lib1MyFunction().
未限定名
没有命名空间分隔符的标识符。例如:MyFunction().
在同一命名空间内工作
考虑以下代码:
myapp1.php
虽然同时包含了lib1.php和lib2.php,但MYCONST, MyFunction, and MyClass只会引用lib1.php中的代码。因为myapp1.php的代码同样位于App\lib1命名空间内。 result: App\Lib1\MYCONST App\Lib1\MyFunction App\Lib1\MyClass::WhoAmI <strong>命名空间导入</strong> 命名空间可通过use操作符导入。例如 myapp2.php3
4
6 7 8 9 App\Lib2\MYCONST App\Lib2\MyFunction App\Lib2\MyClass::WhoAmI |
namespace AppLib1;
require_once ( 'lib1.php' );
require_once ( 'lib2.php' );
echo MYCONST . "n" ;
echo MyFunction() . "n" ;
echo MyClass::WhoAmI() . "n" ;
?>
|
1 2 3 4 5 6 7 8 9 |
<?php
use AppLib2;
require_once ( 'lib1.php' );
require_once ( 'lib2.php' );
header( 'Content-type: text/plain' );
echo Lib2MYCONST . "n" ;
echo Lib2MyFunction() . "n" ;
echo Lib2MyClass::WhoAmI() . "n" ;
?>
|
1 2 3 4 5 6 7 8 9 10 11 |
<?php
use AppLib1 as L;
use AppLib2MyClass as Obj;
header( 'Content-type: text/plain' );
require_once ( 'lib1.php' );
require_once ( 'lib2.php' );
echo LMYCONST . "n" ;
echo LMyFunction() . "n" ;
echo LMyClass::WhoAmI() . "n" ;
echo Obj::WhoAmI() . "n" ;
?>
|
第一个use语句设定"App/Lib1"的别名为“L”,任何使用"L"的部分限定名都会在编译阶段被解释为“App/Lib1”。因此可使用 L\MYCONST 和 L\MyFunction来代替完全限定名。
第二个use语句非常有趣。它将'Obj'设定为App\Lib2命名空间内的类MyClass的别名。这种别名设定方式只对类有效,常量与函数不适用。
设定类别名后,就可以用new Obj()来创建实例或直接调用静态方法了。
App\Lib1\MYCONST App\Lib1\MyFunction App\Lib1\MyClass::WhoAmI App\Lib2\MyClass::WhoAmI
PHP命名解析规则
PHP标识符命名解析遵循以下规则:
1. 在编译阶段解析拥有完全限定名的常量、类或函数的调用。
2. 未限定和部分限定名根据命名空间导入规则进行解释。例如,如果命名空间A\B\C被导入且设定别名为C,那么C\D\e()调用将被解释为A\B\C\D\e()
3. 在命名空间内,所有未根据命名空间导入规则进行解释的部分限定名都会拥有当前命名空间的前缀。例如:如果在命名空间A\B内执行C\D\e()调用,它将会被解释为 A\B\C\D\e()。
4. 未限定的类名会根据当前导入的命名空间进行解释,同时会使用全名来代替缩写后的名称。例如:如果在命名空间A\B内的类C被导入为X,new X()将被解释为 new A\B\C()。<br> 5. 命名空间内的未限定函数调用会在运行时解析。例如:如果MyFunction()在命名空间A\B内被调用,PHP首先会搜索函数\A\B\MyFunction(),如果未找到匹配项,PHP会在全局空间内搜索\MyFunction()函数。<br> 6. 调用未限定或部分限定名的类会在运行时解析。例如:如果在命名空间A\B内调用new C(),PHP会搜索类A\B\C,如果未能找到匹配项,PHP将会试图自动载入类A\B\C。