> The original motivation was, IIRC, to initialize class constants with an
> expression.
> Following KISS, I feel we should constrain our efforts to that scope, and
> that's how I voted:
> "Yes" on class constants and static members, "No" on everything else.
I'd also personally found the `static $x = null; if ($x === null) { $x =
some_fn(); }` pattern inconvenient and wanted that to be just `static $x =
some_fn();` to make the initial value `some_fn()` if possible. It also seemed
more readable (e.g. doesn't matter if some_fn() can be constant).
My personal use case for parameter defaults and global constants isn't as
strong (especially with define()),
but it's still something I'd want.
- I'd be able to use `private const FOO_DEFAULT = array_keys(...); function
foo($x = FOO_DEFAULT) {}` if constants did pass, so that's less off a concern,
and ensures&makes clear that the expression is only evaluated once, so it might
be a better place to start with.
> As for implementation, we must manage the complexity.
> I'm a hard "No" on restricting use to an arbitrary, variable list of blessed
> functions.
> It's a dev UX nightmare to include a bevy of array_* functions but no str*
> functions.
If the RFC succeeded with a whitelist, I'd planned to create RFCs to add to the
whitelist.
Locale dependency of string functions was the reason for leaving it out
initially, since that may also be a hard "no" for others.
> Consequently, the only way to safely initialize class constants and static
> members is at run-time, and I can only think of one way to do it.
> Apologies if this has already been suggested: taking a cue from C#, a static
> class constructor ([1])
> would allow us to have expression-initialized constants and static members.
> <?php
> class Config {
> public const URL = null; // compile time initialization
> protected static $mtime = null; // compile time initialization
> private static function __constructStatic() {
> $env = json_decode(file_get_contents('config.json'));
> self::URL = $env->url;
> self::$mtime = filemtime('config.json');
> }
> public function reload() {
> if (self::$mtime < filemtime('config.json'))
> self::__constructStatic();
> }
> }
> }
>
> echo Config::URL; // assert: runtime has already called _constructStatic
> $config = new Config; // assert: __constructStatic called only once by runtime
> $config->reload(); // instance may call its own static constructor
?>
This seems to allow the same functionality my RFC would allow, in a slightly
different way.
I didn't go with that syntax because it'd move the values of the
constants/properties away from the declaration.
Also, if `json_decode()` threw, then there would never be a way for a PHP
implementation to initialize `self::$mtime` in that example, even though
they're unrelated in how they're initialized there.
Allowing calling `__constructStatic` multiple times (thus assigning to the
**const** `self::URL` multiple times) would be a significant change to php
internals, and out of the scope of this RFC
https://wiki.php.net/rfc/calls_in_constant_expressions_poll ("Put the RFC(Poll)
URL into all your replies.")
- Tyson
--
PHP Internals - PHP Runtime Development Mailing List
To unsubscribe, visit: http://www.php.net/unsub.php