'Details of PHP 7.4 Reference Counting of Zvals
I have some points of clarification based on reading the internal implementations for zvals described here Internal value representation in PHP 7 - Part 1 and Internal value representation in PHP 7 - Part 2.
Before explaining my confusion in detail, I think it can be summed up by:
(i) I do not see why object reference counting differs from that of arrays and strings
(ii) why reference counting for strings differs from that of arrays.
From what I understand, for "complex" data types like strings, arrays, and objects, those entities are reference counted types (in PHP 7.4).
- Within the _zval_struct class, there is member to track its type and a member of type zend_value, which is a union.
- Members of this union include pointers to zend_array, zend_object, zend_string, zend_reference
- The zend_array and the likes internally store a reference count and other internals necessary for functionality
1 So based on this simplistic view, I would imagine the reference counting for arrays, strings, and objects in PHP are the same, but it appears not to be the case. Can someone explain where this (apparently) overly simplistic view falls apart?
I used the 'debug_zval_dump' function to count references (I do not have privileges to install xdebug to use 'xdebug_debug_zval'). In my subsequent analysis, I am assuming that the function's own parameter can influence the reference counts displayed.
2 For objects, the following counts make sense to me, but it would be great to have confirmation.
class Foo{}
$a = new Foo();
debug_zval_dump($a); // #1: $a refcount = 2 -- and assume this is run after each line of code for each variable
$b = $a; // #2: $a, $b refcount = 3
$c = &$a; // #3: $a, $b, and $c refcount = 3
$a = new Foo(); // #4 $a, $b, and $c refcount = 2
Just to be clear: every time a variable is defined or updated, I pass all existing variables to debug_zval_dump in a PHP 7.4 engine. That's what the refcounts refer to. I'm just saving space.
#1: There are two _zval_structs pointing to the object (one is the function parameter).
#2: There are three _zval_structs pointing to the object (counting the function parameter)
#3: The function parameter, $b, and a zend_reference (shared by $a and $c) point to the object
#4: $a and $c refer to the same object through a common zend_reference (plus the function parameter) and $b and the function parameter refer to the same object.
Is this counting wrong? Please correct me if so. Otherwise, we move to the more confusing items:
3 Arrays:
$a=[]; // #1: $a refcount = 1
$b=$a; // #2: $a, $b refcount = 1
$c=&$a; // #3: $a, $b, and $c refcount = 1
$a[]=0; // #4: $a and $c refcount = 2, $b refcount = 1
I would expect the same numbers as for 2.#1-#4 and that's not what what we get. This appears to be at odds with the PHP article linked to as, I would expect something more like the following at #4:
$a, $c -> zend_reference1(refcount=2) -> zend_array2(refcount=2,value=[0])
$parameter -------------------------------^
$b, $parameter -> zend_array1(refcount=2,value=[])
4 There is then yet different counting for strings.
$a=''; // #1: $a refcount = 1
$b=$a; // #2: $a, $b refcount = 1
$c=&$a; // #3: $a, $b, and $c refcount = 1
$a='foo'; // #4: $a, $b, and $c refcount = 1
I would have the same diagram here as for 3.
What details am I overlooking for this reference counting?
5 As a bonus, what happens when a reference is made to a number now? For example
$a=0;
$b=&$a; // $a, b -> zend_reference(refcount=2) -> zend_value(value=0)
Is the comment diagram correct, assuming that zend_value is stack based (since numeric values are not reference counted)?
Solution 1:[1]
I think Daniel (iJungleBoy) already answered this in the comments here. Correct way to get access to Dnn Stuff while using Modern 2sxc Hybrid Views
"to get the Dnn object you must use the base class Custom.Dnn.Razor12 - it's only difference is the Dnn object. That cannot work on Hybrid, because Oqtane won't have a Dnn object - otherwise it's the same as Custom.Hybrid.Razor12"
Sources
This article follows the attribution requirements of Stack Overflow and is licensed under CC BY-SA 3.0.
Source: Stack Overflow
Solution | Source |
---|---|
Solution 1 | Jeremy Farrance |