先上代码,非常简单。
class Controller { public function __construct() { if (method_exists($this, '_initialize')) { $this->_initialize(); } } // 父类的_initialize,我不会被执行到 public function _initialize() { echo "I am executed by myself"; } } Class BaseController extends Controller { public function __construct() { parent::__construct(); } // 子类的_initialize public function _initialize() { echo "I am executed by parent"; } }
这里有两个疑问:
父类怎么可以调用子类的方法,好奇怪啊。
父类里的构造方法调用_initialize
怎么调用的是子类的方法,为什么。
先上代码,非常简单。
class Controller { public function __construct() { if (method_exists($this, '_initialize')) { $this->_initialize(); } } // 父类的_initialize,我不会被执行到 public function _initialize() { echo "I am executed by myself"; } } Class BaseController extends Controller { public function __construct() { parent::__construct(); } // 子类的_initialize public function _initialize() { echo "I am executed by parent"; } }
这里有两个疑问:
父类怎么可以调用子类的方法,好奇怪啊。
父类里的构造方法调用_initialize
怎么调用的是子类的方法,为什么。
可以的,你的2个问题归结到一个原因,就是父类的__construct里,使用的$this实际上是子类对象,而不是父类对象。
public function __construct() { if (method_exists($this, '_initialize')) { print(get_class($this)); //这里可以看出实际绑定的时候,this指向的是子类 $this->_initialize(); } }
这个完全看你是怎么new
了吧,你在父类的构造函数里面用的是$this
的_initialize
方法,所以调用哪个_initialize
完全取决于$this
:
$parent = new Controller(); // I am executed by myself $child = new BaseController(); // I am executed by parent
extends 是php中最基本的继承的概念,可以这么理解。
子类继承了父类之后,如果子类中有和父类同名的方法(包括构造函数),那么调用同名方法时(在子类中调用),全部执行子类中的方法。这是PHP语法就这么定的,你只要理解就好。
希望能帮助到你。
这其实就是能否深入理解$this
关键字。
$this
是一个指向它直接所属对象的引用,通常是$this
所从属的对象,但也可能是另一个对象。所以,在子对象中,$this
还是指向子对象本身而不会指向父对象。在你的代码中,父对象使用了$this
并通过它执行自身特定的方法,这很好理解。但当这个对象被继承时,子对象的$this
就会指向子对象,如果在子对象中调用了父对象的方法,而这个方法中使用了$this
关键字,那么你调用的父对象的方法中的$this
依然是指向子对象的,看起来就像是父对象调用了子对象的方法,但其实是父对象的代码被继承了。
顺带提一下,另一个$this
常用的地方是实现“链式操作”。
class A { function doSomething() { return $this; } function doSomeOtherthing() { return $this; } } $a = new A; $a->doSomething()->doSomeOtherthing();
楼上的几人点出了重点 你在basecontroller
中调用了父类的构造方法,父类构造方法中调用了初始化方法,注意这里的$this
其实是basecontroller对象
所以当basecontroller
中定义了新的初始化方法(这个方法覆盖了父类的初始化方法)后$this->_initialize()
就是调用了basecontroller对象
中的这个方法
你的意思是酱紫的,再补充下代码,new一个BaseController的对象。
$obj = new BaseController;
具体按流程走吧~(以下子类为BaseController,父类为Controller)
1.new BaseController,此时触发子类的构造方法
2.子类构造方法调用父类的构造方法,判断$this是否存在_initialize()
3.高潮来了,而由于此时$this是子类的映射对象,所以$this->_initialize();调用的是子类的_initialize()
具体,在父类的构造方法中,打印一下$this就可以明了。
public function __construct() { if (method_exists($this, '_initialize')) { var_dump($this); $this->_initialize(); } }
面向对象的“多态”概念你理解不?
$this指代函数调用对象,你是在子类中调用的所以执行子类的初始化方法
同意@jock 的观点
1、首先是 $this,parent,self 指针问题,问主可以参考这篇PHP中this,self,parent的区别,$this指的是实例化对象,而不是类
2、这个语法设计实际是为了解决继承时需要扩展父类方法的需求,相当于部分复用了父类的方法。(我觉得PHP这么设计其实是有一定歧义的,感觉是使用了静态方法,不过怎么说呢,习惯了就好了,问主要是想到了更好的写法也可以给PHP提意见)
3、java用的是super()方法
又看到几篇可能帮助理解的:
PHP 面向对象:parent::关键字
PHP,非静态方法调用静态方法,
c++里这叫虚函数