默認情況下,所有常量、類和函數名都放在全局空間下,就和PHP支持命名空間之前一樣。
命名空間通過關鍵字namespace 來聲明。如果一個文件中包含命名空間,它必須在其它所有代碼之前聲明命名空間。語法格式如下;
?php // 定義代碼在 'MyProject' 命名空間中 namespace MyProject; // ... 代碼 ...
你也可以在同一個文件中定義不同的命名空間代碼,如:
?php namespace MyProject;const CONNECT_OK = 1;html' target='_blank'>class Connection { /* ... */ }function connect() { /* ... */ }namespace AnotherProject;const CONNECT_OK = 1;class Connection { /* ... */ }function connect() { /* ... */ }
不建議使用這種語法在單個文件中定義多個命名空間。建議使用下面的大括號形式的語法。
?phpnamespace MyProject { const CONNECT_OK = 1; class Connection { /* ... */ } function connect() { /* ... */ }namespace AnotherProject { const CONNECT_OK = 1; class Connection { /* ... */ } function connect() { /* ... */ }
將全局的非命名空間中的代碼與命名空間中的代碼組合在一起,只能使用大括號形式的語法。全局代碼必須用一個不帶名稱的 namespace 語句加上大括號括起來,例如:
?phpnamespace MyProject {const CONNECT_OK = 1;class Connection { /* ... */ }function connect() { /* ... */ }namespace { // 全局代碼session_start();$a = MyProject/connect();echo MyProject/Connection::start();?
在聲明命名空間之前唯一合法的代碼是用于定義源文件編碼方式的 declare 語句。所有非 PHP 代碼包括空白符都不能出現在命名空間的聲明之前。
?phpdeclare(encoding='UTF-8'); //定義多個命名空間和不包含在命名空間中的代碼namespace MyProject {const CONNECT_OK = 1;class Connection { /* ... */ }function connect() { /* ... */ }namespace { // 全局代碼session_start();$a = MyProject/connect();echo MyProject/Connection::start();
以下代碼會出現語法錯誤:
html ?phpnamespace MyProject; // 命名空間前出現了 html 會致命錯誤 - 命名空間必須是程序腳本的第一條語句子命名空間與目錄和文件的關系很像,PHP 命名空間也允許指定層次化的命名空間的名稱。因此,命名空間的名字可以使用分層次的方式定義:
?phpnamespace 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 文件代碼
?phpnamespace Foo/Bar/subnamespace; const FOO = 1;function foo() {}class foo static function staticmethod() {}
file2.php 文件代碼
?phpnamespace Foo/Bar;include 'file1.php';const FOO = 2;function foo() {}class foo static function staticmethod() {}/* 非限定名稱 */foo(); // 解析為函數 Foo/Bar/foofoo::staticmethod(); // 解析為類 Foo/Bar/foo ,方法為 staticmethodecho FOO; // 解析為常量 Foo/Bar/FOO/* 限定名稱 */subnamespace/foo(); // 解析為函數 Foo/Bar/subnamespace/foosubnamespace/foo::staticmethod(); // 解析為類 Foo/Bar/subnamespace/foo, // 以及類的方法 staticmethodecho subnamespace/FOO; // 解析為常量 Foo/Bar/subnamespace/FOO/* 完全限定名稱 *//Foo/Bar/foo(); // 解析為函數 Foo/Bar/foo/Foo/Bar/foo::staticmethod(); // 解析為類 Foo/Bar/foo, 以及類的方法 staticmethodecho /Foo/Bar/FOO; // 解析為常量 Foo/Bar/FOO
注意訪問任意全局類、函數或常量,都可以使用完全限定名稱,例如 /strlen() 或 /Exception 或 /INI_ALL。
在命名空間內部訪問全局類、函數和常量:
?phpnamespace Foo;function strlen() {}const INI_ALL = 3;class Exception {}$a = /strlen('hi'); // 調用全局函數strlen$b = /INI_ALL; // 訪問全局常量 INI_ALL$c = new /Exception('error'); // 實例化全局類 Exception命名空間和動態語言特征
PHP 命名空間的實現受到其語言自身的動態特征的影響。因此,如果要將下面的代碼轉換到命名空間中,動態訪問元素。
example1.php 文件代碼:
?phpclass classname function __construct() echo __METHOD__,"/n";function funcname() echo __FUNCTION__,"/n";const constname = "global";$a = 'classname';$obj = new $a; // prints classname::__construct$b = 'funcname';$b(); // prints funcnameecho constant('constname'), "/n"; // prints global必須使用完全限定名稱(包括命名空間前綴的類名稱)。注意因為在動態的類名稱、函數名稱或常量名稱中,限定名稱和完全限定名稱沒有區別,因此其前導的反斜杠是不必要的。
動態訪問命名空間的元素
?phpnamespace namespacename;class classname function __construct() echo __METHOD__,"/n";function funcname() echo __FUNCTION__,"/n";const constname = "namespaced";include 'example1.php';$a = 'classname';$obj = new $a; // 輸出 classname::__construct$b = 'funcname';$b(); // 輸出函數名echo constant('constname'), "/n"; // 輸出 global/* 如果使用雙引號,使用方法為 "http://namespacename//classname"*/$a = '/namespacename/classname';$obj = new $a; // 輸出 namespacename/classname::__construct$a = 'namespacename/classname';$obj = new $a; // 輸出 namespacename/classname::__construct$b = 'namespacename/funcname';$b(); // 輸出 namespacename/funcname$b = '/namespacename/funcname';$b(); // 輸出 namespacename/funcnameecho constant('/namespacename/constname'), "/n"; // 輸出 namespacedecho constant('namespacename/constname'), "/n"; // 輸出 namespacednamespace關鍵字和__NAMESPACE__常量
PHP支持兩種抽象的訪問當前命名空間內部元素的方法,__NAMESPACE__ 魔術常量和namespace關鍵字。
常量__NAMESPACE__的值是包含當前命名空間名稱的字符串。在全局的,不包括在任何命名空間中的代碼,它包含一個空的字符串。__NAMESPACE__ 示例, 在命名空間中的代碼
?phpnamespace MyProject;echo '"', __NAMESPACE__, '"'; // 輸出 "MyProject"
__NAMESPACE__ 示例,全局代碼
?phpecho '"', __NAMESPACE__, '"'; // 輸出 ""常量 __NAMESPACE__ 在動態創建名稱時很有用,例如:
使用__NAMESPACE__動態創建名稱
?phpnamespace MyProject;function get($classname) $a = __NAMESPACE__ . '//' . $classname; return new $a;關鍵字 namespace 可用來顯式訪問當前命名空間或子命名空間中的元素。它等價于類中的 self 操作符。namespace操作符,命名空間中的代碼
?phpnamespace MyProject;use blah/blah as mine; // see "Using namespaces: importing/aliasing"blah/mine(); // calls function blah/blah/mine()namespace/blah/mine(); // calls function MyProject/blah/mine()namespace/func(); // calls function MyProject/func()namespace/sub/func(); // calls function MyProject/sub/func()namespace/cname::method(); // calls static method "method" of class MyProject/cname$a = new namespace/sub/cname(); // instantiates object of class MyProject/sub/cname$b = namespace/CONSTANT; // assigns value of constant MyProject/CONSTANT to $bnamespace操作符, 全局代碼
?phpnamespace/func(); // calls function func()namespace/sub/func(); // calls function sub/func()namespace/cname::method(); // calls static method "method" of class cname$a = new namespace/sub/cname(); // instantiates object of class sub/cname$b = namespace/CONSTANT; // assigns value of constant CONSTANT to $b使用命名空間:別名/導入 PHP 命名空間支持 有兩種使用別名或導入方式:為類名稱使用別名,或為命名空間名稱使用別名。在PHP中,別名是通過操作符 use 來實現的. 下面是一個使用所有可能的三種導入方式的例子:
1、使用use操作符導入/使用別名
?phpnamespace foo;use My/Full/Classname as Another;// 下面的例子與 use My/Full/NSname as NSname 相同use My/Full/NSname;// 導入一個全局類use /ArrayObject;$obj = new namespace/Another; // 實例化 foo/Another 對象$obj = new Another; // 實例化 My/Full/Classname 對象NSname/subns/func(); // 調用函數 My/Full/NSname/subns/func$a = new ArrayObject(array(1)); // 實例化 ArrayObject 對象// 如果不使用 "use /ArrayObject" ,則實例化一個 foo/ArrayObject 對象
2、 一行中包含多個use語句
?phpuse My/Full/Classname as Another, My/Full/NSname;$obj = new Another; // 實例化 My/Full/Classname 對象NSname/subns/func(); // 調用函數 My/Full/NSname/subns/func導入操作是在編譯執行的,但動態的類名稱、函數名稱或常量名稱則不是。
3、導入和動態名稱
?phpuse My/Full/Classname as Another, My/Full/NSname;$obj = new Another; // 實例化一個 My/Full/Classname 對象$a = 'Another';$obj = new $a; // 實際化一個 Another 對象另外,導入操作只影響非限定名稱和限定名稱。完全限定名稱由于是確定的,故不受導入的影響。4、導入和完全限定名稱
?phpuse My/Full/Classname as Another, My/Full/NSname;$obj = new Another; // 實例化 My/Full/Classname 類$obj = new /Another; // 實例化 Another 類$obj = new Another/thing; // 實例化 My/Full/Classname/thing 類$obj = new /Another/thing; // 實例化 Another/thing 類
在一個命名空間中,當 PHP 遇到一個非限定的類、函數或常量名稱時,它使用不同的優先策略來解析該名稱。類名稱總是解析到當前命名空間中的名稱。因此在訪問系統內部或不包含在命名空間中的類名稱時,必須使用完全限定名稱,例如:1、在命名空間中訪問全局類
?phpnamespace A/B/C;class Exception extends /Exception {}$a = new Exception('hi'); // $a 是類 A/B/C/Exception 的一個對象$b = new /Exception('hi'); // $b 是類 Exception 的一個對象$c = new ArrayObject; // 致命錯誤, 找不到 A/B/C/ArrayObject 類
對于函數和常量來說,如果當前命名空間中不存在該函數或常量,PHP 會退而使用全局空間中的函數或常量。
2、 命名空間中后備的全局函數/常量
?phpnamespace A/B/C;const E_ERROR = 45;function strlen($str) return /strlen($str) - 1;echo E_ERROR, "/n"; // 輸出 "45"echo INI_ALL, "/n"; // 輸出 "7" - 使用全局常量 INI_ALLecho strlen('hi'), "/n"; // 輸出 "2"if (is_array('hi')) { // 輸出 "is not array" echo "is array/n";} else { echo "is not array/n";
使用全局空間說明
?phpnamespace A/B/C;/* 這個函數是 A/B/C/fopen */function fopen() { /* ... */ $f = /fopen(...); // 調用全局的fopen函數 return $f;命名空間的順序
自從有了命名空間之后,最容易出錯的該是使用類的時候,這個類的尋找路徑是什么樣的了。
?phpnamespace A;use B/D, C/E as F;// 函數調用foo(); // 首先嘗試調用定義在命名空間"A"中的函數foo() // 再嘗試調用全局函數 "foo"/foo(); // 調用全局空間函數 "foo" my/foo(); // 調用定義在命名空間"A/my"中函數 "foo" F(); // 首先嘗試調用定義在命名空間"A"中的函數 "F" // 再嘗試調用全局函數 "F"http:// 類引用new B(); // 創建命名空間 "A" 中定義的類 "B" 的一個對象 // 如果未找到,則嘗試自動裝載類 "A/B"new D(); // 使用導入規則,創建命名空間 "B" 中定義的類 "D" 的一個對象 // 如果未找到,則嘗試自動裝載類 "B/D"new F(); // 使用導入規則,創建命名空間 "C" 中定義的類 "E" 的一個對象 // 如果未找到,則嘗試自動裝載類 "C/E"new /B(); // 創建定義在全局空間中的類 "B" 的一個對象 // 如果未發現,則嘗試自動裝載類 "B"new /D(); // 創建定義在全局空間中的類 "D" 的一個對象 // 如果未發現,則嘗試自動裝載類 "D"new /F(); // 創建定義在全局空間中的類 "F" 的一個對象 // 如果未發現,則嘗試自動裝載類 "F"http:// 調用另一個命名空間中的靜態方法或命名空間函數B/foo(); // 調用命名空間 "A/B" 中函數 "foo"B::foo(); // 調用命名空間 "A" 中定義的類 "B" 的 "foo" 方法 // 如果未找到類 "A/B" ,則嘗試自動裝載類 "A/B"D::foo(); // 使用導入規則,調用命名空間 "B" 中定義的類 "D" 的 "foo" 方法 // 如果類 "B/D" 未找到,則嘗試自動裝載類 "B/D"/B/foo(); // 調用命名空間 "B" 中的函數 "foo" /B::foo(); // 調用全局空間中的類 "B" 的 "foo" 方法 // 如果類 "B" 未找到,則嘗試自動裝載類 "B"http:// 當前命名空間中的靜態方法或函數A/B::foo(); // 調用命名空間 "A/A" 中定義的類 "B" 的 "foo" 方法 // 如果類 "A/A/B" 未找到,則嘗試自動裝載類 "A/A/B"/A/B::foo(); // 調用命名空間 "A" 中定義的類 "B" 的 "foo" 方法 // 如果類 "A/B" 未找到,則嘗試自動裝載類 "A/B"名稱解析遵循下列規則:
Alex Gump
182***69281@163.com
可以把非限定名稱類比為文件名(例如 comment.php)、.限定名稱類比為相對路徑名(例如 ./article/comment.php)、完全限定名稱類比為絕對路徑名(例如 /blog/article/comment.php),這樣可能會更容易理解。
再添一例:
?php //創建空間Blognamespace Blog;class Comment { }//非限定名稱,表示當前Blog空間//這個調用將被解析成 Blog/Comment();$blog_comment = new Comment();//限定名稱,表示相對于Blog空間//這個調用將被解析成 Blog/Article/Comment();$article_comment = new Article/Comment(); //類前面沒有反斜桿///完全限定名稱,表示絕對于Blog空間//這個調用將被解析成 Blog/Comment();$article_comment = new /Blog/Comment(); //類前面有反斜桿///完全限定名稱,表示絕對于Blog空間//這個調用將被解析成 Blog/Article/Comment();$article_comment = new /Blog/Article/Comment(); //類前面有反斜桿///創建Blog的子空間Articlenamespace Blog/Article;class Comment { }?
鄭重聲明:本文版權歸原作者所有,轉載文章僅為傳播更多信息之目的,如作者信息標記有誤,請第一時間聯系我們修改或刪除,多謝。
新聞熱點
疑難解答