当前位置:Gxlcms > PHP教程 > php的语法问题,这是怎么回事。

php的语法问题,这是怎么回事。

时间:2021-07-01 10:21:17 帮助过:7人阅读

先上代码,非常简单。

  1. <code class="php">class Controller {
  2. public function __construct() {
  3. if (method_exists($this, '_initialize')) {
  4. $this->_initialize();
  5. }
  6. }
  7. // 父类的_initialize,我不会被执行到
  8. public function _initialize() {
  9. echo "I am executed by myself";
  10. }
  11. }
  12. Class BaseController extends Controller {
  13. public function __construct() {
  14. parent::__construct();
  15. }
  16. // 子类的_initialize
  17. public function _initialize() {
  18. echo "I am executed by parent";
  19. }
  20. }
  21. </code>

这里有两个疑问:

  1. 父类怎么可以调用子类的方法,好奇怪啊。

  2. 父类里的构造方法调用_initialize怎么调用的是子类的方法,为什么。

回复内容:

先上代码,非常简单。

  1. <code class="php">class Controller {
  2. public function __construct() {
  3. if (method_exists($this, '_initialize')) {
  4. $this->_initialize();
  5. }
  6. }
  7. // 父类的_initialize,我不会被执行到
  8. public function _initialize() {
  9. echo "I am executed by myself";
  10. }
  11. }
  12. Class BaseController extends Controller {
  13. public function __construct() {
  14. parent::__construct();
  15. }
  16. // 子类的_initialize
  17. public function _initialize() {
  18. echo "I am executed by parent";
  19. }
  20. }
  21. </code>

这里有两个疑问:

  1. 父类怎么可以调用子类的方法,好奇怪啊。

  2. 父类里的构造方法调用_initialize怎么调用的是子类的方法,为什么。

可以的,你的2个问题归结到一个原因,就是父类的__construct里,使用的$this实际上是子类对象,而不是父类对象。

  1. <code>public function __construct() {
  2. if (method_exists($this, '_initialize')) {
  3. print(get_class($this)); //这里可以看出实际绑定的时候,this指向的是子类
  4. $this->_initialize();
  5. }
  6. }</code>

这个完全看你是怎么new了吧,你在父类的构造函数里面用的是$this_initialize方法,所以调用哪个_initialize完全取决于$this

  1. <code>
  2. $parent = new Controller(); // I am executed by myself
  3. $child = new BaseController(); // I am executed by parent
  4. </code>

extends 是php中最基本的继承的概念,可以这么理解。
子类继承了父类之后,如果子类中有和父类同名的方法(包括构造函数),那么调用同名方法时(在子类中调用),全部执行子类中的方法。这是PHP语法就这么定的,你只要理解就好。
希望能帮助到你。

这其实就是能否深入理解$this关键字。

$this是一个指向它直接所属对象的引用,通常是$this所从属的对象,但也可能是另一个对象。所以,在子对象中,$this还是指向子对象本身而不会指向父对象。在你的代码中,父对象使用了$this并通过它执行自身特定的方法,这很好理解。但当这个对象被继承时,子对象的$this就会指向子对象,如果在子对象中调用了父对象的方法,而这个方法中使用了$this关键字,那么你调用的父对象的方法中的$this依然是指向子对象的,看起来就像是父对象调用了子对象的方法,但其实是父对象的代码被继承了。

顺带提一下,另一个$this常用的地方是实现“链式操作”。

  1. <code>class A
  2. {
  3. function doSomething()
  4. {
  5. return $this;
  6. }
  7. function doSomeOtherthing()
  8. {
  9. return $this;
  10. }
  11. }
  12. $a = new A;
  13. $a->doSomething()->doSomeOtherthing();</code>

楼上的几人点出了重点 你在basecontroller中调用了父类的构造方法,父类构造方法中调用了初始化方法,注意这里的$this 其实是basecontroller对象 所以当basecontroller中定义了新的初始化方法(这个方法覆盖了父类的初始化方法)后$this->_initialize()就是调用了basecontroller对象中的这个方法

你的意思是酱紫的,再补充下代码,new一个BaseController的对象。

  1. <code>$obj = new BaseController;</code>

具体按流程走吧~(以下子类为BaseController,父类为Controller)
1.new BaseController,此时触发子类的构造方法
2.子类构造方法调用父类的构造方法,判断$this是否存在 _initialize()
3.高潮来了,而由于此时$this是子类的映射对象,所以 $this->_initialize();调用的是子类的 _initialize()


具体,在父类的构造方法中,打印一下$this就可以明了。

  1. <code>public function __construct() {
  2. if (method_exists($this, '_initialize')) {
  3. var_dump($this);
  4. $this->_initialize();
  5. }
  6. }</code>

面向对象的“多态”概念你理解不?

$this指代函数调用对象,你是在子类中调用的所以执行子类的初始化方法

同意@jock 的观点

1、首先是 $this,parent,self 指针问题,问主可以参考这篇PHP中this,self,parent的区别,$this指的是实例化对象,而不是类
2、这个语法设计实际是为了解决继承时需要扩展父类方法的需求,相当于部分复用了父类的方法。(我觉得PHP这么设计其实是有一定歧义的,感觉是使用了静态方法,不过怎么说呢,习惯了就好了,问主要是想到了更好的写法也可以给PHP提意见)
3、java用的是super()方法

又看到几篇可能帮助理解的:

PHP 面向对象:parent::关键字
PHP,非静态方法调用静态方法,

c++里这叫虚函数

人气教程排行