> On Mar 27, 2021, at 12:05 PM, Rowan Tommins <rowan.coll...@gmail.com> wrote:
> 
> On 27/03/2021 00:05, Nuno Maduro wrote:
>> 
>> I've just added a few more tests with the exact examples you have presented 
>> in this mail list: 
>> https://github.com/php/php-src/pull/6246/commits/c3a50d671c5d8fa4b775ec67fe77d0cbd5cc8030
>>  
>> <https://github.com/php/php-src/pull/6246/commits/c3a50d671c5d8fa4b775ec67fe77d0cbd5cc8030>.
> 
> 
> Hi Nuno,
> 
> Thanks, I hadn't thought of writing out test cases, that makes a lot of 
> sense; although I now realise my question wasn't very clear.
> 
> My biggest concern with automatic capture is the potential for *accidentally* 
> capturing variables - that is, intending to introduce a local variable inside 
> the closure, but capturing a variable from the outer scope that happens to 
> have the same name. This is less of an issue with capture by value, but can 
> still mean resources not being freed, e.g. a large array of data not being 
> freed from memory, or an object destructor not executing when expected.
> 
> This is more likely in PHP than many other languages, because there is no 
> requirement, or even an option, to explicitly declare a local variable in the 
> closure. It's not a new problem, but since single-expression closures are 
> unlikely to use many local variables, it's probably not one that's been given 
> lots of thought.
> 
> 
> I've written some tests that demonstrate the current behaviour using an 
> object destructor: 
> https://github.com/nunomaduro/php-src/commit/ae18662cc92f5d07520b4574dcae71d38a9e0a41
> 
> Based on those, there seems to be no way to prevent a variable being 
> captured, even if its value is immediately discarded each time the closure 
> runs. This may be less than ideal:
> 
> $results = getLargeReportFromDatabase();
> // ...
> $fn = fn() {
>     $results = []; // coincidentally the same name, immediately assigned its 
> own value
>     // ...
> }
> unset($results); // does not free the array, because $fn has captured the 
> value

In this case doesn't $results get released when $fn goes out of scope?  

IOW, if fn() is used and discarded by the end of the function in which is it 
declared then this accidental capture is not an issue (unless it breaks the 
outer code, of course), right?

If it breaks in the outer code then I would say that is not really any 
different than overwriting the variable in a later loop, for example. So not 
really an indictment of auto-capture.

It is only if the function allows the closure to live on by returned it 
directly or indirectly, assigning it to a by-reference parameter or assigning 
to a property of an object that was passed in that it can "leak."

Which then begs the question of "How often do closure get returned in a manner 
they have a long life AND a large number of the same types of closures are 
created?  I can envision how that might happen, but I question how common that 
scenario would be?

And finally, most to the time a PHP page terminates after a page load thus 
releasing all these captured variables.  Yes CLI apps and "Fiber" apps might be 
different, but they are certainly less common than request/response web pages.

IOW, it feels like there would need be a confluence of many factors that 
collectively are likely rare before this scenario could become a real problem. 

Where one solution would be, of course, to not write long functions such that 
it is hard to identify accidentally captured variables. Which seems like a best 
practice anyway?

Did I get anything wrong above? I don't think so but then I could well have 
missed something.

In summary, is this accidental-capture-which-turns-into-a-resource-leak 
actually a problem worth worrying about?


> I wonder if the capture analysis could be made smarter to make this less 
> likely.
> 
> 
> PS I'd like to apologise if some of my messages in this thread have come 
> across as harsh or demanding, I do appreciate you bringing up this feature, 
> as I know it's one a lot of people want, even if I'm sceptical.
> 
> Regards,
> 
> -- 
> Rowan Tommins
> [IMSoP]
> 
> -- 
> PHP Internals - PHP Runtime Development Mailing List
> To unsubscribe, visit: https://www.php.net/unsub.php
> 

--
PHP Internals - PHP Runtime Development Mailing List
To unsubscribe, visit: https://www.php.net/unsub.php

Reply via email to