rfc:nameof
Differences
This shows you the differences between two versions of the page.
Both sides previous revisionPrevious revisionNext revision | Previous revision | ||
rfc:nameof [2023/05/07 22:19] – remove confusing qualifier withinboredom | rfc:nameof [2024/01/14 22:49] (current) – reflect discussions withinboredom | ||
---|---|---|---|
Line 2: | Line 2: | ||
* Date: 2023-05-07 | * Date: 2023-05-07 | ||
* 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 ===== | ||
- | Currently, in PHP, it is possible to get the name of nearly anything. You can retrieve | + | During |
+ | |||
+ | This can be used in attributes: | ||
+ | |||
+ | <code PHP> | ||
+ | # | ||
+ | class Person { | ||
+ | public function serialize() { /* do stuff */ } | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | In error messages: | ||
+ | |||
+ | <code PHP> | ||
+ | function divide(float $numerator, float $denominator): | ||
+ | if($denominator === 0) throw new InvalidArgumentException(nameof($denominator) . ' | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | In match statements: | ||
+ | |||
+ | <code PHP> | ||
+ | match($propertyName) { | ||
+ | nameof(MyClass-> | ||
+ | nameof(MyClass-> | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | Or anywhere a literal string | ||
+ | |||
+ | <code PHP> | ||
+ | $arr[nameof(myfunc(...))] = true; | ||
+ | </ | ||
+ | |||
+ | As named parameters become more prevalent, providing the end user with the name of the parameter is more important than ever, in a way that it is easy to locate when refactoring, | ||
+ | |||
+ | <code PHP> | ||
+ | function | ||
+ | // potentially many lines later... | ||
+ | if(is_called_with_deprecated_value($oldParam)) { | ||
+ | $logger-> | ||
+ | } | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | Here are some further examples showing how the identifier is transformed: | ||
+ | |||
+ | <code PHP> | ||
+ | echo nameof($variable); | ||
+ | echo nameof(MyClass-> | ||
+ | echo nameof(MyClass:: | ||
+ | echo nameof(Enum:: | ||
+ | echo nameof(Object:: | ||
+ | echo nameof(myFunction(...)); // myFunction | ||
+ | echo nameof(\MyNamespace\myFunction(...)); | ||
+ | echo nameof(MY_CONSTANT); | ||
+ | </ | ||
+ | |||
+ | It's important to note that some things have an ambiguous name that is a compile error: | ||
+ | |||
+ | <code PHP> | ||
+ | echo nameof($a[$b]); | ||
+ | echo nameof($a === $b); // ?? | ||
+ | echo nameof($a($b)); | ||
+ | echo nameof($$a); | ||
+ | echo nameof($a-> | ||
+ | </ | ||
+ | |||
+ | These will generate a compile error: " | ||
+ | |||
+ | Note that only the right-most part of the identifier is translated to a string. This is because this is almost always what you want. For example, in the case of '' | ||
+ | |||
+ | To continue the ''" | ||
===== Proposal ===== | ===== Proposal ===== | ||
- | This RFC proposes introducing a new global function called | + | A '' |
- | This function returns the 'name of' | + | There is a special case for objects: |
- | ====== Examples ====== | + | 1. A ''::'' |
+ | 2. A '' | ||
+ | 3. The left-hand side must be the type name; it can be an abstract type or an interface. | ||
+ | 4. The left-hand side cannot be a variable. | ||
- | '' | + | If the first two rules are broken (e.g., a '' |
- | '' | + | There are a limited number of expressions that can resolve to a name using '' |
- | '' | + | * variables (but not variable-variables): |
+ | * properties: '' | ||
+ | * first-class callables: '' | ||
+ | * static properties and constants: '' | ||
+ | * constants: '' | ||
- | '' | + | The name will always be fully qualified, with a beginning slash, if it is a first-class callable or constant and it refers to a globally accessible identifier. For example: |
- | '' | + | <code PHP> |
+ | namespace Example; | ||
- | '' | + | class A { |
+ | public function doStuff(): void {} | ||
+ | } | ||
+ | |||
+ | function doStuff(): void {} | ||
+ | |||
+ | echo nameof(doStuff(...)); | ||
+ | echo nameof(A-> | ||
+ | </ | ||
+ | |||
+ | Additionally, | ||
+ | |||
+ | <code PHP> | ||
+ | trait A { | ||
+ | public function example() {} | ||
+ | } | ||
+ | |||
+ | trait B { | ||
+ | public function example() {} | ||
+ | } | ||
+ | |||
+ | class C { | ||
+ | use A, B { | ||
+ | A::example insteadof B; | ||
+ | B::example as exampleB; | ||
+ | } | ||
+ | } | ||
+ | |||
+ | echo nameof(C-> | ||
+ | echo nameof(B-> | ||
+ | </ | ||
+ | |||
+ | This adheres to the basic refactoring safety mentioned earlier, since refactoring | ||
===== Backward Incompatible Changes ===== | ===== Backward Incompatible Changes ===== | ||
- | What breaks, | + | There are no backward incompatible changes at this time except that `nameof` will become a reserved word. |
+ | |||
+ | ===== Future Scope ===== | ||
+ | |||
+ | This could be expanded in the future to allow classes | ||
===== Proposed PHP Version(s) ===== | ===== Proposed PHP Version(s) ===== | ||
- | 8.3: release | + | * 8.4: release |
- | 8.2.x: '' | + | |
===== RFC Impact ===== | ===== RFC Impact ===== | ||
Line 45: | Line 161: | ||
==== To Opcache ==== | ==== To Opcache ==== | ||
- | The function call is compiled to a string. | + | |
+ | a new AST node (similar to isset) will need to be handled. | ||
==== New Constants ==== | ==== New Constants ==== | ||
Line 51: | Line 168: | ||
===== Open Issues ===== | ===== Open Issues ===== | ||
- | None thus far | ||
===== Unaffected PHP Functionality ===== | ===== Unaffected PHP Functionality ===== | ||
- | PHP will largely be unaffected by this change, except that a new global function is introduced. | + | PHP will largely be unaffected by this change. |
===== Future Scope ===== | ===== Future Scope ===== | ||
Line 60: | Line 176: | ||
===== Proposed Voting Choices ===== | ===== Proposed Voting Choices ===== | ||
- | Include these so readers know where you are heading and can discuss the proposed voting options. | + | This is a simple yes-or-no vote to include this feature. 2/3 majority required to pass. |
===== Patches and Tests ===== | ===== Patches and Tests ===== | ||
- | https:// | + | experimental implementation: |
===== Implementation ===== | ===== Implementation ===== | ||
Line 74: | Line 190: | ||
===== References ===== | ===== References ===== | ||
- | Links to external references, discussions or RFCs | + | - internals discussion: https:// |
===== Rejected Features ===== | ===== Rejected Features ===== | ||
- | Keep this updated with features that were discussed on the mail lists. | + | |
+ | Classes need not be supported by '' | ||
+ | |||
+ | There were some suggestions to use '':: | ||
+ | |||
+ | <code PHP> | ||
+ | class A { | ||
+ | public const name = ' | ||
+ | } | ||
+ | |||
+ | function example(): A { | ||
+ | return new A(); | ||
+ | } | ||
+ | |||
+ | echo example():: | ||
+ | echo A::name; | ||
+ | </ | ||
+ | |||
+ | The author believes '' | ||
+ | |||
+ | <code PHP> | ||
+ | const A = ' | ||
+ | |||
+ | class A { | ||
+ | public const name = ' | ||
+ | } | ||
+ | |||
+ | echo A::name; | ||
+ | </ | ||
+ | |||
+ | Are we getting the string " |
rfc/nameof.1683497963.txt.gz · Last modified: 2023/05/07 22:19 by withinboredom