On Mon, Apr 22, 2019, at 4:47 PM, Benjamin Morel wrote: > Hi internals, > > I'd like to revive an old discussion <https://externals.io/message/67131> > about > object type casting. > > The idea would be to allow (ClassName) casting: > > $service = (EmailService) $diContainer->get('email.service'); > > The above code would throw a TypeError if the value is not an instance of > the given class. I see the following advantages: > > - Type safety: we can be sure that the value is of the correct type or that > we'll get an Error. This syntax allows to fail early if the variable > happens to not be of the expected type, and avoids much more verbose checks; > - Static analysis: IDEs and static code analysis tools can now understand > the type of the variable, without having to resort to `@var` annotations. > > These combine into a third advantage: readability. Today's equivalent of > the above one-liner could be: > > /** @var EmailService $service */ > $service = $diContainer->get('email.service'); > if (! $service instanceof EmailService) { > throw new TypeError('Expected instance of EmailService, ...'); > } > > Which is a lot of boilerplate code that could be easily avoided by > introducing this new syntax. > > Before moving forward and working on a formal RFC, I'd like to hear your > thoughts: what's your early feeling about this? Did I miss other > discussions around this subject? Are there any technical issues that come > to mind? Could this feature help the upcoming JIT compiler produce more > efficient machine code by knowing the type of the variable at compile time? > etc. > > Note: "casting" might not be the perfect name here as what we're really > doing is a type check, but this reuses the type casting syntax and > resembles Java's object casting. > > Thank you, > Ben
Hi Ben. First thought: I'm all for easy ways to be more type-explicit, so yay on the concept. Second thought: That said, how many use cases for that are there other than function boundaries, which we already have covered? I can think of two: foreach() loops and returns where you know the return type with more specificity than the method you're calling. Example: /** @var Foo $foo *// foreach ($arrayOfFoo as $foo) { $foo->bar(); } Example from PSR-14, in which you know the object you're getting back MUST be the same one that's passed in but dispatch() has no return type: /** @var Foo $event **/ $event = $dispatcher->dispatch(new Foo()); The second I can see being easily handled by this syntax. The former, how would that look? Third thought: Casting is the wrong name here, and feels also misleading as a syntax. (float)$anInt means "type coerce this thing into a float", which cannot error. You're suggesting (Foo)$bar to mean "if this isn't already a Foo, throw." That's a very different behavior semantic for the same syntax. Is that a land mine? I would expect (Foo)$bar to mean "recast $bar into an instance of Foo if possible, and error if not". Which... I suppose "is it already" is a subcase of that, but it's still not the behavior I'd expect from that syntax. --Larry Garfield -- PHP Internals - PHP Runtime Development Mailing List To unsubscribe, visit: http://www.php.net/unsub.php