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/12 20:35] – typo withinboredom | rfc:nameof [2025/04/03 13:08] (current) – external edit 127.0.0.1 | ||
---|---|---|---|
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, it is possible to inspect virtually any aspect | + | During PHP execution, it is possible to reflect on most of the code currently running. For example, it is easy to get the name of a class and using reflection utilities: names of properties, methods, and so on. However, getting |
- | '' | + | This can be used in attributes: |
- | This RFC proposes a global '' | + | <code PHP> |
+ | # | ||
+ | class Person { | ||
+ | public | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | In error messages: | ||
<code PHP> | <code PHP> | ||
- | echo nameof($variable); // variable | + | function divide(float $numerator, float $denominator): |
- | echo nameof($object-> | + | if($denominator === 0) throw new InvalidArgumentException(nameof($denominator) . ' is zero'); |
- | echo nameof(Enum::Case); // Case | + | } |
- | echo nameof(Object::Const); // Const | + | </code> |
- | echo nameof(myFunction(...)); | + | |
- | echo nameof(MY_CONSTANT); | + | In match statements: |
+ | |||
+ | <code PHP> | ||
+ | match($propertyName) { | ||
+ | nameof(MyClass-> | ||
+ | nameof(MyClass-> | ||
+ | } | ||
+ | </code> | ||
+ | |||
+ | Or anywhere a literal string can be used: | ||
+ | |||
+ | <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, in a way that it is easy to locate when refactoring, | + | As named parameters become more prevalent, providing the end user with the name of the parameter is more important |
<code PHP> | <code PHP> | ||
Line 33: | Line 52: | ||
</ | </ | ||
- | Further, using '' | + | Here are some further examples showing how the identifier is transformed: |
<code PHP> | <code PHP> | ||
- | // evaluates to # | + | echo nameof($variable); |
- | # | + | echo nameof(MyClass-> |
- | class Operator { | + | echo nameof(MyClass:: |
- | public function handler() { | + | echo nameof(Enum::Case); // Case |
- | // implementation | + | echo nameof(Object::Const); // Const |
- | } | + | echo nameof(myFunction(...)); // myFunction |
- | } | + | echo nameof(\MyNamespace\myFunction(...)); |
+ | echo nameof(MY_CONSTANT); | ||
</ | </ | ||
Line 51: | Line 71: | ||
echo nameof($a === $b); // ?? | echo nameof($a === $b); // ?? | ||
echo nameof($a($b)); | echo nameof($a($b)); | ||
+ | echo nameof($$a); | ||
+ | echo nameof($a-> | ||
</ | </ | ||
- | All of these will generate a compile error: " | + | These will generate a compile error: " |
- | ===== Proposal ===== | + | 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 '' |
- | The '' | + | To continue the '' |
- | <code PHP> | + | ===== Proposal |
- | ${nameof($variable)} | + | |
- | $object-> | + | |
- | constant(Enum:: | + | |
- | constant(Object:: | + | |
- | (nameof(myFunction(...))() === myFunction(); | + | |
- | constant(nameof(MY_CONSTANT)) === MY_CONSTANT; | + | |
- | </ | + | |
- | When getting the name of constants and functions, the name will NOT be the full name, but the lexical name. This means that if the name is '' | + | A '' |
- | <code PHP> | + | There is a special case for objects: |
- | namespace Name { | + | |
- | const MY_CONST = true; | + | |
- | function namedFunction() {} | + | |
- | } | + | |
- | namespace Other { | + | 1. A ''::'' |
- | echo nameof(\Name\MY_CONST); | + | 2. A '' |
- | echo nameof(\Name\namedFunction(...)); // \Name\namedFunction | + | 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. |
- | namespace Using { | + | If the first two rules are broken (e.g., a ''::'' |
- | use const Name\MY_CONST as ; | + | |
- | use const Name\MY_CONST as ALIASED_CONST; | + | |
- | use function Name\namedFunction; | + | |
- | echo nameof(MY_CONST); // MY_CONST | + | There are a limited number of expressions that can resolve to a name using '' |
- | echo nameof(\Name\MY_CONST); | + | |
- | echo nameof(ALIASED_CONST); | + | |
- | echo nameof(namedFunction(...)); | + | |
- | echo nameof(\Name\namedFunction(...)); | + | |
- | } | + | |
- | </ | + | |
- | ===== Error Handling ===== | + | * variables (but not variable-variables): |
+ | * properties: '' | ||
+ | * first-class callables: '' | ||
+ | * static properties and constants: '' | ||
+ | * constants: '' | ||
- | There are several ways to handle errors, each one has its own merits but ultimately | + | The name will always |
- | 1. No error handling | + | <code PHP> |
+ | namespace Example; | ||
- | In this case, there is no error handling. This would allow using '' | + | class A { |
- | + | public | |
- | <code PHP> | + | |
- | class MyClass | + | |
- | public | + | |
} | } | ||
- | nameof(MyClass:: | ||
- | nameof($aVariable); | ||
- | </ | ||
- | The downside is that PHP won't natively inform a developer that they' | + | function doStuff(): void {} |
- | 2. Warnings as usual | + | echo nameof(doStuff(...)); // \Example\doStuff |
+ | echo nameof(A-> | ||
+ | </ | ||
- | In this case, PHP would output warnings | + | Additionally, consider traits |
<code PHP> | <code PHP> | ||
- | class MyClass | + | trait A { |
- | public | + | public |
} | } | ||
- | nameof(MyClass:: | ||
- | // Uncaught Error: Undefined constant MyClass:: | ||
- | nameof($aVariable); | ||
- | // Warning: Undefined variable $aVariable | ||
- | </ | ||
- | 3. Warnings and a way to disable them | + | trait B { |
+ | public function example() {} | ||
+ | } | ||
+ | |||
+ | class C { | ||
+ | use A, B { | ||
+ | A::example insteadof B; | ||
+ | B::example as exampleB; | ||
+ | } | ||
+ | } | ||
+ | |||
+ | echo nameof(C-> | ||
+ | echo nameof(B-> | ||
+ | </ | ||
- | Warnings/ | + | This adheres to the basic refactoring safety mentioned earlier, since refactoring |
===== Backward Incompatible Changes ===== | ===== Backward Incompatible Changes ===== | ||
Line 135: | Line 146: | ||
===== Future Scope ===== | ===== Future Scope ===== | ||
- | This could be expanded in the future to allow classes and types. For example, '' | + | This could be expanded in the future to allow classes and types. For example, '' |
===== Proposed PHP Version(s) ===== | ===== Proposed PHP Version(s) ===== | ||
- | * 8.3: release | + | * 8.4: release |
- | * 8.2.x: '' | + | * 8.3.x: '' |
===== RFC Impact ===== | ===== RFC Impact ===== | ||
Line 150: | 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 156: | Line 168: | ||
===== Open Issues ===== | ===== Open Issues ===== | ||
- | - Should first-class callables be allowed? It's relatively easy to get the name of a first-class callable and only included for completeness. | ||
===== 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 166: | Line 177: | ||
===== Proposed Voting Choices ===== | ===== Proposed Voting Choices ===== | ||
This is a simple yes-or-no vote to include this feature. 2/3 majority required to pass. | This is a simple yes-or-no vote to include this feature. 2/3 majority required to pass. | ||
- | |||
- | There is a subvote for the type of error checking to include. Majority wins. | ||
===== Patches and Tests ===== | ===== Patches and Tests ===== | ||
- | experimental implementation: | + | experimental implementation: |
===== Implementation ===== | ===== Implementation ===== | ||
Line 181: | 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.1683923736.txt.gz · Last modified: 2025/04/03 13:08 (external edit)