Newsgroups: php.internals Path: news.php.net Xref: news.php.net php.internals:124497 X-Original-To: internals@lists.php.net Delivered-To: internals@lists.php.net Received: from php-smtp4.php.net (php-smtp4.php.net [45.112.84.5]) by qa.php.net (Postfix) with ESMTPS id AEDDF1A00B7 for ; Thu, 18 Jul 2024 19:38:43 +0000 (UTC) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=php.net; s=mail; t=1721331614; bh=oCcRbgNtiQer6zmr1qJMGEReqduijtTzPiHK8h9/Cdo=; h=Date:Subject:To:Cc:References:From:In-Reply-To:From; b=fNiLsL2U5QsdDaUPAvHqm3wqlgK+LaMDM9Im5azNNoVeDvsFTdLie1Pc7gCuFeu9q 23SoidEgGyv1gkX51qtsbFNOyPIWkLvfPIbvEasH2R/fe8br3QsochjX1MArwfFUcd g3CyGFtJSNA4RbGqpNi1iubpwMkJzTp5VvHRoQZWOWbe7+Klkx95vwxTmIV+DjMkCT afDydjxi6zzbFzCt8S/Ewjaj6Gzxwd4DOtYAqPT7ROJica7w8nNQdj0cuiwa7Jkb+9 oybh1jK8o2PRW3Uap9GbY05XB9IM2F9HRpDswqrad/gmWRo981AHhJxusOWu0nmd70 JGBNN4HGVARtw== Received: from php-smtp4.php.net (localhost [127.0.0.1]) by php-smtp4.php.net (Postfix) with ESMTP id 4EB54180722 for ; Thu, 18 Jul 2024 19:40:13 +0000 (UTC) X-Spam-Checker-Version: SpamAssassin 4.0.0 (2022-12-13) on php-smtp4.php.net X-Spam-Level: X-Spam-Status: No, score=0.6 required=5.0 tests=BAYES_50,DKIM_SIGNED, DKIM_VALID,DKIM_VALID_AU,DKIM_VALID_EF,DMARC_PASS,SPF_HELO_NONE, SPF_PASS autolearn=no autolearn_force=no version=4.0.0 X-Spam-Virus: No X-Envelope-From: Received: from chrono.xqk7.com (chrono.xqk7.com [176.9.45.72]) (using TLSv1.3 with cipher TLS_AES_256_GCM_SHA384 (256/256 bits) key-exchange X25519 server-signature RSA-PSS (2048 bits) server-digest SHA256) (No client certificate requested) by php-smtp4.php.net (Postfix) with ESMTPS for ; Thu, 18 Jul 2024 19:40:12 +0000 (UTC) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=bastelstu.be; s=mail20171119; t=1721331520; bh=QxWOcCQ0+4NGNFPhRtd7nGRHb8yL5xsunmWVcY9IB5A=; h=Message-ID:Date:MIME-Version:Subject:To:Cc:References:From: In-Reply-To:Content-Type:from:to:cc:subject:message-id; b=Nt2pMVQJSB81r+DSZ+H2nIaD/vLcMSUq6ij2hcBgnsho9JsHij5Sv6jg+aM+Ojkde xCFrOZvlOJBI5ia8m4v4G+e9aHghj2Nl5YESAuhMA0i0NpdAvgghEH1ajjhFKStgHy YUIOAHsVSJxBaY5/bRDN4l1z0xTWbeyzyaSKGpaSPdS7lgZgEkN4Mf4MeA6gNPRCZN bXw8ZojB6TQLnKxOLKjV6rZL8jVY7e49DqSglx+UkUPUHXcXLc85ieI+xd5+9bCjNG wXHJkO73Vy3kQ+JgcOu+FrrgR9ExEj3C23IzLZppnz9yGLjf8lFucI6FwZlKtpio2x Pgk248AowlG9Q== Message-ID: <658b9367-a231-4d9f-9fa5-8cf493768614@bastelstu.be> Date: Thu, 18 Jul 2024 21:38:39 +0200 Precedence: bulk list-help: list-post: List-Id: internals.lists.php.net x-ms-reactions: disallow MIME-Version: 1.0 Subject: Re: [PHP-DEV] [RFC] Lazy Objects To: Nicolas Grekas Cc: PHP Internals List References: <1118bbcd-a7b4-47bf-bf35-1a36ab4628e1@bastelstu.be> <45847b93-02bf-459f-bcd2-81ba35a12c24@bastelstu.be> <46bd4098-2936-4e46-98e9-fe55118325c2@bastelstu.be> <61ab36bc-b045-452a-84e0-87367d4c680e@bastelstu.be> Content-Language: en-US In-Reply-To: Content-Type: text/plain; charset=UTF-8; format=flowed Content-Transfer-Encoding: 8bit From: tim@bastelstu.be (=?UTF-8?Q?Tim_D=C3=BCsterhus?=) Hi On 7/15/24 10:23, Nicolas Grekas wrote: >> To me this is what the language evolution should do: Enable users to do >> things that previously needed to be provided by userland libraries, >> because they were complicated and fragile, not enabling userland >> libraries to simplify things that they should not need to provide in the >> first place because the language already provides it. >> > > That's exactly it: instead of using a third party lib (or in this case > implementing a poor man's subset of a correct lazy object implementation), > the engine would enable using a native feature to achieve a fully correct > behavior in a very simple way. In this case, LazyServiceEntityRepository > would directly use ReflectionClass::resetAsLazyGhost to make itself lazy, > so that its consumers get a packaged behavior and don't need to care about > the topic while consuming the class. > I guess we have to agree to disagree here. >> Yes, I think it is clearer. Let me try to rephrase this differently to >> see if my understanding is correct: >> >> --- >> >> For every property on that exists on the real instance, the property on >> the proxy instance effectively [1] is replaced by a property hook like >> the following: >> >> public PropertyType $propertyName { >> get { >> return $this->realInstance->propertyName; >> } >> set(PropertyType $value) { >> $this->realInstance->propertyName = $value; >> } >> } >> >> And value that is stored in the property will be freed (including >> calling the destructor if it was the last reference), as if `unset()` >> was called on the property. >> >> [1] No actual property hook will be created and the `realInstance` >> property does not actually exist, but the semantics behave as if such a >> hook would be applied. >> > > Conceptually, you've got it right yes! > Sweet. Unless I've missed anything the bit about the value being unset and the destructor implications is missing in the RFC text. It should be added. Also the "Properties that are declared on the real instance are bound to the proxy instance" bit because slightly misleading with the newest change, because the proxy may no longer define additional properties. May I suggest something along the lines of the following: The proxy's properties will be bound to the proxy instance, so that accessing any of these properties on the proxy forwards the operation to the corresponding property on the real instance as if the proxy's property was a virtual property. Any value stored within the proxy's properties will be unset() and the destructor will be called if the proxy held the last reference. This includes properties used with ReflectionProperty::skipLazyInitialization() or setRawValueWithoutLazyInitialization(). >> Frankly, thinking about this cloning behavior gives me a headache, >> because it quickly leads to very weird semantics. Consider the following >> example: >> >> $predefinedObject = new SomeObj(); >> $initializer = function () use ($predefinedObject) { >> return $predefinedObject; >> }; >> $myProxy = $r->newLazyProxy($initializer); >> $otherProxy = $r->newLazyProxy($initializer); >> $clonedProxy = clone $myProxy; >> $r->initialize($myProxy); >> $r->initialize($otherProxy); >> $r->initialize($clonedProxy); >> >> To my understanding both $myProxy and $otherProxy would share the >> $predefinedObject as the real instance and $clonedProxy would have a >> clone of the $predefinedObject at the time of the initialization as its >> real instance? >> > > Correct. The sharing is not specifically related to cloning. But when > cloning happens, the expected behavior is well defined: we should have > separate states. > Yes, it's clear that the should have separate states. The issue I'm having here is that the actual cloning does not happen at the time of the `clone` operation, but at an arbitrary later point in time and this can have some odd consequences for the object lifecycles. Perhaps my example was too simplified. Let me try to expand the example a little. class SomeObj { public string $foo = 'A'; public string $dummy; } $predefinedObject = new SomeObj(); $initializer = function () use ($predefinedObject) { return $predefinedObject; }; $myProxy = $r->newLazyProxy($initializer); $otherProxy = $r->newLazyProxy($initializer); $r->getProperty('foo')->skipLazyInitialization($myProxy); $clonedProxy = clone $myProxy; var_dump($clonedProxy->foo); $r->initialize($myProxy); $r->initialize($otherProxy); $myProxy->foo = 'B'; $r->initialize($clonedProxy); var_dump($clonedProxy->foo); I would expect that this dumps 'A' both of the times, because at the time of cloning the $foo property held the the value 'A'. But my understanding is that it returns 'A' at the first time and 'B' at the second time, because `$predefinedObject` is cloned at the time of the `$r->initialize($clonedProxy);` call. > >> To me this sounds like cloning an uninitialized proxy would need to >> trigger an initialization to result in semantics that do not violate the >> principle of least astonishment. >> > > Forcing an initialization when cloning would be unexpected. E.g. in > Doctrine, when you clone an uninitialized entity, you don't trigger a > database roundtrip. Instead, you create a new object that still references > the original state internally, but under a different object identity. This > cloning behavior is the one we've had for more than 10 years and I think > it's also the least astonishing one - at least if we consider this example > as a real world trial of this principle. > See above. Best regards Tim Düsterhus