How do I get a PHP class constructor to call its parent's parent's constructor?

前端 未结 15 1284
滥情空心
滥情空心 2020-11-28 21:38

I need to have a class constructor in PHP call its parent\'s parent\'s (grandparent?) constructor without calling the parent constructor.

//         


        
相关标签:
15条回答
  • 2020-11-28 21:58

    You must use Grandpa::__construct(), there's no other shortcut for it. Also, this ruins the encapsulation of the Papa class - when reading or working on Papa, it should be safe to assume that the __construct() method will be called during construction, but the Kiddo class does not do this.

    0 讨论(0)
  • 2020-11-28 22:01

    I agree with "too much php", try this:

    class Grandpa 
    {
        public function __construct()
        {
            echo 'Grandpa<br/>';
        }
    
    }
    
    class Papa extends Grandpa
    {
        public function __construct()
        {
            echo 'Papa<br/>';
            parent::__construct();
        }
    }
    
    class Kiddo extends Papa
    {
        public function __construct()
        {
            // THIS IS WHERE I NEED TO CALL GRANDPA'S
            // CONSTRUCTOR AND NOT PAPA'S
            echo 'Kiddo<br/>';
            Grandpa::__construct();
        }
    }
    
    $instance = new Kiddo;
    

    I got the result as expected:

    Kiddo

    Grandpa

    This is a feature not a bug, check this for your reference:

    https://bugs.php.net/bug.php?id=42016

    It is just the way it works. If it sees it is coming from the right context this call version does not enforce a static call.

    Instead it will simply keep $this and be happy with it.

    parent::method() works in the same way, you don't have to define the method as static but it can be called in the same context. Try this out for more interesting:

    class Grandpa 
    {
        public function __construct()
        {
            echo 'Grandpa<br/>';
            Kiddo::hello();
        }
    
    }
    
    class Papa extends Grandpa
    {
        public function __construct()
        {
            echo 'Papa<br/>';
            parent::__construct();
        }
    }
    
    class Kiddo extends Papa
    {
        public function __construct()
        {
            // THIS IS WHERE I NEED TO CALL GRANDPA'S
            // CONSTRUCTOR AND NOT PAPA'S
            echo 'Kiddo<br/>';
            Grandpa::__construct();
        }
    
        public function hello()
        {
            echo 'Hello<br/>';
        }
    }
    
    $instance = new Kiddo;
    

    It also works as expected:

    Kiddo

    Grandpa

    Hello

    But if you try to initialize a new Papa, you will get an E_STRICT error:

    $papa = new Papa;
    

    Strict standards: Non-static method Kiddo::hello() should not be called statically, assuming $this from incompatible context

    You can use instanceof to determine if you can call a Children::method() in a parent method:

    if ($this instanceof Kiddo) Kiddo::hello();
    
    0 讨论(0)
  • 2020-11-28 22:01

    You can call Grandpa::__construct from where you want and the $this keyword will refer to your current class instance. But be carefull with this method you cannot access to protected properties and methods of current instance from this other context, only to public elements. => All work and officialy supported.

    Example

    // main class that everything inherits
    class Grandpa 
    {
        public function __construct()
        {
            echo $this->one; // will print 1
            echo $this->two; // error cannot access protected property
        }
    
    }
    
    class Papa extends Grandpa
    {
        public function __construct()
        {
            // call Grandpa's constructor
            parent::__construct();
        }
    }
    
    class Kiddo extends Papa
    {
        public $one = 1;
        protected $two = 2;
        public function __construct()
        {
            Grandpa::__construct();
        }
    }
    
    new Kiddo();
    
    0 讨论(0)
  • 2020-11-28 22:02

    There's an easier solution for this, but it requires that you know exactly how much inheritance your current class has gone through. Fortunately, get_parent_class()'s arguments allow your class array member to be the class name as a string as well as an instance itself.

    Bear in mind that this also inherently relies on calling a class' __construct() method statically, though within the instanced scope of an inheriting object the difference in this particular case is negligible (ah, PHP).

    Consider the following:

    class Foo {
        var $f = 'bad (Foo)';
    
        function __construct() {
            $this->f = 'Good!';
        }
    }
    
    class Bar extends Foo {
        var $f = 'bad (Bar)';
    }
    
    class FooBar extends Bar {
        var $f = 'bad (FooBar)';
    
        function __construct() {
            # FooBar constructor logic here
            call_user_func(array(get_parent_class(get_parent_class($this)), '__construct'));
        }
    }
    
    $foo = new FooBar();
    echo $foo->f; #=> 'Good!'
    

    Again, this isn't a viable solution for a situation where you have no idea how much inheritance has taken place, due to the limitations of debug_backtrace(), but in controlled circumstances, it works as intended.

    0 讨论(0)
  • 2020-11-28 22:04

    I ended up coming up with an alternative solution that solved the problem.

    • I created an intermediate class that extended Grandpa.
    • Then both Papa and Kiddo extended that class.
    • Kiddo required some intermediate functionality of Papa but didn't like it's constructor so the class has that additional functionality and both extend it.

    I've upvoted the other two answers that provided valid yet ugly solutions for an uglier question:)

    0 讨论(0)
  • 2020-11-28 22:04
        class Grandpa 
    {
        public function __construct()
        {
            echo"Hello Kiddo";
        }    
    }
    
    class Papa extends Grandpa
    {
        public function __construct()
        {            
        }
        public function CallGranddad()
        {
            parent::__construct();
        }
    }
    
    class Kiddo extends Papa
    {
        public function __construct()
        {
    
        }
        public function needSomethingFromGrandDad
        {
           parent::CallGranddad();
        }
    }
    
    0 讨论(0)
提交回复
热议问题