rfc:nullable_types

PHP RFC: Nullable Types

Introduction

It is common in many programming languages including PHP to allow a variable to be of some type or null. This null often indicates an error or lack of something to return. This can already be done using PHP's dynamic type system by omitting type declarations. It can also be done for parameters in PHP by using a default parameter of null. However, this does not work for return types which do not have a concept of a default value. This RFC proposes a unified way to add nullable types to both parameters and returns.

Proposal

This proposal adds a leading question mark symbol (?) to indicate that a type can also be null. Nullable types can be formed from any currently permitted type. Nullable types are permitted anywhere type declarations are allowed but are subject to some inheritance rules which are outlined later in the RFC.

Here are a few examples to demonstrate basic functionality:

function answer(): ?int  {
    return null; //ok
}
 
function answer(): ?int  {
    return 42; // ok
}
 
function answer(): ?int {
    return new stdclass(); // error
}
function say(?string $msg) {
    if ($msg) {
        echo $msg;
    }
}
 
say('hello'); // ok -- prints hello
say(null); // ok -- does not print
say(); // error -- missing parameter
say(new stdclass); //error -- bad type

Return Types

When subtyping a return type the nullability can be removed by a subclass, but it cannot be added:

interface Fooable {
    function foo(): ?Fooable;
}
interface StrictFooable extends Fooable {
    function foo(): Fooable; // valid
}
interface Fooable {
    function foo(): Fooable;
}
interface LooseFooable extends Fooable {
    function foo(): ?Fooable; // invalid
}

Parameter Types

The nullable type cannot be removed in a sub-class; it can be added if not present in a super-class. This behavior is consistent with the Liskov substitution principle.

// Valid use: loosening the nullable marker in a parameter:
interface Fooable {
    function foo(Fooable $f);
}
interface LooseFoo extends Fooable {
    function foo(?Fooable $f);
}
// Invalid use: tightening the nullable marker in a parameter:
interface Fooable {
    function foo(?Fooable $f);
}
interface StrictFoo extends Fooable {
    function foo(Fooable $f);
}

Default Values

Differences from default values

Parameters with a nullable type do not have a default value. If omitted the value does not default to null and will result in an error:

function f(?callable $p) {}
f(); // invalid; function f does not have a default

Relationship with default values

PHP's existing semantics allow giving a null default value for a parameter to make it nullable and optional:

function foo_default(Bar $bar = null) {}
 
foo_default(new Bar); // valid
foo_default(null); // valid
foo_default(); // valid

This existing behaviour is not changed by this RFC. The new nullable type feature offers a subset of the functionality of = null with both making a parameter nullable but only = null making a parameter optional and have a default value:

function foo_nullable(?Bar $bar) {}
 
foo_nullable(new Bar); // valid
foo_nullable(null); // valid
foo_nullable(); // INVALID!

As = null offers a superset of ?'s functionality, it could be said that = null implies ?. However, it is perfectly legal to use both to make a parameter's nullability explicit:

function foo_both(?Bar $bar = null) {}
 
foo_both(new Bar); // valid
foo_both(null); // valid
foo_both(); // valid

Because a parameter with = null is a superset of ?, you can use a parameter with a default value of null where a nullable type existed in the parent.

interface Contract {
    function method(?Foo $foo): bool;
}
 
class Implementation implements Contract {
    function method(?Foo $foo = null): bool {
        return is_null($foo);
    }
}

The reverse is not true, however: you cannot use only a nullable type where a default value existed in the parent, because the parameter is no longer optional.

PHP Version

This RFC targets PHP 7.1.

Voting Choices

The vote for this RFC is split into two votes. One vote will be for accepting the idea of explicitly nullable types with the short-hand syntax. The second vote determines whether to merge only nullable return types or to also merge nullable parameter types as well.

Voting began Tuesday, May 10th, 2016 and will close on Friday, May 20th, 2016.


Accept explicitly nullable types? (2/3 affirmative required)
Real name Yes No
ajf (ajf)  
beberlei (beberlei)  
bishop (bishop)  
bwoebi (bwoebi)  
colinodell (colinodell)  
daverandom (daverandom)  
davey (davey)  
derick (derick)  
dmitry (dmitry)  
francois (francois)  
galvao (galvao)  
guilhermeblanco (guilhermeblanco)  
hywan (hywan)  
jhdxr (jhdxr)  
jwage (jwage)  
kalle (kalle)  
kguest (kguest)  
klaussilveira (klaussilveira)  
lcobucci (lcobucci)  
lstrojny (lstrojny)  
malukenho (malukenho)  
mariano (mariano)  
mbeccati (mbeccati)  
mcmic (mcmic)  
mfonda (mfonda)  
mgocobachi (mgocobachi)  
mike (mike)  
nikic (nikic)  
ocramius (ocramius)  
omars (omars)  
peehaa (peehaa)  
philstu (philstu)  
pierrick (pierrick)  
pollita (pollita)  
salathe (salathe)  
sammyk (sammyk)  
stas (stas)  
svpernova09 (svpernova09)  
till (till)  
tpunt (tpunt)  
trowski (trowski)  
weierophinney (weierophinney)  
zimt (zimt)  
Final result: 41 2
This poll has been closed.

Merge which features? (majority wins)
Real name Both nullable parameter and return types Only nullable return types
ajf (ajf)  
beberlei (beberlei)  
bishop (bishop)  
bwoebi (bwoebi)  
colinodell (colinodell)  
danack (danack)  
daverandom (daverandom)  
davey (davey)  
derick (derick)  
dragoonis (dragoonis)  
francois (francois)  
galvao (galvao)  
guilhermeblanco (guilhermeblanco)  
hywan (hywan)  
jhdxr (jhdxr)  
jwage (jwage)  
kalle (kalle)  
kguest (kguest)  
kinncj (kinncj)  
klaussilveira (klaussilveira)  
lstrojny (lstrojny)  
malukenho (malukenho)  
marcio (marcio)  
mariano (mariano)  
mcmic (mcmic)  
mfonda (mfonda)  
mgocobachi (mgocobachi)  
mike (mike)  
nikic (nikic)  
ocramius (ocramius)  
omars (omars)  
peehaa (peehaa)  
philstu (philstu)  
pierrick (pierrick)  
pollita (pollita)  
rdohms (rdohms)  
salathe (salathe)  
sammyk (sammyk)  
stas (stas)  
svpernova09 (svpernova09)  
till (till)  
tpunt (tpunt)  
trowski (trowski)  
weierophinney (weierophinney)  
zimt (zimt)  
Final result: 44 1
This poll has been closed.

Patches and Tests

The pull request for this RFC is here: https://github.com/php/php-src/pull/1893.

RFC Impact

To Backward Compatibility

There is a backwards compatibility break in certain cases. This was previously fixed as a bug but it was decided that because of the BC break that it would be pushed to this RFC. See bug 72119 for more info on the BC break.

This BC break is to reject parameter covariance for nullable types:

interface Fooable {
    function foo(?Fooable $f);
}
interface StrictFoo extends Fooable {
    // Invalid; parent type allows null so subtype must also allow it
    function foo(Fooable $f);
}

However, it breaks this code:

interface Fooable {
    function foo(array $f = null);
}
interface LooseFoo extends Fooable {
    function foo(array $f = []);
}

Such code should be modified to also allow null:

interface LooseFoo extends Fooable {
    function foo(?array $f = []);
}

Note that more handling is probably necessary to make the code robust, but this small change is sufficient for any previously working code to continue to work.

To Existing Extensions

Only extensions that deal with the AST need to be updated. They should be aware of the ZEND_TYPE_NULLABLE attribute that gets set when a ? is present.

To Union Types

Nullable types are a special case of union types where there only two types in the union, one of which is always null. If the union_types RFC is accepted then ?Foo will be exactly equivalent to Foo | Null. The union types RFC will be responsible for intersecting decisions, such as whether ? can be used in conjunction with other union types.

Unaffected PHP Functionality

This RFC does not deprecate the default value syntax. While there is some overlap of features between it and this RFC, they serve different purposes. As such, the default value syntax will remain.

Future Scope

Implementation

After the project is implemented, this section should contain

  1. the version(s) it was merged to
  2. a link to the git commit
  3. a link to the PHP manual entry for the feature

References

  1. Discussion on mailing list
  2. Background for multiple type enhancements: http://news.php.net/php.internals/92252
rfc/nullable_types.txt · Last modified: 2021/07/12 18:01 by bwoebi