rfc:immutability
Differences
This shows you the differences between two versions of the page.
Both sides previous revisionPrevious revisionNext revision | Previous revision | ||
rfc:immutability [2016/08/16 20:12] – added note in examples brzuchal | rfc:immutability [2025/03/22 16:59] (current) – mark as obsolete danielescherzer | ||
---|---|---|---|
Line 1: | Line 1: | ||
====== PHP RFC: Immutable classes and properties ====== | ====== PHP RFC: Immutable classes and properties ====== | ||
- | * Version: 0.1 | + | * Version: 0.2 |
- | * Date: 2016-08-12 | + | * Date: 2018-02-19 |
* Author: Michal Brzuchalski < | * Author: Michal Brzuchalski < | ||
* Author: Silvio Marijic < | * Author: Silvio Marijic < | ||
- | * Status: | + | * Status: |
* First Published at: [[rfc: | * First Published at: [[rfc: | ||
+ | |||
+ | > This proposal has been **superseded** by [[rfc: | ||
===== Introduction ===== | ===== Introduction ===== | ||
- | This RFC proposes introduction of immutable classes and properties. Currently | + | This RFC proposes |
+ | If this mechanism is introduced, developers can be sure that they are programming with no side effects, meaning that state of the object can not be changed without developers being made aware. This is especially useful when dealing with concurrency, | ||
**Pros** | **Pros** | ||
- Immutability guaranteed by language instead of user-land implementation. | - Immutability guaranteed by language instead of user-land implementation. | ||
+ | - Programming with no side effects. | ||
- Safe for concurrency. | - Safe for concurrency. | ||
- Value objects, DTO's etc. can be easily created. | - Value objects, DTO's etc. can be easily created. | ||
- | - Properties can be public which removes need for getters without allowing state modification. | + | - Properties can be public which removes |
- (Please point it out more advantages) | - (Please point it out more advantages) | ||
**Cons** | **Cons** | ||
- | - Cloning has to be disabled for immutable | + | - Currently arrays on immutable |
- | - (Please point it out more disadvantages) | + | |
- | **Before** | + | ===== Proposal ===== |
- | <code php> | + | |
- | <?php | + | |
- | class Email { | + | ==== Immutable Class ==== |
- | private $_email; | + | |
- | public function __construct ($email) { | + | A class defined as immutable will imply immutability across all of its properties by default. After the object is constructed, |
- | // validation | + | |
- | + | ||
- | $this-> | + | |
- | } | + | |
- | + | ||
- | public function getValue() { | + | |
- | | + | |
- | } | + | |
- | } | + | |
- | </ | + | |
- | **After** | ||
<code php> | <code php> | ||
- | <?php | ||
immutable class Email { | immutable class Email { | ||
Line 51: | Line 39: | ||
public function __construct ($email) { | public function __construct ($email) { | ||
- | // validation | ||
- | |||
$this-> | $this-> | ||
} | } | ||
} | } | ||
+ | |||
+ | $email = new Email(" | ||
+ | $email-> | ||
+ | |||
</ | </ | ||
- | ===== Proposal ===== | + | Changes to inheritance are made to add constraints when extending an immutable |
- | + | ||
- | Class defined as immutable | + | |
<code php> | <code php> | ||
+ | immutable class Foo{} | ||
+ | class Bar extends Foo{} // Will result in Fatal Error | ||
+ | </ | ||
- | immutable class Email { | ||
- | public $email; | ||
- | public function __construct ($email) { | + | ==== Immutable Properties ==== |
- | // validation | + | |
+ | Classes have the ability to enforce immutability to only a subset of properties if needed, in that case, immutability will be implied only on properties that are declared as immutable. | ||
+ | |||
+ | <code php> | ||
+ | class User { | ||
+ | private $id; | ||
+ | public immutable $email; | ||
+ | |||
+ | public function __construct ($id, $email) { | ||
+ | $this-> | ||
$this-> | $this-> | ||
} | } | ||
} | } | ||
- | |||
- | $email = new Email(" | ||
- | $email-> | ||
- | |||
</ | </ | ||
- | Regular classes can define immutability per property. | ||
+ | If an immutable property contains an object, to preserve immutability, | ||
<code php> | <code php> | ||
- | class Email { | + | immutable |
+ | |||
+ | class User { | ||
public immutable $email; | public immutable $email; | ||
- | public function __construct ($email) { | + | public function __construct (Email $email) { |
- | // validation | + | |
$this-> | $this-> | ||
} | } | ||
} | } | ||
- | |||
- | $email = new Email(" | ||
- | $email-> | ||
</ | </ | ||
- | Any class that extends | + | Resources are not allowed to be assigned to immutable properties because of fact that resources by nature are not immutable. |
+ | <code php> | ||
+ | class File { | ||
+ | public | ||
+ | public function __construct ($handle) { | ||
+ | $this-> | ||
+ | } | ||
+ | } | ||
+ | |||
+ | $file = new File(fopen(' | ||
+ | </ | ||
+ | |||
+ | Arrays are not allowed to be assigned to immutable properties.. | ||
<code php> | <code php> | ||
- | immutable | + | class A { |
- | class Bar extends Foo{} // Will result in Fatal Error | + | |
+ | |||
+ | public function __construct ($x) { | ||
+ | $this->x = $x; | ||
+ | | ||
+ | } | ||
+ | |||
+ | $a = new A([' | ||
</ | </ | ||
+ | |||
+ | If an immutable property of a parent class is overridden in a child class, it has to be declared as immutable. | ||
+ | Since the focus of this RFC is immutable objects, having static properties in immutable classes will result in a compile error. | ||
- | It will not be possible to assign value by reference | + | ==== References ==== |
+ | |||
+ | Assigning | ||
<code php> | <code php> | ||
immutable class Email { | immutable class Email { | ||
Line 121: | Line 136: | ||
</ | </ | ||
- | If immutable | + | ==== Comparison ==== |
+ | |||
+ | Identity of immutable object | ||
<code php> | <code php> | ||
+ | immutable class Email { | ||
+ | public $email; | ||
- | class Bar{} | + | |
- | + | ||
- | immutable class Foo { | + | |
- | public $bar; | + | |
- | + | ||
- | | + | |
// validation | // validation | ||
- | $this->bar = $bar; | + | $this->email = $email; |
} | } | ||
} | } | ||
- | $foo = new Foo(new Bar()); // Will result in a error because Bar is not instance of immutable class | + | $email1 |
+ | $email2 = new Email(" | ||
+ | var_dump($email1 === $email2); // bool(true) | ||
</ | </ | ||
- | |||
===== Examples ===== | ===== Examples ===== | ||
- | Notice in above examples removing | + | Notice in above example, changing |
- | Every example shows where internal object state is important. Any references to objects passed into immutable class constructor cannot be references to scalars or may be immutable class instances. That gives high guarantee you will keep internal state unchanged for all the time. | + | Every example shows where internal object state is important. Any references to objects passed into an immutable class constructor cannot be references to scalars or may be immutable class instances. That gives high guarantee you will keep internal state unchanged for all the time. |
---- | ---- | ||
Line 151: | Line 166: | ||
==== Money ==== | ==== Money ==== | ||
- | Money Pattern, defined by Martin Fowler and published in Patterns of Enterprise Application Architecture, | + | Money Pattern, defined by Martin Fowler and published in Patterns of Enterprise Application Architecture, |
<code php> | <code php> | ||
Line 285: | Line 300: | ||
</ | </ | ||
- | There is no need for getters because internally immutable object | + | There is no need for getters because |
- | ==== URI ==== | ||
- | Uri's domain is like Money complex ValueObjects and can be presented for eg. in this way. | ||
- | |||
- | <code php> | ||
- | class Scheme { | ||
- | |||
- | private $protocol; | ||
- | |||
- | private function __construct(string $protocol) { | ||
- | $this-> | ||
- | } | ||
- | |||
- | public function getProtocol() | ||
- | { | ||
- | return $this-> | ||
- | } | ||
- | |||
- | public static function HTTP() { | ||
- | return new Scheme(' | ||
- | } | ||
- | } | ||
- | |||
- | class UserInfo { | ||
- | |||
- | private $username; | ||
- | private $password; | ||
- | |||
- | public function __construct(string $username, string $password) | ||
- | { | ||
- | $this-> | ||
- | $this-> | ||
- | } | ||
- | |||
- | public function getUsername() : string | ||
- | { | ||
- | return $this-> | ||
- | } | ||
- | |||
- | public function getPassword() : string | ||
- | { | ||
- | return $this-> | ||
- | } | ||
- | } | ||
- | |||
- | class Authority { | ||
- | |||
- | private $host; | ||
- | private $port; | ||
- | private $userInfo; | ||
- | |||
- | public function __construct(string $host, int $port = 80, UserInfo $userInfo = null) | ||
- | { | ||
- | $this-> | ||
- | $this-> | ||
- | $this-> | ||
- | } | ||
- | |||
- | public function getHost() : string | ||
- | { | ||
- | return $this-> | ||
- | } | ||
- | |||
- | public function getPort() : int | ||
- | { | ||
- | return $this-> | ||
- | } | ||
- | |||
- | public function getUserInfo() : UserInfo | ||
- | { | ||
- | return $this-> | ||
- | } | ||
- | } | ||
- | |||
- | class Path { | ||
- | |||
- | private $path; | ||
- | |||
- | public function __construct(string $path) | ||
- | { | ||
- | $this-> | ||
- | } | ||
- | |||
- | public function getPath() : string | ||
- | { | ||
- | return $this-> | ||
- | } | ||
- | } | ||
- | |||
- | class Uri { | ||
- | |||
- | private $scheme; | ||
- | private $authority; | ||
- | private $path; | ||
- | |||
- | public function __construct(Scheme $scheme, Authority $authority, Path $path) | ||
- | { | ||
- | $this-> | ||
- | $this-> | ||
- | $this-> | ||
- | } | ||
- | |||
- | public function getScheme() : Scheme | ||
- | { | ||
- | return $this-> | ||
- | } | ||
- | |||
- | public function getAuthority() : Authority | ||
- | { | ||
- | return $this-> | ||
- | } | ||
- | |||
- | public function getPath() : Path | ||
- | { | ||
- | return $this-> | ||
- | } | ||
- | } | ||
- | |||
- | $phpNetUri = new Uri(Scheme:: | ||
- | </ | ||
- | |||
- | After refactoring to immutable class implementation the model is getting simplier: | ||
- | |||
- | <code php> | ||
- | immutable class Scheme { | ||
- | |||
- | /** @var string */ | ||
- | public $protocol; | ||
- | |||
- | private function __construct(string $protocol) { | ||
- | $this-> | ||
- | } | ||
- | | ||
- | public static function HTTP() { | ||
- | return new Scheme(' | ||
- | } | ||
- | |||
- | public static function HTTPS() { | ||
- | return new Scheme(' | ||
- | } | ||
- | } | ||
- | |||
- | immutable class UserInfo { | ||
- | |||
- | /** @var string */ | ||
- | public $username; | ||
- | /** @var string */ | ||
- | public $password; | ||
- | |||
- | public function __construct(string $username, string $password) | ||
- | { | ||
- | $this-> | ||
- | $this-> | ||
- | } | ||
- | } | ||
- | |||
- | immutable class Authority { | ||
- | |||
- | /** @var string */ | ||
- | public $host; | ||
- | /** @var int */ | ||
- | public $port; | ||
- | /** @var UserInfo */ | ||
- | public $userInfo; | ||
- | |||
- | public function __construct(string $host, int $port = 80, UserInfo $userInfo = null) | ||
- | { | ||
- | $this-> | ||
- | $this-> | ||
- | $this-> | ||
- | } | ||
- | } | ||
- | |||
- | immutable class Path { | ||
- | |||
- | /** @var string */ | ||
- | public $path; | ||
- | |||
- | public function __construct(string $path) | ||
- | { | ||
- | $this-> | ||
- | } | ||
- | } | ||
- | |||
- | immutable class Uri { | ||
- | |||
- | /** @var Scheme */ | ||
- | public $scheme; | ||
- | /** @var Authority */ | ||
- | public $authority; | ||
- | /** @var Path */ | ||
- | public $path; | ||
- | |||
- | public function __construct(Scheme $scheme, Authority $authority, Path $path) | ||
- | { | ||
- | $this-> | ||
- | $this-> | ||
- | $this-> | ||
- | } | ||
- | | ||
- | public function changeScheme(Scheme $scheme) : Uri { | ||
- | return new self($scheme, | ||
- | } | ||
- | } | ||
- | |||
- | $http = new Uri(Scheme:: | ||
- | $https = $http-> | ||
- | </ | ||
===== Backward Incompatible Changes ===== | ===== Backward Incompatible Changes ===== | ||
- | No backward | + | No backwardly |
===== Proposed PHP Version(s) ===== | ===== Proposed PHP Version(s) ===== | ||
- | - PHP 7.2 | + | To be discussed. |
===== RFC Impact ===== | ===== RFC Impact ===== | ||
Line 511: | Line 319: | ||
==== To Existing Extensions ==== | ==== To Existing Extensions ==== | ||
- | - Reflection. | + | - Reflection |
==== To Opcache ==== | ==== To Opcache ==== | ||
Line 533: | Line 341: | ||
===== Future Scope ===== | ===== Future Scope ===== | ||
+ | Add support for arrays on immutable properties. | ||
+ | Expand immutability to regular variables also. | ||
===== Proposed Voting Choices ===== | ===== Proposed Voting Choices ===== | ||
Line 540: | Line 350: | ||
===== Patches and Tests ===== | ===== Patches and Tests ===== | ||
- | + | [[https:// | |
Line 551: | Line 360: | ||
===== References ===== | ===== References ===== | ||
- | |||
===== Rejected Features ===== | ===== Rejected Features ===== | ||
+ | - Immutable interfaces |
rfc/immutability.1471378348.txt.gz · Last modified: 2017/09/22 13:28 (external edit)