php中self與static的區別
阿新 • • 發佈:2018-12-02
1、假設我們有個Car類
,它有2個方法:model()
和getModel()
。
class Car{
public function model(){
//這裡我們使用了關鍵字self
self::getModel();
}
protected function getModel(){
echo 'I am car';
}
}
例項化後呼叫方法:
$car = new Car();
$car->model(); // 輸出:I am car
關鍵字self
使得我們呼叫了Car類
的getModel()
2、新增一個新類作為Car類
的子類:
class Mercedes extends Car
{
protected function getModel()
{
echo "I am mercedes";
}
}
// 例項化後 呼叫model()方法
$mercedes = new Mercedes();
$mercedes->model();
我們知道子類會繼承父類的方法,我們在子類Mercedes
中重寫了getModel()
方法。
這時候例項化Mercedes類
呼叫model()
方法,會輸出字串"I am car" 還是“I am mercedes”?
你可能會覺得結果是:I am mercedes
。
但實際輸出是:
I am car
why?
3、關鍵字self
的工作原理是:它會呼叫當前類的方法。
因為model()
方法只在Car類
中定義的,所以對self
來說當前類就是Car類
。self::getModel()
就是呼叫Car類
中的getModel
方法。
這似乎不是我們想要的,如何解決呢?
4、解決方案一:在Mercedes類
中覆蓋model()
方法
class Mercedes extends Car { public function model(){ //這裡我們使用了關鍵字self self::getModel(); } protected function getModel() { echo "I am mercedes"; } } // 例項化後 呼叫model()方法 $mercedes = new Mercedes(); $mercedes->model(); // 輸出:I am mercedes
但這肯定不是好辦法啊,方法都重寫了,還繼承個毛啊。
5、解決方案二:把self
關鍵字換成static
class Car{
public function model(){
//這裡我們使用了關鍵字static
static::getModel();
}
protected function getModel(){
echo 'I am car';
}
}
我們只是將Car類
中的self
替換成了static
,並未對Mercedes類
作修改。
此時我們呼叫:
$mercedes = new Mercedes();
$mercedes->model(); // 輸出:I am mercedes
6、小結
在php5.3中加入了一個新特性,叫做延遲靜態繫結
。可以幫我們實現多型。
簡單說,延遲靜態繫結意味著,當我們使用static
關鍵字呼叫一個繼承方法時,它將在執行時
才繫結呼叫類。