(PHP 5 >= 5.3.0, PHP 7, PHP 8)
本文分兩節(jié):常見問題、有助于完全理解的實現(xiàn)詳情。
首先,常見問題。
\my\name
和 \name
這樣的名稱是如何解析的?
my\name
這樣的名稱是如何解析的?
name
這樣的非限定類名是如何解析的?
name
這樣的非限定常量和函數(shù)名是如何解析的?
為了幫助理解,我們提供了一些命名空間實現(xiàn)細節(jié)。
不需要。命名空間不影響現(xiàn)存的代碼,也不影響即將要寫下的不含命名空間的代碼。 想要的話可以這樣寫:
示例 #1 在命名空間之外訪問全局類
<?php
$a = new \stdClass;
以上等同于:
示例 #2 在命名空間之外訪問全局類
<?php
$a = new stdClass;
示例 #3 在命名空間內(nèi)訪問內(nèi)置的類
<?php
namespace foo;
$a = new \stdClass;
function test(\ArrayObject $parameter_type_example = null) {}
$a = \DirectoryIterator::CURRENT_AS_FILEINFO;
// 擴展內(nèi)置或全局的 class
class MyException extends \Exception {}
?>
示例 #4 在命名空間中訪問內(nèi)置的類、函數(shù)、常量
<?php
namespace foo;
class MyClass {}
// 以當前命名空間中的 class 作為參數(shù)的類型
function test(MyClass $parameter_type_example = null) {}
// 以當前命名空間中的 class 作為參數(shù)的類型的另一種方式
function test(\foo\MyClass $parameter_type_example = null) {}
// 在當前命名空間中擴展一個類
class Extended extends MyClass {}
// 訪問全局函數(shù)
$a = \globalfunc();
// 訪問全局常量
$b = \INI_ALL;
?>
\my\name
和 \name
這樣的名稱是如何解析的?
以 \
開頭的名稱總是會解析成原樣,
因此 \my\name
實際上是 my\name
,
而 \Exception
是 Exception
。
示例 #5 完全限定名稱
<?php
namespace foo;
$a = new \my\name(); // class "my\name" 的實例
echo \strlen('hi'); // 調(diào)用函數(shù) "strlen"
$a = \INI_ALL; // $a 的值設置成常量 "INI_ALL"
?>
my\name
這樣的名稱是如何解析的?
像 my\name
這樣包含反斜線的名稱,但不以反斜線開頭的名稱,
能夠以兩種不同的方式解析。
如果有個導入語句,將其他名字設置別名為 my
,
則導入別名會應用到 my\name
的 my
部分。
如果沒有導入,就會追加當前的命名空間名稱為 my\name
的前綴。
示例 #6 限定名稱
<?php
namespace foo;
use blah\blah as foo;
$a = new my\name(); // class "foo\my\name" 的實例
foo\bar::name(); // 調(diào)用 class "blah\blah\bar" 的靜態(tài)方法 "name"
my\bar(); // 調(diào)用函數(shù) "foo\my\bar"
$a = my\BAR; // 設置 $a 的值為 "foo\my\BAR"
?>
name
這樣的非限定名稱是如何解析的?
像 name
這樣不包含反斜線的名稱,
能夠以兩種不同的方式解析。
如果有導入語句,設置別名為 name
,就會應用導入別名。
如果沒有,就會把當前命名空間添加到 name
的前綴。
示例 #7 非限定類名
<?php
namespace foo;
use blah\blah as foo;
$a = new name(); // class "foo\name" 的實例
foo::name(); // 調(diào)用 class "blah\blah" 的靜態(tài)方法 "name"
?>
name
這樣的非限定常量和函數(shù)名是如何解析的?
像 name
這樣不包含反斜線的常量和函數(shù)名,能以兩種不同的方式解析。
首先,當前命名空間會添加到 name
的前綴。
然后,如果當前命名空間不存在函數(shù)和常量 name
,
而全局存在,就會使用全局的函數(shù)和常量 name
。
示例 #8 非限定函數(shù)和常量名
<?php
namespace foo;
use blah\blah as foo;
const FOO = 1;
function my() {}
function foo() {}
function sort(&$a)
{
sort($a);
$a = array_flip($a);
return $a;
}
my(); // 調(diào)用 "foo\my"
$a = strlen('hi'); // 由于 "foo\strlen" 不存在,所以調(diào)用全局的 "strlen"
$arr = array(1,3,2);
$b = sort($arr); // 調(diào)用函數(shù) "foo\sort"
$c = foo(); // 未導入,調(diào)用函數(shù) "foo\foo"
$a = FOO; // 未導入,設置 $a 為常量 "foo\FOO" 的值
$b = INI_ALL; // 設置 $b 為全局常量 "INI_ALL" 的值
?>
允許以下腳本中的組合:
file1.php
<?php
namespace my\stuff;
class MyClass {}
?>
another.php
<?php
namespace another;
class thing {}
?>
file2.php
<?php
namespace my\stuff;
include 'file1.php';
include 'another.php';
use another\thing as MyClass;
$a = new MyClass; // class "thing" 的實例來自于命名空間 another
?>
盡管在 my\stuff
命名空間中存在 MyClass
,
因為類定義在了獨立的文件中,所以不會發(fā)生名稱沖突。
不過,接下來的例子中,因為 MyClass 定義在了 use 語句的同一個文件中,
所以發(fā)生了名稱沖突,導致了 fatal 錯誤。
<?php
namespace my\stuff;
use another\thing as MyClass;
class MyClass {} // fatal error: MyClass conflicts with import statement
$a = new MyClass;
?>
PHP 不允許嵌套 namespace
<?php
namespace my\stuff {
namespace nested {
class foo {}
}
}
?>
<?php
namespace my\stuff\nested {
class foo {}
}
?>
重要的是,字符串中反斜線是一個轉(zhuǎn)義字符,因此在字符串中使用時,必須要寫兩遍。 否則就會在無意中造成一些后果:
示例 #9 在雙引號字符串中使用命名空間的危險性
<?php
$a = new "dangerous\name"; // 在雙引號字符串中,\n 是換行符!
$obj = new $a;
$a = new 'not\at\all\dangerous'; // 這里沒有問題
$obj = new $a;
?>
像 FOO
這樣的非限定名稱常量,如果使用的時候還沒定義,
會產(chǎn)生一個 notice。PHP 會假設該常量的值是 FOO
。
如果沒有找到包含反斜線的常量,無論是完全或者不完全限定的名稱,都會產(chǎn)生 fatal 錯誤。
示例 #10 未定義的常量
<?php
namespace bar;
$a = FOO; // 產(chǎn)生 notice - undefined constants "FOO" assumed "FOO";
$a = \FOO; // fatal error, undefined namespace constant FOO
$a = Bar\FOO; // fatal error, undefined namespace constant bar\Bar\FOO
$a = \Bar\FOO; // fatal error, undefined namespace constant Bar\FOO
?>
在命名空間內(nèi)定義特殊的內(nèi)置常量,會導致 fatal 錯誤
示例 #11 未定義的常量
<?php
namespace bar;
const NULL = 0; // fatal error;
const true = 'stupid'; // 也是 fatal error;
// etc.
?>