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

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

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

先上代码,非常简单。

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++里这叫虚函数

人气教程排行