rfc:optional-interfaces
Differences
This shows you the differences between two versions of the page.
Next revision | Previous revision | ||
rfc:optional-interfaces [2024/12/28 20:00] – created tontonsb | rfc:optional-interfaces [2025/04/03 13:08] (current) – external edit 127.0.0.1 | ||
---|---|---|---|
Line 1: | Line 1: | ||
====== PHP RFC: Optional Interfaces ====== | ====== PHP RFC: Optional Interfaces ====== | ||
- | * Version: | + | * Version: 1.0 |
* Date: 2024-12-28 | * Date: 2024-12-28 | ||
* Author: Juris Evertovskis, | * Author: Juris Evertovskis, | ||
- | * Status: | + | * Status: |
* First Published at: http:// | * First Published at: http:// | ||
Line 61: | Line 61: | ||
<PHP> | <PHP> | ||
interface MyInterface extends RequiredInterface, | interface MyInterface extends RequiredInterface, | ||
+ | </ | ||
+ | |||
+ | ==== Dynamically defined interfaces ==== | ||
+ | |||
+ | The presence of optional interfaces is evaluated when the class is defined. | ||
+ | |||
+ | <PHP> | ||
+ | // This class does not implement OptionalInterface as it doesn' | ||
+ | class OldClass implements ? | ||
+ | $oldClass = new OldClass; | ||
+ | |||
+ | eval(' | ||
+ | |||
+ | // This class implements OptionalInterface as it does exist now | ||
+ | class NewClass implements ? | ||
+ | |||
+ | // The instance of OldClass is unaffected | ||
+ | class_implements($oldclass); | ||
+ | |||
+ | // The definition of OldClass is not affected either | ||
+ | // New instances will still NOT implement the interface that didn't exist when the class was defined | ||
+ | $oldClass2 = new OldClass; | ||
+ | class_implements($oldClass2)); | ||
+ | </ | ||
+ | |||
+ | ==== Interface implementation ==== | ||
+ | |||
+ | The presence of ''?'' | ||
+ | |||
+ | If an optional interface exists, the implementation is checked as it is with non-optional interfaces. This code would still be invalid: | ||
+ | |||
+ | <PHP> | ||
+ | interface Iface | ||
+ | { | ||
+ | public function someMethod(); | ||
+ | } | ||
+ | |||
+ | class NaughtyClass implements ?Iface {} | ||
+ | |||
+ | // Fatal error: Class NaughtyClass contains 1 abstract method and must therefore be declared abstract or implement the remaining method (Iface:: | ||
+ | </ | ||
+ | |||
+ | If a method is marked as an Override, it must override or implement something in an interface that is actually implemented. | ||
+ | |||
+ | <PHP> | ||
+ | interface ExistingInterface | ||
+ | { | ||
+ | public function method(); | ||
+ | } | ||
+ | |||
+ | class TestClass implements ? | ||
+ | { | ||
+ | # | ||
+ | public function method() {} // This is fine as it implements an interface method | ||
+ | |||
+ | # | ||
+ | public function other() {} // This is still not allowed | ||
+ | } | ||
+ | |||
+ | // Fatal error: TestClass:: | ||
+ | </ | ||
+ | |||
+ | By using ''# | ||
+ | |||
+ | <PHP> | ||
+ | class Signable implements ? | ||
+ | { | ||
+ | # | ||
+ | public function getDoc(): string | ||
+ | { | ||
+ | return $this-> | ||
+ | } | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | |||
+ | ==== Type checks ==== | ||
+ | |||
+ | A class only conforms to types that are actually implemented. | ||
+ | |||
+ | <PHP> | ||
+ | interface ExistingInterface {} | ||
+ | |||
+ | class TestClass implements ? | ||
+ | |||
+ | function f1(ExistingInterface $x) { echo " | ||
+ | function f2(NonExistantInterface $x) { echo " | ||
+ | |||
+ | $c = new TestClass; | ||
+ | |||
+ | f1($c); // F1 | ||
+ | f2($c); // Fatal error: Uncaught TypeError: f2(): Argument #1 ($x) must be of type NonExistantInterface, | ||
+ | </ | ||
+ | |||
+ | ==== Example 4: Use case — compatibility with an optional library === | ||
+ | |||
+ | Consider a package that allows one to build DB expressions with an OOP syntax. On it's own it could be used via PDO like this: | ||
+ | |||
+ | <PHP> | ||
+ | $expression = new ColumnExpression(' | ||
+ | |||
+ | $statement = $pdo-> | ||
+ | update links | ||
+ | set url = $expression | ||
+ | where id = ? | ||
+ | "); | ||
+ | |||
+ | // merge bindings with other bindings | ||
+ | $statement-> | ||
+ | </ | ||
+ | |||
+ | But to use these expressions in Laravel' | ||
+ | |||
+ | One of the possibilities observed in the wild is to extract the actual implementation of our expression class to a '' | ||
+ | |||
+ | <PHP> | ||
+ | use Illuminate\Contracts\Database\Query\Expression as LaravelExpression; | ||
+ | |||
+ | if (interface_exists(LaravelExpression:: | ||
+ | /** | ||
+ | * Class description | ||
+ | */ | ||
+ | class BaseExpression implements \Stringable, | ||
+ | { | ||
+ | use BaseExpressionImplementation; | ||
+ | } | ||
+ | } else { | ||
+ | /** | ||
+ | * Class description | ||
+ | */ | ||
+ | class BaseExpression implements \Stringable | ||
+ | { | ||
+ | use BaseExpressionImplementation; | ||
+ | } | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | However by utilizing optional interfaces one can create this class in a more clear and straightforward way: | ||
+ | |||
+ | <PHP> | ||
+ | use Illuminate\Contracts\Database\Query\Expression as LaravelExpression; | ||
+ | |||
+ | /** | ||
+ | * Class description | ||
+ | */ | ||
+ | class BaseExpression implements \Stringable, | ||
+ | { | ||
+ | // The actual implementation | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | ==== Example 5: Use case — compatibility with multiple versions ==== | ||
+ | |||
+ | Consider you're working on the perfect collections' | ||
+ | |||
+ | What do you do? Should you drop the PHP 8.13 support and add the interfaces? Maintain two versions? Do some kind of conditional class definition like in Example 4? If your library had some kind of booting process, you could define dummy '' | ||
+ | |||
+ | With optional interfaces you can add optional support to the optional features: | ||
+ | |||
+ | <PHP> | ||
+ | class Collection implements ArrayAccess, | ||
+ | { | ||
+ | // the implementation here | ||
+ | } | ||
</ | </ | ||
Line 74: | Line 238: | ||
==== To Opcache ==== | ==== To Opcache ==== | ||
- | > It is necessary to develop RFC's with opcache in mind, since opcache is a core extension distributed with PHP. | ||
- | > Please explain how you have verified your RFC's compatibility | + | Opcache will store the interface names along with their optionality information. This means that a cached class will implement an optional interface once it exists even if the interface didn't exist during the request when the class was cached. The opposite is also true — if an optional interface is no longer loaded during |
- | + | ||
- | I have not verified | + | |
==== To Reflection API ==== | ==== To Reflection API ==== | ||
- | Currently this RFC does not add the optionality information in the Reflection API. | + | This RFC does not add the optionality information in the Reflection API. Reflection will only list the interfaces that are actually implemented. |
+ | |||
+ | < | ||
+ | interface IfaceA {} | ||
+ | |||
+ | class C implements ?IfaceA, ?IfaceB {} | ||
+ | |||
+ | $reflection = new ReflectionClass(' | ||
+ | $reflection-> | ||
+ | </ | ||
+ | |||
+ | ==== To Autoloading ==== | ||
+ | |||
+ | The presence of the ''?'' | ||
===== Proposed Voting Choices ===== | ===== Proposed Voting Choices ===== | ||
- | Include these so readers know where you are heading | + | |
+ | Yes or no vote. 2/3 required to pass. | ||
+ | |||
+ | Voting started on 2025-03-14 | ||
+ | |||
+ | <doodle title=" | ||
+ | * Yes | ||
+ | * No | ||
+ | </ | ||
===== Implementation ===== | ===== Implementation ===== | ||
+ | |||
+ | * Implementation (WIP): https:// | ||
+ | |||
After the project is implemented, | After the project is implemented, | ||
- the version(s) it was merged into | - the version(s) it was merged into | ||
Line 95: | Line 280: | ||
===== References ===== | ===== References ===== | ||
- | Links to external references, discussions or RFCs | + | |
+ | * Discussion: https:// | ||
===== Rejected Features ===== | ===== Rejected Features ===== | ||
Keep this updated with features that were discussed on the mail lists. | Keep this updated with features that were discussed on the mail lists. |
rfc/optional-interfaces.1735416037.txt.gz · Last modified: 2025/04/03 13:08 (external edit)