Scope Resolution Operator (::)
今天 看joomla源碼的時候,才意識到。原來這個操作符還可以訪問類的非靜態方法啊。真的讓我吃驚不好。一直以為作用域解析運算符只能訪問類的static方法和static成員變量。
如果各位不相信,下面有個簡單的小測試代碼可以證明這個。
復制代碼 代碼如下:
class A{
private $_name = 'A';
function __construct(){
echo 'A construct <br />';
}
function test(){
echo 'A test() <br />';
}
}
class B extends A{
private $_name = 'B';
function __construct(){
parent::__construct();
echo 'B construct <br />';
}
function test(){
echo 'B test()';
}
}
A::test();
echo '######### <br />';
B::test();
這段代碼輸入的結果為:
復制代碼 代碼如下:
A test()
#########
B test()
雖然A類中的test()和B類中的test都不是 static方法,但是一樣可以用 “類名::方法名稱(參數列表)” 的樣式進行正確調用。他的效果和 new 一個類的實例,然後用這個實例調用
test方法是一個樣的。
但是,如果我需要在test方法中打印name屬性,直接用::來調用 會是怎麼個情況那.我們首先來修改下 上面的代碼。
復制代碼 代碼如下:
class A{
private $_name = 'A';
function __construct(){
echo 'A construct <br />';
}
function test(){
echo 'A test() <br />', $this->$_name,'<br />';
}
}
class B extends A{
private $_name = 'B';
function __construct(){
parent::__construct();
echo 'B construct <br />';
}
function test(){
echo 'B test()', $this->_name,'<br />';
}
}
A::test();
echo '######### <br />';
B::test();
上面的代碼運行的結果 如下:
復制代碼 代碼如下:
Fatal error: Using $this when not in object context in D:\www\test\scoperefe.php on line 9
[html]
那有的朋友就說了。你壓根就沒有實例化類A,當然不能直接用$this->_name的方式來訪問成員變量$_name了,那麼,是不是修改成self::$_name就行了哪?
說干就干,下面把上面的代碼修改下
[code]
class A{
private $_name = 'A';
function __construct(){
echo 'A construct <br />';
}
function test(){
echo 'A test() <br />', self::$_name,'<br />';
}
}
class B extends A{
private $_name = 'B';
function __construct(){
parent::__construct();
echo 'B construct <br />';
}
function test(){
echo 'B test()', $this->_name,'<br />';
}
}
A::test();
echo '######### <br />';
B::test();
再運行上面的代碼,結果如下:
復制代碼 代碼如下:
A test() Fatal error: Access to undeclared static property: A::$_name in D:\www\test\scoperefe.php on line 9
哦,原來不能用self 關鍵字訪問當前類的非static方法。
現在,如果想正確的調用這個方法,有2個做法:
1、首先實例化類,然後用對象調用就可以直接使用$this->_name進行調用了;
2、將成員變量$_name設置為static;
上面的問題,相信大家都能夠正確的處理。
其實我真正想說的是:
如果一個方法可以不進行實例化就調用,那麼我們最好把這個方法使用static關鍵字修飾下。在實現方法的時候,只調用該類的static成員變量。這樣就不會出現上面遇到問題了。
如果一個方法沒有設置為static的方法。那麼,最安全的做法還是用實例對象進行調用更為安全,因為,說不定什麼時候就需要修改該方法的實現,在修改的時候,說不定就要調用該類中的
非static成員變量(因為,很大程度上在修改方法的實現的時候,已經忘記還有用類名直接調用這麼一說)。
個人愚見。