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
关键字调用一个继承方法时,它将在运行时
才绑定调用类。