Skip to content
Advertisement

PHP Traits: How to circumvenient constructors or force them to be called?

Have a look at the following trait:

JavaScript

As you can see the trait makes sure that the using class holds a certain model instance and it implements certain methods. This works as long as the implementing class does not override the constructor.

So here is my question: I want to make sure that either the constructor is called or a better solution, such that I can instantiate this model on initialization.

Please make in answer which respects Multiple inheritance as well es Multi-Level inheritance.

Advertisement

Answer

I think you are trying to make the trait do a job it is not designed for.

Traits are not a form of multiple inheritance, but rather “horizontal reuse” – they’re often described as “compiler-assisted copy-and-paste”. As such, the job of a trait is to provide some code, so that you don’t have to copy it into the class manually. The only relationship it has is with the class where the use statement occurs, where the code is “pasted”. To aid in this role, it can make some basic requirements of that target class, but after that, the trait takes no part in inheritance.

In your example, you are concerned that a sub-class might try to access $primaryModel without running the constructor code which initialises it, and you are trying to use the trait to enforce that; but this is not actually the trait’s responsibility.

The following definitions of class Sub are completely equivalent:

JavaScript

vs:

JavaScript

In either case, class Sub could have its own definition of foo(), and by-pass the logic you’d written in the parent class.

The only contract that can prevent that is the final keyword, which in your case would mean marking your constructor as final. You can then provide an extension point that can be overridden for sub-classes to add their own initialisation:

JavaScript

A trait can also mark its constructor as final, but this is part of the code being pasted, not a requirement on the class using the trait. You could actually use a trait with a constructor, but then write a new constructor as well, and it would mask the trait’s version completely:

JavaScript

As far as the trait is concerned, this is like buying a bottle of beer and pouring it down the sink: you asked for its code and didn’t use it, but that’s your problem.

Crucially, though, you can also alias the methods of the trait, creating a new method with the same code but a different name and/or a different visibility. This means you can mix in code from traits which declare constructors, and use that code in a more complex constructor, or somewhere else in the class altogether.

The target class might also use the “final + hook” pattern:

JavaScript

The trait hasn’t forced the Mixed class to implement this pattern, but it has enabled it, in keeping with its purpose of facilitating code reuse.

Interestingly, the below code doesn’t work, because the as keyword adds an alias, rather than renaming the normal method, so this ends up trying to override the final constructor from Mixed:

JavaScript
User contributions licensed under: CC BY-SA
8 People found this is helpful
Advertisement