1. 程式人生 > >php中self與static的區別

php中self與static的區別

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()

方法,輸出了文字“I am car”。

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關鍵字呼叫一個繼承方法時,它將在執行時才繫結呼叫類