PHP 作用域解析運算符(::)
更新時間:2010年07月27日 19:54:29 作者:
今天 看joomla源碼的時候,才意識到。原來這個操作符還可以訪問類的非靜態(tài)方法啊。真的讓我吃驚不好。一直以為作用域解析運算符只能訪問類的static方法和static成員變量。
Scope Resolution Operator (::)
今天 看joomla源碼的時候,才意識到。原來這個操作符還可以訪問類的非靜態(tài)方法啊。真的讓我吃驚不好。一直以為作用域解析運算符只能訪問類的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();
這段代碼輸入的結(jié)果為:
A test()
#########
B test()
雖然A類中的test()和B類中的test都不是 static方法,但是一樣可以用 “類名::方法名稱(參數(shù)列表)” 的樣式進行正確調(diào)用。他的效果和 new 一個類的實例,然后用這個實例調(diào)用
test方法是一個樣的。
但是,如果我需要在test方法中打印name屬性,直接用::來調(diào)用 會是怎么個情況那.我們首先來修改下 上面的代碼。
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();
上面的代碼運行的結(jié)果 如下:
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();
再運行上面的代碼,結(jié)果如下:
A test() Fatal error: Access to undeclared static property: A::$_name in D:\www\test\scoperefe.php on line 9
哦,原來不能用self 關鍵字訪問當前類的非static方法。
現(xiàn)在,如果想正確的調(diào)用這個方法,有2個做法:
1、首先實例化類,然后用對象調(diào)用就可以直接使用$this->_name進行調(diào)用了;
2、將成員變量$_name設置為static;
上面的問題,相信大家都能夠正確的處理。
其實我真正想說的是:
如果一個方法可以不進行實例化就調(diào)用,那么我們最好把這個方法使用static關鍵字修飾下。在實現(xiàn)方法的時候,只調(diào)用該類的static成員變量。這樣就不會出現(xiàn)上面遇到問題了。
如果一個方法沒有設置為static的方法。那么,最安全的做法還是用實例對象進行調(diào)用更為安全,因為,說不定什么時候就需要修改該方法的實現(xiàn),在修改的時候,說不定就要調(diào)用該類中的
非static成員變量(因為,很大程度上在修改方法的實現(xiàn)的時候,已經(jīng)忘記還有用類名直接調(diào)用這么一說)。
個人愚見。
今天 看joomla源碼的時候,才意識到。原來這個操作符還可以訪問類的非靜態(tài)方法啊。真的讓我吃驚不好。一直以為作用域解析運算符只能訪問類的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();
這段代碼輸入的結(jié)果為:
復制代碼 代碼如下:
A test()
#########
B test()
雖然A類中的test()和B類中的test都不是 static方法,但是一樣可以用 “類名::方法名稱(參數(shù)列表)” 的樣式進行正確調(diào)用。他的效果和 new 一個類的實例,然后用這個實例調(diào)用
test方法是一個樣的。
但是,如果我需要在test方法中打印name屬性,直接用::來調(diào)用 會是怎么個情況那.我們首先來修改下 上面的代碼。
復制代碼 代碼如下:
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();
上面的代碼運行的結(jié)果 如下:
復制代碼 代碼如下:
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();
再運行上面的代碼,結(jié)果如下:
復制代碼 代碼如下:
A test() Fatal error: Access to undeclared static property: A::$_name in D:\www\test\scoperefe.php on line 9
哦,原來不能用self 關鍵字訪問當前類的非static方法。
現(xiàn)在,如果想正確的調(diào)用這個方法,有2個做法:
1、首先實例化類,然后用對象調(diào)用就可以直接使用$this->_name進行調(diào)用了;
2、將成員變量$_name設置為static;
上面的問題,相信大家都能夠正確的處理。
其實我真正想說的是:
如果一個方法可以不進行實例化就調(diào)用,那么我們最好把這個方法使用static關鍵字修飾下。在實現(xiàn)方法的時候,只調(diào)用該類的static成員變量。這樣就不會出現(xiàn)上面遇到問題了。
如果一個方法沒有設置為static的方法。那么,最安全的做法還是用實例對象進行調(diào)用更為安全,因為,說不定什么時候就需要修改該方法的實現(xiàn),在修改的時候,說不定就要調(diào)用該類中的
非static成員變量(因為,很大程度上在修改方法的實現(xiàn)的時候,已經(jīng)忘記還有用類名直接調(diào)用這么一說)。
個人愚見。

