Hey internals,
You may remember my RFC about Records (https://wiki.php.net/rfc/records). There was something intentionally left out, and due to life events, I am quite behind schedule on this one. BUT, what I left out (and the main reason for the short syntax) is the ability to have locally scoped records in a class definition.
Example using an anonymous record:
$x = new class {
public record Point(int $x, int $y);
// from inside the class, there is no special scope
public function foo(): Point {
return new Point(1, 1);
}
}
$point = new $x::Point(1, 2);
Anyway, I think I've settled on an implementation that is acceptable, but I wanted your feedback before moving forward.
Instead of this being limited to Records -- an entirely new type of type and syntax -- would we rather have short class declarations (similar to the rules in the Records RFC) and locally scoped classes?
class Outer {
// able to be instantiated from anywhere
public class Inner(int $y);
// a private/protected class can only be instantiated within methods/classes bound to Outer
protected readonly class ProtectedInner(int $z);
}
Instantiation is done by static access: new Outer::Inner(1), which is currently a syntax error. Public nested classes may be used as type hints and return types as well as properties in other classes.
There are quite a few more things left to investigate before updating the RFC (or drafting a new one), but I am at a crossroads and I wanted to gather people's feedback first. If this is a big "NO" (or if someone else has started working on this and I'm just so happening to accidentally step on their toes, again): I don't want to spend several days/weeks digging into the details for classes. However, it would probably look something like the above.
Note, this doesn't preclude you from using a longer syntax, if that is your preference.
class Outer {
public class Inner {
public function __construct(public int $z) {}
}
}
Anyway, I'd love to hear any preferences or thoughts -- strong or otherwise. I probably won't reply, but I will read everything and take it into consideration as I continue down this road.
Also, if there is already someone working on this ... please speak up so I don't get accused of stealing ideas again!
— Rob
Hi
Am 2025-02-07 15:23, schrieb Rob Landers:
Instantiation is done by static access: new Outer::Inner(1), which is
currently a syntax error.
How would I access static members on an inner class without having
ambiguity? Outer::Inner::Const
already is valid syntax.
If this is a big "NO" (or if someone else has started working on this
and I'm just so happening to accidentally step on their toes, again): I
don't want to spend several days/weeks digging into the details for
classes. However, it would probably look something like the above.
I believe Ilija has some proof-of-concept regarding file-private
classes, which would sidestep the above problem of disambiguating static
member access.
Best regards
Tim Düsterhus
Hi
Am 2025-02-07 15:23, schrieb Rob Landers:
Instantiation is done by static access: new Outer::Inner(1), which is
currently a syntax error.How would I access static members on an inner class without having
ambiguity?Outer::Inner::Const
already is valid syntax.
I can actually answer this one. The name of an inner class must not conflict with either a const or a method in the outer class because this is also currently valid syntax new (Outer::Inner::Const)
where it resolves to a string of an existing class or new (Outer::Inner::Method())
where it returns a string to an existing class.
— Rob
Hi there
I believe Ilija has some proof-of-concept regarding file-private
classes, which would sidestep the above problem of disambiguating static
member access.
Indeed. I shared a draft internally and responses were mixed.
https://wiki.php.net/rfc/private-classes-and-functions
https://github.com/php/php-src/compare/master...iluuu1994:php-src:private-classes
The RFC proposes to allow declaring classes and functions as private,
which will limit their usage to the current file, or more accurately,
to the current namespace block (i.e. namespace {}, not namespace
across files)
The motivation is to allow declaring small helper classes (e.g. mocks
for tests) in the file where they are used. Of course, this is
currently already possible, but discouraged to preserve the ability to
autoload classes. However, autoloading is not necessary for private
members, so this concern goes away.
Furthermore, private mangles the class/function name so that it
doesn't conflict with other files declaring a class/function with the
same private name. For example, multiple tests can create a private
class DbMock without worrying about conflicts.
To summarize, the main benefits for private is:
- Single-use classes/functions live closer to their point of usage.
- It signals to users that the members are not intended to be used
outside the given file. - It avoids naming conflicts for cases between private members.
The main pushback was that "private" does not sufficiently indicate
the semantics of this feature. This could simply be solved with a
different keyword. I guess for others, "namespace private" is a more
interesting concept.
One thing I disliked about the implementation is that private members
still pollute the global symbol tables. That said, it's not likely to
matter in practice.
Anyway, I didn't have any immediate plans to pursue this RFC, so feel
free to take over in any way you like.
Ilija