rfc:operator_overrides_lite
Differences
This shows you the differences between two versions of the page.
Both sides previous revisionPrevious revisionNext revision | Previous revision | ||
rfc:operator_overrides_lite [2024/06/28 19:37] – add method rules and how to use them withinboredom | rfc:operator_overrides_lite [2024/07/07 14:01] (current) – withdrawn withinboredom | ||
---|---|---|---|
Line 1: | Line 1: | ||
====== PHP RFC: Operator Overrides -- Lite Edition ====== | ====== PHP RFC: Operator Overrides -- Lite Edition ====== | ||
- | * Version: 0.0 | + | * Version: 0.1 |
* Date: 2024-06-28 | * Date: 2024-06-28 | ||
* Author: Robert Landers, landers.robert@gmail.com | * Author: Robert Landers, landers.robert@gmail.com | ||
- | * Status: | + | * Status: |
* First Published at: http:// | * First Published at: http:// | ||
===== Introduction ===== | ===== Introduction ===== | ||
- | Nearly three years ago, [[rfc: | + | Nearly three years ago, [[rfc: |
+ | |||
+ | This is to integers as ArrayAccess is to arrays, and Stringable is to strings, providing a simplified framework for defining integer mathematics, | ||
+ | |||
+ | ===== What this is not ===== | ||
+ | |||
+ | This is not a full operator overrides implementation; | ||
+ | |||
+ | ===== Why the GMP extension? ===== | ||
+ | |||
+ | Several curious people have wondered why this is focused on the GMP extension. The reasons are quite simple: | ||
+ | |||
+ | - Numbers are surprisingly hard to implement "from scratch" | ||
+ | - GMP has a practically infinite numerical range, meaning if you wanted to write a " | ||
+ | - People using the GMP extension are already familiar with the way overloading works in the engine. | ||
+ | - The extension already has all the plumbing in-place to handle casting, operator overloads, and all in an OOP oriented way. | ||
===== Proposal ===== | ===== Proposal ===== | ||
- | The \GMP class will have the following | + | The \GMP class will be changed to the following |
- | * add(mixed $left, mixed $right): static | + | <code php> |
- | * subtract(mixed $left, mixed $right): static | + | readonly class GMP |
- | * multiply(mixed $left, mixed $right): static | + | { |
- | * divide(mixed $left, mixed $right): static | + | |
- | * mod(mixed $left, mixed $right): static | + | |
- | * exp(mixed $left, mixed $right): static | + | |
- | * comparable(mixed $left, mixed right): bool | + | |
- | These methods will enable operator-like behavior when extended. Overriding equals, less-than, greater-than, | + | public function __serialize(): |
- | Binary operations will be handled as such: | + | public function __unserialize(array $data): void {} |
- | - If both operands are instances of `\GMP`, standard behavior applies. | + | protected function add(GMP|int|string $left, GMP|int|string $right): GMP {} |
- | - If one operand is a subclass of `\GMP`, the subclass' | + | |
- | - If both operands are subclasses, only the left operand' | + | |
- | Here' | + | protected function multiply(GMP|int|string $left, GMP|int|string $right): GMP {} |
+ | |||
+ | protected function subtract(GMP|int|string $left, GMP|int|string $right): GMP {} | ||
+ | |||
+ | protected function divide(GMP|int|string $left, GMP|int|string $right): GMP {} | ||
+ | |||
+ | protected function mod(GMP|int|string $left, GMP|int|string $right): GMP {} | ||
+ | |||
+ | protected function pow(GMP|int|string $base, GMP|int|string $exp): GMP {} | ||
+ | |||
+ | protected function comparable(GMP|int|string $op1, GMP|int|string $op2): bool {} | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | The changes are listed below: | ||
+ | |||
+ | - The class is made '' | ||
+ | - The constructor accepts another GMP instance instead of only '' | ||
+ | - Mathematical methods are added to the class as protected. It is expected that developers wanting to support an operation make the associated operation '' | ||
+ | |||
+ | For any existing GMP code, absolutely nothing changes. The extension' | ||
+ | |||
+ | Note the lack of equals, less-than, and greater-than operators. These are deliberately left out and replaced by a " | ||
+ | |||
+ | - PHP reorders comparables as needed | ||
+ | - The core tenant here is that these objects should represent types of integers, and it is expected that they are internally comparable but possibly not comparable to other types. For example, the number of goats on a farm may not be comparable to the speed of a car. | ||
+ | |||
+ | Shift and bitwise operations are also left out of the class because it is expected that these behaviors won't change for any type of number that could be implemented here. | ||
+ | |||
+ | Below is a listing showing a polyfill, a Duration class that only allows adding, multiplying, | ||
<code php> | <code php> | ||
- | class Duration extends \GMP { | + | <?php |
- | public function __construct(int|\GMP|float $time) { | + | |
- | | + | if(!class_exists(' |
+ | | ||
+ | readonly class GMP { | ||
+ | | ||
+ | |||
+ | protected function add(GMP|int|string | ||
+ | return new self($left? | ||
+ | | ||
+ | |||
+ | protected function multiply(GMP|int|string $left, GMP|int|string $right): GMP { | ||
+ | return new self($left? | ||
+ | } | ||
+ | |||
+ | protected function subtract(GMP|int|string $left, GMP|int|string $right): GMP { | ||
+ | return new self($left? | ||
+ | } | ||
+ | |||
+ | protected function divide(GMP|int|string $left, GMP|int|string $right): GMP { | ||
+ | return new self($left? | ||
+ | } | ||
+ | |||
+ | protected function mod(GMP|int|string $left, GMP|int|string $right): GMP { | ||
+ | return new self($left? | ||
+ | } | ||
+ | |||
+ | protected function pow(GMP|int|string $base, GMP|int|string $exp): GMP { | ||
+ | return new self($base? | ||
+ | } | ||
+ | |||
+ | protected function comparable(GMP|int|string $op1, GMP|int|string $op2): bool { | ||
+ | return is_numeric($op1) && is_numeric($op2); | ||
+ | } | ||
} | } | ||
+ | } | ||
- | private static | + | readonly class Duration extends GMP { |
- | if ($arg instanceof self || is_numeric($arg) || (is_object($arg) && get_class($arg) === '\GMP')) { | + | public |
- | return; | + | |
+ | if($num < 0) { | ||
+ | throw new ArithmeticError('Duration cannot be negative.'); | ||
} | } | ||
- | throw new LogicException(' | + | |
+ | parent:: | ||
} | } | ||
- | private | + | private function |
- | if (is_numeric($arg) || (is_object($arg) && get_class($arg) === ' | + | if(is_numeric($maybeScalar)) { |
return; | return; | ||
} | } | ||
- | throw new LogicException(' | ||
- | } | ||
- | protected static function add(mixed $left, mixed $right): static | + | if(get_class($maybeScalar) === ' |
- | | + | |
- | | + | } |
- | | + | |
- | protected static function subtract(mixed $left, mixed $right): static { | + | throw new ValueError("Can only perform |
- | self:: | + | |
- | return new self(parent:: | + | |
} | } | ||
- | | + | |
- | if ($left | + | |
- | if ($right | + | if($left |
- | return new self(parent:: | + | $this-> |
+ | return; | ||
+ | } | ||
+ | |||
+ | if($right | ||
+ | $this-> | ||
+ | return; | ||
+ | } | ||
+ | |||
+ | throw new LogicException('$left or $right | ||
} | } | ||
- | | + | |
- | self:: | + | return new self($result); |
- | return new self(parent:: | + | |
} | } | ||
- | | + | |
- | | + | |
+ | | ||
+ | |||
+ | return $this-> | ||
} | } | ||
- | | + | |
- | | + | |
+ | | ||
+ | |||
+ | return $this-> | ||
} | } | ||
- | | + | |
- | return $left instanceof self && | + | |
+ | $this-> | ||
+ | |||
+ | if($base !== $this) { | ||
+ | throw new LogicException(' | ||
+ | } | ||
+ | |||
+ | return $this-> | ||
} | } | ||
- | public function | + | public function |
- | return $this / 60; | + | |
+ | return $op1 instanceof self && $op2 instanceof self; | ||
} | } | ||
- | + | } | |
- | /* and so on */ | + | |
- | | + | |
- | </ | + | |
- | This approach provides valuable semantics and type safety, especially in contexts like attributes and properties: | + | $duration = new Duration(10); |
+ | $other = new Duration(200); | ||
+ | $regular = new GMP(10); | ||
- | <code php> | + | function do_op($description, |
- | define('SECOND', | + | global $duration, $other, $regular; |
+ | try { | ||
+ | $result = eval(' | ||
+ | echo " | ||
+ | } catch (Throwable $exception) { | ||
+ | echo " | ||
+ | } | ||
+ | } | ||
+ | |||
+ | do_op('Duration', | ||
+ | do_op(' | ||
+ | do_op(' | ||
- | // Use in attributes | + | do_op(' |
- | #[Delay(5 * SECOND)] | + | do_op(' |
+ | do_op(' | ||
+ | do_op(' | ||
+ | do_op(' | ||
+ | do_op(' | ||
+ | do_op(' | ||
+ | do_op(' | ||
+ | do_op(' | ||
- | // Use in properties | + | /** |
- | public | + | * Output: |
+ | * Duration: $duration = 10 | ||
+ | * Regular: $regular = 10 | ||
+ | * Other: $other = 200 | ||
+ | * Regular: $regular + 10 = 20 | ||
+ | * Duration: $duration + 10 = 20 | ||
+ | * Duration + Duration: $duration + $other = [ValueError: | ||
+ | * Duration + Regular: $duration + $regular = 20 | ||
+ | * Division not allowed: $duration | ||
+ | * Multiplication: | ||
+ | * No negatives: $duration + -20 = [ArithmeticError: | ||
+ | * Comparison: | ||
+ | * Comparison failure: $duration < 20 = [ArithmeticError: | ||
+ | */ | ||
</ | </ | ||
- | ====== Methods and their definitions ====== | + | The following exceptions can be thrown by the engine while attempting to perform mathematical operations: |
- | Add, subtract, multiply, divide, | + | * Error: when trying to perform an unsupported operation, which shows as " |
+ | * ArithmeticError: | ||
- | The comparable method determines if the objects being compared can be more than, less than, or equal to each other. For example, money is probably | + | Additionally, |
===== Backward Incompatible Changes ===== | ===== Backward Incompatible Changes ===== | ||
Line 141: | Line 263: | ||
* support for other operators | * support for other operators | ||
* serialization/ | * serialization/ | ||
+ | * support for non-integers via BCMath | ||
===== Proposed Voting Choices ===== | ===== Proposed Voting Choices ===== | ||
- | * Allow extending the \GMP class and use a form of operator overloading | + | * 2/3 YES|NO vote: Allow extending the \GMP class and use a form of operator overloading |
- | * Disallow extending the \GMP class | + | * 2/3 YES|NO secondary vote: Disallow extending the \GMP class if this RFC fails |
- | + | ||
- | A " | + | |
===== Patches and Tests ===== | ===== Patches and Tests ===== | ||
- | A prototype | + | Prototype |
===== Implementation ===== | ===== Implementation ===== | ||
After the project is implemented, | After the project is implemented, | ||
- | - the version(s) it was merged into | + | - |
- a link to the git commit(s) | - a link to the git commit(s) | ||
- a link to the PHP manual entry for the feature | - a link to the PHP manual entry for the feature | ||
Line 163: | Line 284: | ||
Links to external references, discussions or RFCs | Links to external references, discussions or RFCs | ||
- | ===== Rejected Features ===== | + | ===== Rejected Features |
- | Keep this updated | + | |
+ | === A separate extension === | ||
+ | |||
+ | On paper, a separate extension sounds like a good idea. However, I've attempted | ||
+ | |||
+ | === Full Operator Overloading === | ||
+ | |||
+ | Some people | ||
+ | |||
+ | === More Operators === | ||
+ | |||
+ | There are many more operators that could be implemented. However, another concern was raised that people may " | ||
+ |
rfc/operator_overrides_lite.1719603434.txt.gz · Last modified: 2024/06/28 19:37 by withinboredom