對于命名空間,官方文檔已經說得很詳細[],我在這里做了一下實踐和總結。
命名空間一個最明確的目的就是解決重名問題,PHP中不允許兩個函數或者類出現相同的名字,否則會產生一個致命的錯誤。這種情況下只要避免命名重復就可以解決,最常見的一種做法是約定一個前綴。
例:項目中有兩個模塊:article和message board,它們各自有一個處理用戶留言的類Comment。之后我可能想要增加對所有用戶留言的一些信息統計功能,比如說我想得到所有留言的數量。這時候調用它們Comment提供的方法是很好的做法,但是同時引入各自的Comment類顯然是不行的,代碼會出錯,在另一個地方重寫任何一個Comment也會降低維護性。那這時只能重構類名,我約定了一個命名規則,在類名前面加上模塊名,像這樣:Article_Comment、MessageBoard_Comment
可以看到,名字變得很長,那意味著以后使用Comment的時候會寫上更多的代碼(至少字符多了)。并且,以后如果要對各個模塊增加更多的一些整合功能,或者是互相調用,發生重名的時候就需要重構名字。當然在項目開始的時候就注意到這個問題,并規定命名規則就能很好的避免這個問題。另一個解決方法可以考慮使用命名空間。
注明:
本文提到的常量:PHP5.3開始const關鍵字可以用在類的外部。const和define都是用來聲明常量的(它們的區別不詳述),但是在命名空間里,define的作用是全局的,而const則作用于當前空間。我在文中提到的常量是指使用const聲明的常量。
基礎
命名空間將代碼劃分出不同的空間(區域),每個空間的常量、函數、類(為了偷懶,我下邊都將它們稱為元素)的名字互不影響, 這個有點類似我們常常提到的‘封裝'的概念。
創建一個命名空間需要使用namespace關鍵字,這樣:
<?php
$path = "/";
class Comment { }
namespace Article;
?>
//例二
//在腳本前面輸出了一些字符
<html></html>
<?php
namespace Article;
?>
下面我創建了兩個命名空間,順便為這兩個空間各自添加了一個Comment類元素:
//創建一個名為'Article'的命名空間
namespace Article;
//此Comment屬于Article空間的元素
class Comment { }
//創建一個名為'MessageBoard'的命名空間
namespace MessageBoard;
//此Comment屬于MessageBoard空間的元素
class Comment { }
?>
namespace Article;
class Comment { }
namespace MessageBoard;
class Comment { }
//調用當前空間(MessageBoard)的Comment類
$comment = new Comment();
//調用Article空間的Comment類
$article_comment = new /Article/Comment();
?>
除了類之外,對函數和常量的用法是一樣的,下面我為兩個空間創建了新的元素,并在MessageBoard空間中輸出了它們的值。
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和message board這兩個模塊其實都是處于同一個blog項目內。如果用命名空間來表達它們的關系,是這樣:
//我用這樣的命名空間表示處于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();
?>
公共空間
我有一個common_inc.php腳本文件,里面有一些好用的函數和類:
function getIP() { }
class FilterXSS { }
?>
namespace Blog/Article;
//引入腳本文件
include './common_inc.php';
$filter_XSS = new FilterXSS(); //出現致命錯誤:找不到Blog/Article/FilterXSS類
$filter_XSS = new /FilterXSS(); //正確
?>
要提一下,其實公共空間的函數和常量不用加 / 也可以正常調用(不明白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),這樣可能會更容易理解。
我用了幾個示例來表示它們:
//創建空間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操作符來實現:
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();
?>
例:
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發生沖突,程序產生致命錯誤
?>
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. 使用雙引號的時候特殊字符可能被轉義
namespace Blog/Article;
class name { }
//我是想調用Blog/Article/name
$class_name = __NAMESPACE__ . "/name"; //但是/n將被轉義為換行符
$name = new $class_name(); //發生致命錯誤
?>
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 { }
?>
新聞熱點
疑難解答