On 02/03/2017 11:53 AM, Levi Morrison wrote:
Thanks to everyone who has participated in the discussion thus far.
Primarily the feedback has been directed at the `fn` keyword. Let me
provide two benefits and drawbacks of using `fn` as a keyword:
1. `fn` is searchable in search engines and in our manual
2. Is more intuitive than just syntax
However, `fn` does have downsides:
1. Can break existing code
2. We already have a similar keyword `function`
To that end, I'd like to gauge interest in a pure syntax based
alternative that is similar to Rust and Ruby.
Instead of:
fn(params) => expr
What about:
|params| => expr
This trades the advantages of the keyword for the advantages of pure
syntax, and happens to be two characters shorter. To be explicit:
1. Preserves 100% backwards compatibility
2. Avoids having two keywords that both mean "function"
3. Is not easily searchable in engines or in the manual
4. Is a tad bit shorter
What do you think, Internals?
One more thing: I'd like to re-emphasize that the syntax that
JavaScript uses and the one that HHVM/Hack uses are ambiguous in the
current class of our grammar. The following will not work unless we
move to a more powerful grammar and parser class:
(params) => expr
(params) ==> expr
This is why an un-ambiguous prefix is necessary: the prefix breaks the
ambiguities. The syntax I have suggested in the RFC and the one I
suggested just now are not ambiguous because distinct prefixes:
fn(params) => expr
|params| => expr
I look forward to more discussion!
Off hand, either of the above options would be acceptable to me. The
union type concern is a valid one, but as noted I don't know how much of
an issue that would be in practice even if we ever did get union types.
(Now, if we also added defined type aliases... :-) )
I am definitely not a fan of reusing `function`. The way I see it, the
point of a short-closure is to take a simple expression and wrap it into
a function so that it can be plugged into a function context. I don't
even think of it in the same way I would a named function/method, more
casting an expression to a function.
Additionally, to the point that was raised about developers being able
to understand syntax reused in different context, that's true... if the
context is in fact different. A bitwise AND and a reference are
extremely different situations so there's no reason for my brain to
expect both to mean the same thing. One is never a viable substitute
for the other.
In contrast, however, we're talking about two different syntaxes for
defining anonymous functions that have different closure semantics.
Reusing `function` would mean:
$n = 2;
$x = function ($a) use ($n) { return $a * $n; }
$y = function ($a) => $a * $n;
Are equivalent. But in order to know what the capture semantics are I
can't tell from the start of the expression. I have to look halfway
down the line to know what the behavior is going to be for these two
extremely similar actions.
In contrast:
$n = 2;
$x = function ($a) use ($n) { return $a * $n; }
$y = fn($a) => $a * $n;
$z = |$a| => $a * $n;
I in versions $y and $z, I can tell right from the beginning of the line
which variant of "anonymous function capture logic" I will be dealing
with. I know from character 1 or 2, not character 12.
And then there's the extra typing and visual space taken up, which is
not something to dismiss entirely. Also, as someone else mentioned this
syntax *seems* like it would support nesting. Compare the following:
$x = function($a) => function($b) => function($c) => $a * $b * $c;
$y = fn($a) => fn($b) => fn($c) => $a * $b * $c;
$z = |$a| => |$b| => |$c| => $a * $b * $c;
print $x(1)(2)(3); // prints 6
Which of these is more readable? Id' argue it's not $x.
--Larry Garfield
--
PHP Internals - PHP Runtime Development Mailing List
To unsubscribe, visit: http://www.php.net/unsub.php