php命名空間可以解決下面兩個問題:
1、用戶撰寫的代碼與php內部的類/函式/常量或第三方類/函式/常量之間的名字沖突
2、為很長的識別符號名稱(通常是為了緩解第一類問題而定義的)創建一個別名(或簡短)的名稱,提高源代碼的可讀性
定義命名空間
通過關鍵字namespace宣告,如果一個檔案中包含命名空間,它必須在其它所有代碼之前宣告命名空間
語法:
namespace myspace;
statement;
可以定義不同的命名空間:
<?php
namespace MyProject {
…
}
namespace AnotherProject {
…
}
?>
這種使用大括號形式的語法更規范,也可以不使用大括號
將全域的非命名空間中的代碼與命名空間中的代碼組合在一起,只能使用大括號形式的語法,全域代碼必須用一個不帶名稱的namespace陳述句加上大括號括起來:
<?php namespace MyProject { … } namespace { // 全域代碼 … } ?>
在宣告命名空間之前唯一合法的代碼是用于定義源檔案編碼方式的declare陳述句,所有非php代碼包括空白符都不能出現在命名空間的宣告之前:
declare(encoding='UTF-8');
php命名空間也允許指定層次化的命名空間的名稱,因此,命名空間的名字可以使用分層次的方式定義:
namespace my\sub\segment;
命名空間中的類名可以通過三種方式參考:
1、非限定名稱,或不包含前綴的類名稱
2、限定名稱,或包含前綴的名稱
3、完全限定名稱,或包含了全域前綴運算子的名稱
比如:
<?php namespace Foo\Bar; include 'file1.php'; const FOO = 2; function foo() {} class foo { static function staticmethod() {} } /* 非限定名稱 */ foo(); // 決議為函式 Foo\Bar\foo foo::staticmethod(); // 決議為類 Foo\Bar\foo ,方法為 staticmethod echo FOO; // 決議為常量 Foo\Bar\FOO /* 限定名稱 */ subnamespace\foo(); // 決議為函式 Foo\Bar\subnamespace\foo subnamespace\foo::staticmethod(); // 決議為類 Foo\Bar\subnamespace\foo, // 以及類的方法 staticmethod echo subnamespace\FOO; // 決議為常量 Foo\Bar\subnamespace\FOO /* 完全限定名稱 */ \Foo\Bar\foo(); // 決議為函式 Foo\Bar\foo \Foo\Bar\foo::staticmethod(); // 決議為類 Foo\Bar\foo, 以及類的方法 staticmethod echo \Foo\Bar\FOO; // 決議為常量 Foo\Bar\FOO ?>
php命名空間的實作受到其語言自身的動態特征的影響,在動態的類名稱、函式名稱或常量名稱中,限定名稱和完全限定名稱沒有區別
__NAMESPACE__魔術常量的值是包含當前命名空間名稱的字串
關鍵字namespace可用來顯式訪問當前命名空間或子命名空間中的元素,它等價于類中的self運算子
比如:
<?php namespace MyProject; use blah\blah as mine; // 引入了 blah\blah 命名空間,并定義了個別名mine mine\mine(); // 呼叫函式 blah\blah\mine() namespace\blah\mine(); // 呼叫函式 MyProject\blah\mine() namespace\func(); // 呼叫函式 MyProject\func() namespace\sub\func(); // 呼叫函式 MyProject\sub\func() namespace\cname::method(); // 呼叫 MyProject\cname 類的靜態方法 $a = new namespace\sub\cname(); // 實體化 MyProject\sub\cname 類的物件 $b = namespace\CONSTANT; // 將常量 MyProject\CONSTANT 的值賦給 $b ?>
使用命名空間:別名/匯入
比如:
<?php use My\Full\Classname as Another, My\Full\NSname; // 使用use運算子匯入/使用別名 $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 遇到一個非限定的類、函式或常量名稱時,它使用不同的優先策略來決議該名稱,類名稱總是決議到當前命名空間中的名稱,因此在訪問系統內部或不包含在命名空間中的類名稱時,必須使用完全限定名稱
如果沒有定義任何命名空間,所有的類與函式的定義都是在全域空間,與php引入命名空間概念前一樣,在名稱前加上前綴\表示該名稱是全域空間中的名稱,即使該名稱位于其它的命名空間中時也是如此
命名空間具有順序
名稱決議遵循下列規則:
1、對完全限定名稱的函式,類和常量的呼叫在編譯時決議,例如 new \A\B 決議為類 A\B,
2、所有的非限定名稱和限定名稱(非完全限定名稱)根據當前的匯入規則在編譯時進行轉換,例如,如果命名空間 A\B\C 被匯入為 C,那么對 C\D\e() 的呼叫就會被轉換為 A\B\C\D\e(),
3、在命名空間內部,所有的沒有根據匯入規則轉換的限定名稱均會在其前面加上當前的命名空間名稱,例如,在命名空間 A\B 內部呼叫 C\D\e(),則 C\D\e() 會被轉換為 A\B\C\D\e() ,
4、非限定類名根據當前的匯入規則在編譯時轉換(用全名代替短的匯入名稱),例如,如果命名空間 A\B\C 匯入為C,則 new C() 被轉換為 new A\B\C() ,
5、在命名空間內部(例如A\B),對非限定名稱的函式呼叫是在運行時決議的,例如對函式 foo() 的呼叫是這樣決議的:
5.1、在當前命名空間中查找名為 A\Boo() 的函式
5.2、嘗試查找并呼叫 全域(global) 空間中的函式 foo(),
6、在命名空間(例如A\B)內部對非限定名稱或限定名稱類(非完全限定名稱)的呼叫是在運行時決議的,下面是呼叫 new C() 及 new D\E() 的決議程序: new C()的決議:
6.1、在當前命名空間中查找A\B\C類,
6.2、嘗試自動裝載類A\B\C,
new D\E()的決議:
在類名稱前面加上當前命名空間名稱變成:A\B\D\E,然后查找該類,
嘗試自動裝載類 A\B\D\E,
為了參考全域命名空間中的全域類,必須使用完全限定名稱 new \C(),
轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/477640.html
標籤:PHP
