rfc:make_ctor_ret_void

PHP RFC: Make constructors and destructors return void

Large credit for this RFC goes to Michael Voříšek who initially reported the bug and created a draft-- patch.

Introduction

At the moment, constructors and destructors can return values. However, these magic methods are supposed to be void (according to the documentation) and should not return a value. This RFC proposes to deprecate this behavior in PHP 8.0 and subsequently in PHP 9.0 enforce void rules on constructors and destructors.

This would apply both implicitly, where no return type is declared for the constructor/destructor, and explicitly where a void return type is declared.

Proposal

Status quo

Currently, void rules are not enforced for constructors and destructors. Thus, it is allowed to return values from those magic methods:

<?php
class Test {
	public function __construct() {
		return 0;
	}
 
	public function __destruct() {
		return 1;
	}
}
 
$test = new Test();
 
// this prints 0
echo $test->__construct();
// this prints 1
echo $test->__destruct();

But the PHP manual states, that constructors have void return type (i. e. don't return a value). Therefore, the current behavior is inconsistent and incorrect. The void return type rule should always be enforced on constructors/destructors, no matter if the void return type declaration is implicit or explicit.

Proposal

This RFC proposes:

  • to deprecate the ability of returning values from constructors and destructors in PHP 8.0.
  • to treat both constructors and destructors that do not have an explicit return type, as if they have a return type of void in PHP 9.0.
  • to allow explicit void return type on constructors and destructors (secondary vote).

A deprecation warning would be generated:

  • for any constructor or destructor that returns a value in PHP 8.0

Note: if there is an explicit void return type, a fatal error will be generated instead. This allows for newer codebases to take advantage of the check in PHP 8.0 already.

A fatal error would be generated:

  • for any constructor or destructor that returns a value in PHP 9.0.
  • for any constructor or destructor that has an explicit return type other than void (secondary vote).
<?php
class Test {
	public function __construct() {
		// this is illegal
		return 0;
	}
 
	// this is also illegal
	public function __destruct(): mixed {}
}
 
class Test2 {
	// this is legal (secondary vote)
	public function __construct(): void {}
 
	// this is also legal
	public function __destruct() {}
}

Backwards incompatible changes

Accepting this RFC results in a small backwards compatibility break in PHP 9.0 since it will no longer be legal to return (mixed and any of its subtypes) values from constructors and destructors.

The position of this RFC is that this BC break is minimal, as returning values from constructors/destructors is not a standard pattern used by many pieces of code. However, to minimize the number of BC breaks even further, the ability of returning values from constructors/destructors is deprecated in PHP 8.0.

Unaffected functionality

Explicit return type declaration is optional

Explicitly declaring the return type declaration would be optional. It would still be allowed to not specify a type at all:

<?php
class Test {
	// this is legal
	public function __construct() {}
 
	// this is also legal
	public function __destruct(): void {}
}

Constructors are exempt from inheritance checks

Since constructors are exempt from inheritance checks, it is allowed to widen the type from a child class. For example, if the parent class has explicitly declared the constructor as void, it would still be allowed to widen the type to no return type. In other words, covariance (for return types) does not apply to constructors.

<?php
class Test {
	// this is legal
	public function __construct(): void {}
}
 
class Test2 extends Test {
	// this is also legal
	public function __construct() {}
}

Why allow void return type on constructors/destructors?

Enforcing void rules on constructors/destructors implictly but not allowing to declare an explicit type is going to create inconsistencies.

It's key to understand that constructors and destructors in PHP don't work the same way that they do in other languages. First of all, unlike in other languages, constructors and destructors are rather normal functions in PHP and can be called directly i. e. through $object->__construct() and $object->__destruct(). Adding an explicit void return type acts as an extra marker that takes the reader from 99% certain to 100% that these functions are not supposed to return anything. This also aligns with the PHP manual which states that constructors/destructors have a return type of void and the Zen of Python's 2nd principle (“Explicit is better than implicit”). Thus, saying that other languages don't have a concept of return types for constructors does not make much sense in PHP's case.

Rowan Tommins comment on the internals mailing list should also be taken into consideration: “The way I look at it, constructors are mostly declared like a normal method - they use the keyword “function”; can be marked public, private, protected, abstract, and final; and can have a parameter list, with types and defaults - so the surprising thing is that there is a special rule *forbidding* them from having a return specifier”.

Another argument that is used against allowing explicit void return type is that it's going to create code-style wars and is duplicate information. But arguably, everyone also already knows what functions such as __toString() return. You know it's going to be a string. That's its whole purpose. It would be surprising to see any code style forbidding that. Moreover, as of PHP 8.0, constructors and destructors will be the only methods that are not allowed to have a return type. This will be quite inconsistent given that __clone magic method will be able to have an explicit void return type even though both object construction and object cloning work in a similar fashion.

Vote

Primary

Make constructors/destructors return void?
Real name Yes No
alcaeus (alcaeus)  
alec (alec)  
asgrim (asgrim)  
ashnazg (ashnazg)  
beberlei (beberlei)  
bmajdak (bmajdak)  
brandon (brandon)  
brzuchal (brzuchal)  
bwoebi (bwoebi)  
carusogabriel (carusogabriel)  
cpriest (cpriest)  
dams (dams)  
daverandom (daverandom)  
derick (derick)  
dragoonis (dragoonis)  
ekin (ekin)  
galvao (galvao)  
girgias (girgias)  
guilhermeblanco (guilhermeblanco)  
heiglandreas (heiglandreas)  
ilutov (ilutov)  
irker (irker)  
jasny (jasny)  
jbnahan (jbnahan)  
kelunik (kelunik)  
kguest (kguest)  
kocsismate (kocsismate)  
kriscraig (kriscraig)  
levim (levim)  
marcio (marcio)  
mariano (mariano)  
narf (narf)  
nicolasgrekas (nicolasgrekas)  
nikic (nikic)  
nohn (nohn)  
ocramius (ocramius)  
patrickallaert (patrickallaert)  
pollita (pollita)  
rasmus (rasmus)  
reywob (reywob)  
rmf (rmf)  
salathe (salathe)  
sergey (sergey)  
sirsnyder (sirsnyder)  
stas (stas)  
subjective (subjective)  
svpernova09 (svpernova09)  
tandre (tandre)  
theodorejb (theodorejb)  
thorstenr (thorstenr)  
trowski (trowski)  
twosee (twosee)  
weierophinney (weierophinney)  
wyrihaximus (wyrihaximus)  
zeev (zeev)  
zimt (zimt)  
Final result: 34 22
This poll has been closed.

Secondary

Allow void return type on constructors/destructors?
Real name Yes No
alcaeus (alcaeus)  
alec (alec)  
asgrim (asgrim)  
ashnazg (ashnazg)  
beberlei (beberlei)  
bmajdak (bmajdak)  
brzuchal (brzuchal)  
bwoebi (bwoebi)  
carusogabriel (carusogabriel)  
cpriest (cpriest)  
daverandom (daverandom)  
derick (derick)  
dm (dm)  
dragoonis (dragoonis)  
ekin (ekin)  
galvao (galvao)  
girgias (girgias)  
guilhermeblanco (guilhermeblanco)  
heiglandreas (heiglandreas)  
irker (irker)  
jasny (jasny)  
jbnahan (jbnahan)  
kelunik (kelunik)  
kguest (kguest)  
kocsismate (kocsismate)  
kriscraig (kriscraig)  
levim (levim)  
marcio (marcio)  
mariano (mariano)  
narf (narf)  
nicolasgrekas (nicolasgrekas)  
nikic (nikic)  
ocramius (ocramius)  
patrickallaert (patrickallaert)  
pollita (pollita)  
rasmus (rasmus)  
reywob (reywob)  
rmf (rmf)  
salathe (salathe)  
sergey (sergey)  
sirsnyder (sirsnyder)  
stas (stas)  
subjective (subjective)  
svpernova09 (svpernova09)  
tandre (tandre)  
theodorejb (theodorejb)  
thorstenr (thorstenr)  
trowski (trowski)  
twosee (twosee)  
weierophinney (weierophinney)  
wyrihaximus (wyrihaximus)  
zimt (zimt)  
Final result: 27 25
This poll has been closed.

Implementation

rfc/make_ctor_ret_void.txt · Last modified: 2020/07/22 15:29 by moliata