欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  后端开发

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

程序员文章站 2022-03-28 10:41:11
...
先上代码,非常简单。
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";
    }
}

这里有两个疑问:

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

  2. 父类里的构造方法调用_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";
    }
}

这里有两个疑问:

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

  2. 父类里的构造方法调用_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++里这叫虚函数

相关标签: php