PHP命名空間(Namespace)的使用詳解

對於命名空間,官方文檔已經說得很詳細[查看],我在這裏作了一下實踐和總結。php

命名空間一個最明確的目的就是解決重名問題,PHP中不容許兩個函數或者類出現相同的名字,不然會產生一個致命的錯誤。這種狀況下只要避免命名重複就能夠解決,最多見的一種作法是約定一個前綴。html

例:項目中有兩個模塊:article和message board,它們各自有一個處理用戶留言的類Comment。以後我可能想要增長對全部用戶留言的一些信息統計功能,好比說我想獲得全部留言的數量。這時候調用它們Comment提供的方法是很好的作法,可是同時引入各自的Comment類顯然是不行的,代碼會出錯,在另外一個地方重寫任何一個Comment也會下降維護性。那這時只能重構類名,我約定了一個命名規則,在類名前面加上模塊名,像這樣:Article_Comment、MessageBoard_Comment函數

能夠看到,名字變得很長,那意味着之後使用Comment的時候會寫上更多的代碼(至少字符多了)。而且,之後若是要對各個模塊增長更多的一些整合功能,或者是互相調用,發生重名的時候就須要重構名字。固然在項目開始的時候就注意到這個問題,並規定命名規則就能很好的避免這個問題。另外一個解決方法能夠考慮使用命名空間。spa


註明:插件

本文提到的常量:PHP5.3開始const關鍵字能夠用在類的外部。const和define都是用來聲明常量的(它們的區別不詳述),可是在命名空間裏,define的做用是全局的,而const則做用於當前空間。我在文中提到的常量是指使用const聲明的常量。htm


基礎blog

命名空間將代碼劃分出不一樣的空間(區域),每一個空間的常量、函數、類(爲了偷懶,我下邊都將它們稱爲元素)的名字互不影響, 這個有點相似咱們經常提到的‘封裝'的概念。文檔

建立一個命名空間須要使用namespace關鍵字,這樣:字符串

複製代碼 代碼以下:get


<?php

//建立一個名爲'Article'的命名空間

namespace Article;

?>


要注意的是,當前腳本文件的第一個命名空間前面不能有任何代碼,下面的寫法都是錯誤的:

複製代碼 代碼以下:


//例一

//在腳本前面寫了一些邏輯代碼

<?php

$path = "/";

class Comment { }

namespace Article;

?>

 

//例二

//在腳本前面輸出了一些字符

<html></html>

<?php

namespace Article;

?>


爲何要說第一個命名空間呢?由於同一腳本文件中能夠建立多個命名空間。

下面我建立了兩個命名空間,順便爲這兩個空間各自添加了一個Comment類元素:

複製代碼 代碼以下:


<?php

//建立一個名爲'Article'的命名空間

namespace Article;

//此Comment屬於Article空間的元素

class Comment { }


//建立一個名爲'MessageBoard'的命名空間

namespace MessageBoard;

//此Comment屬於MessageBoard空間的元素

class Comment { }

?>


在不一樣空間之間不能夠直接調用其它元素,須要使用命名空間的語法:

複製代碼 代碼以下:


<?php

namespace Article;

class Comment { }


namespace MessageBoard;

class Comment { }

//調用當前空間(MessageBoard)的Comment類

$comment = new Comment();

//調用Article空間的Comment類

$article_comment = new \Article\Comment();

?>


能夠看到,在MessageBoard空間中調用article空間裏的Comment類時,使用了一種像文件路徑的語法: \空間名\元素名

除了類以外,對函數和常量的用法是同樣的,下面我爲兩個空間建立了新的元素,並在MessageBoard空間中輸出了它們的值。

複製代碼 代碼以下:


<?php

namespace Article;

const PATH = '/article';

function getCommentTotal() {

    return 100;

}

class Comment { }


namespace MessageBoard;

const PATH = '/message_board';

function getCommentTotal() {

    return 300;

}

class Comment { }

//調用當前空間的常量、函數和類

echo PATH; ///message_board

echo getCommentTotal(); //300

$comment = new Comment();

//調用Article空間的常量、函數和類

echo \Article\PATH; ///article

echo \Article\getCommentTotal(); //100

$article_comment = new \Article\Comment();

?>


而後個人確獲得了Article空間的元素數據。


子空間

命名空間的調用語法像文件路徑同樣是有道理的,它容許咱們自定義子空間來描述各個空間之間的關係。

抱歉我忘了說,article和message board這兩個模塊其實都是處於同一個blog項目內。若是用命名空間來表達它們的關係,是這樣:

複製代碼 代碼以下:


<?php

//我用這樣的命名空間表示處於blog下的article模塊

namespace Blog\Article;

class Comment { }


//我用這樣的命名空間表示處於blog下的message board模塊

namespace Blog\MessageBoard;

class Comment { }

//調用當前空間的類

$comment = new Comment();

//調用Blog\Article空間的類

$article_comment = new \Blog\Article\Comment();

?>


並且,子空間還能夠定義不少層次,好比說 Blog\Article\Archives\Date


公共空間

我有一個common_inc.php腳本文件,裏面有一些好用的函數和類:

複製代碼 代碼以下:


<?php

function getIP() { }

class FilterXSS { }

?>


在一個命名空間裏引入這個腳本,腳本里的元素不會歸屬到這個命名空間。若是這個腳本里沒有定義其它命名空間,它的元素就始終處於公共空間中:

複製代碼 代碼以下:


<?php

namespace Blog\Article;

//引入腳本文件

include './common_inc.php';

$filter_XSS = new FilterXSS(); //出現致命錯誤:找不到Blog\Article\FilterXSS類

$filter_XSS = new \FilterXSS(); //正確

?>


調用公共空間的方式是直接在元素名稱前加 \ 就能夠了,不然PHP解析器會認爲我想調用當前空間下的元素。除了自定義的元素,還包括PHP自帶的元素,都屬於公共空間。

要提一下,其實公共空間的函數和常量不用加 \ 也能夠正常調用(不明白PHP爲何要這樣作),可是爲了正確區分元素,仍是建議調用函數的時候加上 \


名稱術語

在說別名和導入以前,須要知道關於空間三種名稱的術語,以及PHP是怎樣解析它們的。官方文檔說得很是好,我就直接拿來套了。

1.非限定名稱,或不包含前綴的類名稱,例如 $comment = new Comment();。若是當前命名空間是Blog\Article,Comment將被解析爲Blog\Article\Comment。若是使用Comment的代碼不包含在任何命名空間中的代碼(全局空間中),則Comment會被解析爲Comment。


2.限定名稱,或包含前綴的名稱,例如 $comment = new Article\Comment();。若是當前的命名空間是Blog,則Comment會被解析爲Blog\Article\Comment。若是使用Comment的代碼不包含在任何命名空間中的代碼(全局空間中),則Comment會被解析爲Comment。


3.徹底限定名稱,或包含了全局前綴操做符的名稱,例如 $comment = new \Article\Comment();。在這種狀況下,Comment老是被解析爲代碼中的文字名(literal name)Article\Comment。

 

其實能夠把這三種名稱類比爲文件名(例如 comment.php)、相對路徑名(例如 ./article/comment.php)、絕對路徑名(例如 /blog/article/comment.php),這樣可能會更容易理解。

我用了幾個示例來表示它們:

複製代碼 代碼以下:


<?php

//建立空間Blog

namespace 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的子空間Article

namespace Blog\Article;

class Comment { }

?>


其實以前我就一直在使用非限定名稱和徹底限定名稱,如今它們終於能夠叫出它們的名稱了。


別名和導入

別名和導入能夠看做是調用命名空間元素的一種快捷方式。PHP並不支持導入函數或常量。

它們都是經過使用use操做符來實現:

複製代碼 代碼以下:


<?php

namespace Blog\Article;

class Comment { }


//建立一個BBS空間(我有打算開個論壇)

namespace BBS;

//導入一個命名空間

use Blog\Article;

//導入命名空間後可以使用限定名稱調用元素

$article_comment = new Article\Comment();

//爲命名空間使用別名

use Blog\Article as Arte;

//使用別名代替空間名

$article_comment = new Arte\Comment();

//導入一個類

use Blog\Article\Comment;

//導入類後可以使用非限定名稱調用元素

$article_comment = new Comment();

//爲類使用別名

use Blog\Article\Comment as Comt;

//使用別名代替空間名

$article_comment = new Comt();

?>


我注意到,若是導入元素的時候,當前空間有相同的名字元素將會怎樣?顯然結果會發生致命錯誤。

例:

複製代碼 代碼以下:


<?php

namespace Blog\Article;

class Comment { }


namespace BBS;

class Comment { }

Class Comt { }


//導入一個類

use Blog\Article\Comment;

$article_comment = new Comment(); //與當前空間的Comment發生衝突,程序產生致命錯誤

//爲類使用別名

use Blog\Article\Comment as Comt;

$article_comment = new Comt(); //與當前空間的Comt發生衝突,程序產生致命錯誤

?>


動態調用

PHP提供了namespace關鍵字和__NAMESPACE__魔法常量動態的訪問元素,__NAMESPACE__能夠經過組合字符串的形式來動態訪問:

複製代碼 代碼以下:


<?php

namespace Blog\Article;

const PATH = '/Blog/article';

class Comment { }


//namespace關鍵字表示當前空間

echo namespace\PATH; ///Blog/article

$comment = new namespace\Comment();

//魔法常量__NAMESPACE__的值是當前空間名稱

echo __NAMESPACE__; //Blog\Article

//能夠組合成字符串並調用

$comment_class_name = __NAMESPACE__ . '\Comment';

$comment = new $comment_class_name();

?>


字符串形式調用問題


上面的動態調用的例子中,咱們看到了字符串形式的動態調用方式,若是要使用這種方式要注意兩個問題。

1. 使用雙引號的時候特殊字符可能被轉義

複製代碼 代碼以下:


<?php

namespace Blog\Article;

class name { }

//我是想調用Blog\Article\name

$class_name = __NAMESPACE__ . "\name"; //可是\n將被轉義爲換行符

$name = new $class_name(); //發生致命錯誤

?>


2. 不會認爲是限定名稱

PHP在編譯腳本的時候就肯定了元素所在的空間,以及導入的狀況。而在解析腳本時字符串形式調用只能認爲是非限定名稱和徹底限定名稱,而永遠不多是限定名稱。

複製代碼 代碼以下:


<?php

namespace Blog;

//導入Common類

use Blog\Article\Common;

//我想使用非限定名稱調用Blog\Article\Common

$common_class_name = 'Common';

//實際會被看成非限定名稱,也就表示當前空間的Common類,但我當前類沒有建立Common類

$common = new $common_class_name(); //發生致命錯誤:Common類不存在

//我想使用限定名稱調用Blog\Article\Common

$common_class_name = 'Article\Common';

//實際會被看成徹底限定名稱,也就表示Article空間下的Common類,但我下面只定義了Blog\Article空間而不是Article空間

$common = new $common_class_name(); //發生致命錯誤:Article\Common類不存在


namespace Blog\Article;

class Common { }

?>


總結

我對PHP的命名空間剛剛接觸,也不能隨便給一些沒有實踐的建議。我我的認爲命名空間的做用和功能都很強大,若是要寫插件或者通用庫的時候不再用擔憂重名問題。不過若是項目進行到必定程度,要經過增長命名空間去解決重名問題,我以爲工做量不會比重構名字少。也不得不認可它的語法會對項目增長必定的複雜度,所以從項目一開始的時候就應該很好的規劃它,並制定一個命名規範。

相關文章
相關標籤/搜索