rfc:container-offset-behaviour

Differences

This shows you the differences between two versions of the page.

Link to this comparison view

Next revision
Previous revision
rfc:container-offset-behaviour [2024/07/04 13:49] – created girgiasrfc:container-offset-behaviour [2025/04/03 13:08] (current) – external edit 127.0.0.1
Line 1: Line 1:
 ====== PHP RFC: Improve language coherence for the behaviour of offsets and containers ====== ====== PHP RFC: Improve language coherence for the behaviour of offsets and containers ======
  
-  * Version: 0.1+  * Version: 0.3
   * Date: 2023-12-07   * Date: 2023-12-07
   * Author: Gina Peter Banyard <girgias@php.net>   * Author: Gina Peter Banyard <girgias@php.net>
   * Status: Under Discussion   * Status: Under Discussion
-  * Target Version: PHP 8.4+  * Target Version: PHP 8.5
   * Implementations:   * Implementations:
-  * New handler API: https://github.com/Girgias/php-src/pull/19 +    * New handler API: <https://github.com/Girgias/php-src/pull/19> 
-  * <php>ArrayObject</php>: https://github.com/php/php-src/pull/12037 +    * <php>ArrayObject</php>: <https://github.com/php/php-src/pull/12037> 
-  * Original string offset clean-up PR: https://github.com/php/php-src/pull/7173 +    * Original string offset clean-up PR: <https://github.com/php/php-src/pull/7173> 
-  * First Published at: http://wiki.php.net/rfc/container-offset-behaviour +  * First Published at: <http://wiki.php.net/rfc/container-offset-behaviour> 
-  * Markdown source: https://github.com/Girgias/php-rfcs/blob/master/container-offset-behaviour.md+  * Markdown source: <https://github.com/Girgias/php-rfcs/blob/master/container-offset-behaviour.md>
  
 ===== Introduction ===== ===== Introduction =====
Line 32: Line 32:
 ==== Operations ==== ==== Operations ====
  
-We consider there to be nine (9) different operations that relate to containers and offsets,+We consider there to be ten (10) different operations that relate to containers and offsets,
 which are the following: which are the following:
  
Line 44: Line 44:
   * Fetch   * Fetch
   * Fetch-Append   * Fetch-Append
 +  * Incrementing/Decrementing the offset of a container
  
 The read, write, read-write, appending, and unsetting operations are what one would expect. The read, write, read-write, appending, and unsetting operations are what one would expect.
  
 We split the existence check operation into two distinct sub-operations as the behaviour between We split the existence check operation into two distinct sub-operations as the behaviour between
-<php>iseet()</php>/<php>empty()</php> and the null coalesce operator <php>??</php> is sometimes different.+<php>isset()</php>/<php>empty()</php> and the null coalesce operator <php>??</php> is sometimes different.
  
 A fetch operation occurs when reference to the offset must be acquired, A fetch operation occurs when reference to the offset must be acquired,
Line 60: Line 61:
 In general, a nested operation will perform all the necessary fetch/read operations, In general, a nested operation will perform all the necessary fetch/read operations,
 interpreting the returned value as a container, until it reaches the final dimension. interpreting the returned value as a container, until it reaches the final dimension.
 +
 +Finally, we mention incrementing/decrementing because in general it should behave like <php>+= 1</php>,
 +but as it works on a "reference" of the value, the behaviour is slightly tricky.
  
 ==== Container types ==== ==== Container types ====
Line 68: Line 72:
   * <php>false</php>   * <php>false</php>
   * <php>true</php>   * <php>true</php>
-  * <php>bool</php> 
   * <php>int</php>   * <php>int</php>
   * <php>float</php>   * <php>float</php>
Line 118: Line 121:
 as the engine treats those container types identically. as the engine treats those container types identically.
  
-  * For read operations, <php>null</php> is returned and the following warning is emitted: <code> +  * For read operations, <php>null</php> is returned and the following warning is emitted: <code>Warning: Trying to access array offset on TYPE</code> 
-Warning: Trying to access array offset on TYPE + 
-</code> +  * For write, read-write, appending, fetch, fetch-append, and increment/decrement operations, the following error is thrown:<code>Cannot use a scalar value as an array</code> 
-  * For write, read-write, appending, fetch, and fetch-append operations, the following error is thrown: <code> + 
-Cannot use a scalar value as an array +  * For the unset operation, the following error is thrown:<code>Cannot unset offset in a non-array variable</code> 
-</code> +
-  * For the unset operation, the following error is thrown: <code> +
-Cannot unset offset in a non-array variable +
-</code>+
   * For existence operations, no warning is emitted and the behaviour is as if the offset did not exist.   * For existence operations, no warning is emitted and the behaviour is as if the offset did not exist.
  
Line 143: Line 143:
 Therefore, the behaviour depending on the operator is as follows: Therefore, the behaviour depending on the operator is as follows:
  
-  * For read operations,<php>null</php> is returned, the container continues to be <php>null</php>, and the following warning is emitted: <code> +  * For read operations, <php>null</php> is returned, the container continues to be <php>null</php>, and the following warning is emitted: <code>Warning: Trying to access array offset on null</code> 
-Warning: Trying to access array offset on null +
-</code>+
   * For write, append, fetch, and fetch-append operations the container is converted to array. And thus behave like an <php>array</php>, meaning the behaviour depends on the offset type. Please see the <php>array</php> section for details.   * For write, append, fetch, and fetch-append operations the container is converted to array. And thus behave like an <php>array</php>, meaning the behaviour depends on the offset type. Please see the <php>array</php> section for details.
-  * For read-write operations, the container is converted to array, before the read operation. And thus behave like an <php>array</php>, meaning the behaviour depends on the offset type. Please see the <php>array</php> section for details.+ 
 +  * For read-write and increment/decrement operations, the container is converted to array, before the read operation. And thus behave like an <php>array</php>, meaning the behaviour depends on the offset type. Please see the <php>array</php> section for details. 
   * For the unset operation, the container continues to be <php>null</php> and no warning or error is emitted/thrown.   * For the unset operation, the container continues to be <php>null</php> and no warning or error is emitted/thrown.
 +
   * For existence operations, no warning is emitted and the behaviour is as if the offset did not exist.   * For existence operations, no warning is emitted and the behaviour is as if the offset did not exist.
  
Line 158: Line 160:
  
 Therefore, the behaviour depending on the operator is as follows: Therefore, the behaviour depending on the operator is as follows:
-  * For read operations,<php>null</php> is returned, the container continues to be <php>false</php>, and the following warning is emitted: <code> +  * For read operations, <php>null</php> is returned, the container continues to be <php>false</php>, and the following warning is emitted: <code>Warning: Trying to access array offset on false</code>
-Warning: Trying to access array offset on false +
-</code>+
  
-  * For write, append, fetch, and fetch-append operations the container is converted to array, Emitting the following deprecation notice: <code> +  * For write, append, fetch, and fetch-append operations the container is converted to array, Emitting the following deprecation notice:<code>Deprecated: Automatic conversion of false to array is deprecated</code> And thus behave like an <php>array</php>, meaning the behaviour depends on the offset type. Please see the <php>array</php> section for details.
-Deprecated: Automatic conversion of false to array is deprecated +
-</code> And thus behave like an <php>array</php>, meaning the behaviour depends on the offset type. Please see the <php>array</php> section for details.+
  
-  * For read-write operations, the container is converted to array, before the read operation, Emitting the following deprecation notice: <code> +  * For read-write and increment/decrement operations, the container is converted to array, before the read operation, 
-Deprecated: Automatic conversion of false to array is deprecated +  Emitting the following deprecation notice: <code>Deprecated: Automatic conversion of false to array is deprecated</code> And thus behave like an <php>array</php>, meaning the behaviour depends on the offset type.  Please see the <php>array</php> section for details.
-</code> And thus behave like an <php>array</php>, meaning the behaviour depends on the offset type. Please see the <php>array</php> section for details.+
  
-  * For the unset operation, the container continues to be <php>false</php> and the following deprecation notice is emitted: <code> +  * For the unset operation, the container continues to be <php>false</php> and the following deprecation notice is emitted: <code>Deprecated: Automatic conversion of false to array is deprecated</code>
-Deprecated: Automatic conversion of false to array is deprecated +
-</code>+
  
   * For existence operations, no warning is emitted and the behaviour is as if the offset did not exist.   * For existence operations, no warning is emitted and the behaviour is as if the offset did not exist.
Line 247: Line 242:
  
 Moreover, some operations are invalid on string offsets: Moreover, some operations are invalid on string offsets:
-  * Read-Write operations on a string offset will throw the following error: <code> +  * Read-Write operations on a string offset will throw the following error: <code>Cannot use assign-op operators with string offsets</code> 
-Cannot use assign-op operators with string offsets + 
-</code> +  * Unset operations on a string offset will throw the following error: <code>Cannot unset string offsets</code> 
-  * Unset operations on a string offset will throw the following error: <code> + 
-Cannot unset string offsets +  * The append and fetch-append operations will throw the following error: <code>[] operator not supported for strings</code> 
-</code> +
-  * The append and fetch-append operations will throw the following error: <code> +
-[] operator not supported for strings +
-</code>+
   * Fetch operations will throw different errors depending on the fetch operation, //after// the type of the offset has been checked:   * Fetch operations will throw different errors depending on the fetch operation, //after// the type of the offset has been checked:
-  * For attempting to retrieve a reference to a string offset: <code> +    * For attempting to retrieve a reference to a string offset: <code>Cannot create references to/from string offsets</code> 
-Cannot create references to/from string offsets +    * For attempting to use the string offset as a container: <code>Cannot use string offset as an array</code> 
-</code> +    * For attempting to use the string offset as an object: <code>Cannot use string offset as an object</code> 
-  * For attempting to use the string offset as a container: <code> +    * For attempting to use increment or decrement the string offset: <code>Cannot increment/decrement string offsets</code>
-Cannot use string offset as an array +
-</code> +
-  * For attempting to use the string offset as an object: <code> +
-Cannot use string offset as an object +
-</code> +
-  * For attempting to use increment or decrement the string offset: <code> +
-Cannot increment/decrement string offsets +
-</code>+
  
 Attempting to read a non initialized string offset emits the following warning: Attempting to read a non initialized string offset emits the following warning:
Line 376: Line 360:
 </PHP> </PHP>
 results in the following output: results in the following output:
-<code>text+<code>
 bool(false) bool(false)
 bool(true) bool(true)
Line 434: Line 418:
 for determining if the value is falsy or not. for determining if the value is falsy or not.
 This is error-prone, and indeed <php>PDORow</php> did not implement the logic for handling calls to <php>empty()</php> This is error-prone, and indeed <php>PDORow</php> did not implement the logic for handling calls to <php>empty()</php>
-properly. [1:https://github.com/php/php-src/pull/13512]+properly. ((https://github.com/php/php-src/pull/13512))
  
 One other requirement of the <php>has_dimension</php> is to return <php>false</php> if the offset exists but the value One other requirement of the <php>has_dimension</php> is to return <php>false</php> if the offset exists but the value
Line 477: Line 461:
 The complexity of these requirements for the <php>read_dimension</php> handler are generally not understood, The complexity of these requirements for the <php>read_dimension</php> handler are generally not understood,
 and was the source of a bug in <php>PDORow</php> which did a <php>NULL</php> pointer dereference for fetch-append operations. and was the source of a bug in <php>PDORow</php> which did a <php>NULL</php> pointer dereference for fetch-append operations.
-[1:https://github.com/php/php-src/pull/13512]+((https://github.com/php/php-src/pull/13512))
  
 The only extension that properly implements all this complexity is SimpleXML The only extension that properly implements all this complexity is SimpleXML
Line 496: Line 480:
 One additional pitfall that is common to all dimension handlers is the need to call <php>ZVAL_DEREF()</php> One additional pitfall that is common to all dimension handlers is the need to call <php>ZVAL_DEREF()</php>
 on the offset <php>zval*</php> so that when PHP references are used they work properly. on the offset <php>zval*</php> so that when PHP references are used they work properly.
-This requirement wasn't followed by <php>DOMNodeMap</php> and <php>DOMNodeList</php> [1:https://github.com/php/php-src/pull/13511]+This requirement wasn't followed by <php>DOMNodeMap</php> and <php>DOMNodeList</php> ((https://github.com/php/php-src/pull/13511))
-<php>ResourceBundle</php> [1:https://github.com/php/php-src/pull/13503]+<php>ResourceBundle</php> ((https://github.com/php/php-src/pull/13503))
-and <php>PDORow</php> [1:https://github.com/php/php-src/pull/13512].+and <php>PDORow</php> ((https://github.com/php/php-src/pull/13512)).
 Moreover, some extensions do dereference the offset, but only indirectly, and it is not know if Moreover, some extensions do dereference the offset, but only indirectly, and it is not know if
 this was done on purpose or happens to work, for example <php>FFI\CData</php> dereferences them via the call to this was done on purpose or happens to work, for example <php>FFI\CData</php> dereferences them via the call to
Line 513: Line 497:
  
   * Read: the <php>ArrayAccess::offsetGet($offset)</php> method is called with <php>$offset</php> being equal to the value between <php>[]</php>   * Read: the <php>ArrayAccess::offsetGet($offset)</php> method is called with <php>$offset</php> being equal to the value between <php>[]</php>
 +
   * Write: the <php>ArrayAccess::offsetSet($offset, $value)</php> method is called with <php>$offset</php> being equal to the value between <php>[]</php> and <php>$value</php> being the value that is being assigned to the offset.    * Write: the <php>ArrayAccess::offsetSet($offset, $value)</php> method is called with <php>$offset</php> being equal to the value between <php>[]</php> and <php>$value</php> being the value that is being assigned to the offset. 
 +
   * Read-Write: the <php>ArrayAccess::offsetGet($offset)</php> method is called with <php>$offset</php> being equal to the value between <php>[]</php>, the binary operation is then performed, and if the binary operation succeeds the <php>ArrayAccess::offsetSet($offset, $value)</php> method is called with <php>$value</php> being the result of the binary operation    * Read-Write: the <php>ArrayAccess::offsetGet($offset)</php> method is called with <php>$offset</php> being equal to the value between <php>[]</php>, the binary operation is then performed, and if the binary operation succeeds the <php>ArrayAccess::offsetSet($offset, $value)</php> method is called with <php>$value</php> being the result of the binary operation 
 +
   * Appending: the <php>ArrayAccess::offsetSet($offset, $value)</php> method is called with <php>$offset</php> being equal to <php>null</php> and <php>$value</php> being the value that is being appended to the container.   * Appending: the <php>ArrayAccess::offsetSet($offset, $value)</php> method is called with <php>$offset</php> being equal to <php>null</php> and <php>$value</php> being the value that is being appended to the container.
 +
   * Unsetting: the <php>ArrayAccess::offsetUnset($offset)</php> method is called with <php>$offset</php> being equal to the value between <php>[]</php>   * Unsetting: the <php>ArrayAccess::offsetUnset($offset)</php> method is called with <php>$offset</php> being equal to the value between <php>[]</php>
 +
   * Existence checks via isset(): the <php>ArrayAccess::offsetExists($offset)</php> method is called with <php>$offset</php> being equal to the value between <php>[]</php>   * Existence checks via isset(): the <php>ArrayAccess::offsetExists($offset)</php> method is called with <php>$offset</php> being equal to the value between <php>[]</php>
 +
   * Existence checks via empty(): the <php>ArrayAccess::offsetExists($offset)</php> method is called with <php>$offset</php> being equal to the value between <php>[]</php> if <php>true</php> is returned, a call to <php>ArrayAccess::offsetGet($offset)</php> is made to check the value is falsy or not.   * Existence checks via empty(): the <php>ArrayAccess::offsetExists($offset)</php> method is called with <php>$offset</php> being equal to the value between <php>[]</php> if <php>true</php> is returned, a call to <php>ArrayAccess::offsetGet($offset)</php> is made to check the value is falsy or not.
 +
   * Existence checks via the null coalesce operator <php>??</php>: the <php>ArrayAccess::offsetExists($offset)</php> method is called with <php>$offset</php> being equal to the value between <php>[]</php> if <php>true</php> is returned, a call to <php>ArrayAccess::offsetGet($offset)</php> is made to retrieve the value. (Note this is handled by the default <php>read_dimension</php> object handler instead of the <php>has_dimension</php> handler)   * Existence checks via the null coalesce operator <php>??</php>: the <php>ArrayAccess::offsetExists($offset)</php> method is called with <php>$offset</php> being equal to the value between <php>[]</php> if <php>true</php> is returned, a call to <php>ArrayAccess::offsetGet($offset)</php> is made to retrieve the value. (Note this is handled by the default <php>read_dimension</php> object handler instead of the <php>has_dimension</php> handler)
 +
   * Fetch: the <php>ArrayAccess::offsetGet($offset)</php> method is called with <php>$offset</php> being equal to the value between <php>[]</php>   * Fetch: the <php>ArrayAccess::offsetGet($offset)</php> method is called with <php>$offset</php> being equal to the value between <php>[]</php>
 +
   * Fetch Append: the <php>ArrayAccess::offsetGet($offset)</php> method is called with <php>$offset</php> being equal to <php>null</php>   * Fetch Append: the <php>ArrayAccess::offsetGet($offset)</php> method is called with <php>$offset</php> being equal to <php>null</php>
 +
 +  * Increment/Decrement: behaves like a fetch operation
  
 Because <php>ArrayAccess::offsetGet($offset)</php> is called for fetching operations, if it does not return an object or by-reference, Because <php>ArrayAccess::offsetGet($offset)</php> is called for fetching operations, if it does not return an object or by-reference,
 the following notice is emitted: the following notice is emitted:
-<code>text+<code>
 Notice: Indirect modification of overloaded element of ClassName has no effect in %s on line %d Notice: Indirect modification of overloaded element of ClassName has no effect in %s on line %d
 </code> </code>
Line 615: Line 610:
  
 $o = new T(); $o = new T();
-$a = new ArrayObject(); 
- 
 $a = new ArrayObject($o); $a = new ArrayObject($o);
 var_dump(isset($a['p'])); var_dump(isset($a['p']));
Line 623: Line 616:
  
 results in the following behaviour: results in the following behaviour:
-<code>text+<code>
 bool(true) bool(true)
  
Line 687: Line 680:
  
   * Call method to verify the offset exists:   * Call method to verify the offset exists:
-  * If it does not exist: return <php>false</php> (<php>true</php> for <php>empty()</php>+    * If it does not exist: return <php>false</php> (<php>true</php> for <php>empty()</php>
-  * Otherwise: call method to get value of offset:+    * Otherwise: call method to get value of offset:
     - If the value is <php>null</php> (or falsy for <php>empty()</php>) return <php>false</php> (<php>true</php> for <php>empty()</php>)     - If the value is <php>null</php> (or falsy for <php>empty()</php>) return <php>false</php> (<php>true</php> for <php>empty()</php>)
     - Otherwise: return <php>true</php> (<php>false</php> for <php>empty()</php>)     - Otherwise: return <php>true</php> (<php>false</php> for <php>empty()</php>)
Line 737: Line 730:
 and if it doesn't it can lead to unintuitive semantics if the handler considers <php>null</php> to be set. and if it doesn't it can lead to unintuitive semantics if the handler considers <php>null</php> to be set.
 These semantics also preventing the widening of the <php>$array</php> parameter type of <php>array_key_exists()</php> to These semantics also preventing the widening of the <php>$array</php> parameter type of <php>array_key_exists()</php> to
-accept objects that support accessing offsets, something that has been requested by userland. [1:https://externals.io/message/122435]+accept objects that support accessing offsets, something that has been requested by userland. ((https://externals.io/message/122435))
  
 Needing to handle <php>empty()</php> suffers most of the same implementation pitfalls and unintuitive semantics if the handler considers non-falsy things empty. Needing to handle <php>empty()</php> suffers most of the same implementation pitfalls and unintuitive semantics if the handler considers non-falsy things empty.
Line 809: Line 802:
 Cross-version compatible code can use DNF types to type their input arguments, e.g: Cross-version compatible code can use DNF types to type their input arguments, e.g:
 <PHP> <PHP>
-function foo(ArrayAccess|(DimensionReadable&DimensionWritable)) { +function foo(ArrayAccess|(DimensionFetchable&DimensionWritable)) { 
-    /// Do something useful ///+    /Do something useful */
 } }
 </PHP> </PHP>
Line 821: Line 814:
 Therefore, we move the handlers out of the <php>zend_object_handlers</php> structure and into the <php>zend_class_entry</php> structure. Therefore, we move the handlers out of the <php>zend_object_handlers</php> structure and into the <php>zend_class_entry</php> structure.
 We add new handlers which correspond to the above interfaces which are all defined in a new struct: We add new handlers which correspond to the above interfaces which are all defined in a new struct:
-<code>c+<code>
 typedef struct _zend_class_dimensions_functions { typedef struct _zend_class_dimensions_functions {
  /* rv is a slot provided by the callee that is returned */  /* rv is a slot provided by the callee that is returned */
Line 827: Line 820:
  bool  (*has_dimension)(zend_object *object, zval *offset);  bool  (*has_dimension)(zend_object *object, zval *offset);
  zval *(*fetch_dimension)(zend_object *object, zval *offset, zval *rv);  zval *(*fetch_dimension)(zend_object *object, zval *offset, zval *rv);
- void  (*write_dimension)(zend_object *object, zval *offset, zval //value);+ void  (*write_dimension)(zend_object *object, zval *offset, zval *value);
  void  (*append)(zend_object *object, zval *value);  void  (*append)(zend_object *object, zval *value);
  zval *(*fetch_append)(zend_object *object, zval *rv);  zval *(*fetch_append)(zend_object *object, zval *rv);
Line 854: Line 847:
  
 Another consequence of using the new algorithm is that some idiosyncratic code that produces side effects Another consequence of using the new algorithm is that some idiosyncratic code that produces side effects
-in the <php>had_dimension</php> handler might not work as before,+in the <php>has_dimension</php> handler might not work as before,
 this also applies to userland classes implementing <php>ArrayAccess</php>. this also applies to userland classes implementing <php>ArrayAccess</php>.
 For example, the following code: For example, the following code:
Line 886: Line 879:
  
 As the <php>offsetExists()</php> wasn't called before, but now is. As the <php>offsetExists()</php> wasn't called before, but now is.
 +
 +== Removal of the zend_class_arrayaccess_funcs struct and CE pointer ==
 +
 +As the <php>zend_class_arrayaccess_funcs</php> struct was only used by SPL,
 +and it cannot fulfill its role anymore with the new dimension handlers,
 +the struct is removed and alongside it the pointer to such a struct on the <php>zend_class_entry</php>.
 +
 +== Throw an Error when trying to increment or decrement an object offset ==
 +
 +Incrementing/decrementing an object offset results in fetch operation,
 +whereas using <php>+=</php>/<php>-=</php> uses a Read-Write sequence,
 +and usually when acting on object properties this is what happens.
 +
 +The current limitation is because we do not have sufficient specialized VM opcodes for this specific case.
 +Therefore, we propose to hard error, like we do for string offsets, so that if we do add the relevant opcodes we can
 +properly support the Read-Write behaviour.
  
 == Changes to ArrayObject == == Changes to ArrayObject ==
Line 896: Line 905:
   * Fix <php>null</php> offset handling (following from the proper support of the appending operation)   * Fix <php>null</php> offset handling (following from the proper support of the appending operation)
   * When using an object as a backing value:   * When using an object as a backing value:
-  * Throw <php>Error</php> on appending +    * Throw <php>Error</php> on appending 
-  * Emit dynamic properties warning when using an object as a backing value that does not allow dynamic properties +    * Emit dynamic properties warning when using an object as a backing value that does not allow dynamic properties 
-  * Throw <php>Error</php> on writing to <php>readonly</php> properties +    * Throw <php>Error</php> on writing to <php>readonly</php> properties 
-  * Throw <php>Error</php> on writing a value of the wrong type to a typed property+    * Throw <php>Error</php> on writing a value of the wrong type to a typed property 
 +    * Throw <php>Error</php> when using an <php>int</php> as a property
   * Continue to ignore any <php>__set()</php>/<php>__get()</php> magic methods   * Continue to ignore any <php>__set()</php>/<php>__get()</php> magic methods
  
Line 910: Line 920:
 but it also "supports" appending, fetching, and fetch-appending. but it also "supports" appending, fetching, and fetch-appending.
  
-Our solution is to add legacy dimension handlers to classes that implement <php>ArrayAccess</php> +The solution we came up with is for <php>ArrayAccess</php> to formally extend 
-reproducing the current behaviour for appending, fetching and fetch-appending.+<php>DimensionReadable</php>, <php>DimensioWriteable</php>, and <php>DimensionUnsettable</php>; 
 +but also add legacy dimension handlers reproducing the current behaviour when appending, fetchingand fetch-appending 
 +such an object.
 However, if one of the new interfaces is implemented for dedicated support to appending, fetching, However, if one of the new interfaces is implemented for dedicated support to appending, fetching,
 and fetch-appending, then the new behaviour is used. and fetch-appending, then the new behaviour is used.
Line 931: Line 943:
 As such we propose to deprecate the aliases in favour of the normal offset methods. As such we propose to deprecate the aliases in favour of the normal offset methods.
  
-== Changes to MultipleIterator==+== Changes to MultipleIterator ==
  
 The implementation of <php>MultipleIterator</php> shares the same internal object handlers as <php>SplObjectStorage</php>. The implementation of <php>MultipleIterator</php> shares the same internal object handlers as <php>SplObjectStorage</php>.
Line 957: Line 969:
 The <php>array_key_exists()</php> function, and any objects mimicking array offsets, is also affected and would have the <php>resource</php> The <php>array_key_exists()</php> function, and any objects mimicking array offsets, is also affected and would have the <php>resource</php>
 type removed from the union type for the <php>$key</php> parameter. type removed from the union type for the <php>$key</php> parameter.
 +
 +Moreover, most code that expects resources as offsets already use an explicit <php>(int)</php> cast to suppress the warnings.
  
 == Emit warnings for invalid offset types on arrays == == Emit warnings for invalid offset types on arrays ==
Line 990: Line 1004:
 </code> </code>
  
-=== Emit warning on read-write operations on <php>null</php> container ===+=== Emit warning on read-write operations on null container ===
  
 Emit the same warning as a simple read operation when using <php>null</php> as a container: Emit the same warning as a simple read operation when using <php>null</php> as a container:
Line 1012: Line 1026:
 </code> </code>
  
-And if the specific operation is not supported the error would ressemble:+And if the specific operation is not supported the error would resemble:
  
-<code>text+<code>
 Cannot OPERATION offset of type TYPE on value of type TYPE Cannot OPERATION offset of type TYPE on value of type TYPE
 </code> </code>
Line 1038: Line 1052:
  
 Promote all warnings to <php>Error</php> Promote all warnings to <php>Error</php>
 +
 +===== Backward Incompatible Changes =====
 +
 +A recap of the BC breaking changes being introduced in PHP 8.4:
 +
 +  * New algorithm when calling <php>isset()</php>
 +  * <php>resource</php> as an offset type for arrays would throw an <php>Error</php> as of PHP 8.4
 +  * Trying to read offsets of a <php>MultipleIterator</php> object would throw an <php>Error</php> as of PHP 8.4
 +  * Leading numeric string used as an offset for strings would throw an <php>Error</php> as of PHP 8.4
 +  * Float numeric strings (i.e. non integer-numeric strings) used as an offset for strings would now throw an <php>Error</php> with the null coalesce operator <php>??</php> (in line with <php>isset()</php>)
 +  * <php>ArrayObject</php> would behave more sensibly and in line with every other PHP object, rather than being weird
 +
 +A recap of the new warnings being introduced in PHP 8.4:
 +
 +  * A warning when checking the existence of an offset on invalid container types (except for <php>null</php>)
 +  * A warning prior to reading an undefined offset when it is part of a read-write operation
 +  * A warning is emitted when trying to use a value of type <php>null</php>, <php>bool</php>, or <php>float</php> as an array offset
 +  * A warning is emitted when trying to check the existence of string offset with invalid offset types
 +
 +A recap of the deprecations being introduced in PHP 8.4:
 +
 +  * The <php>SplObjectStorage::contains()</php> method is deprecated in favour of <php>SplObjectStorage::offsetExists()</php>
 +  * The <php>SplObjectStorage::detatch()</php> method is deprecated in favour of <php>SplObjectStorage::offsetUnset()</php>
 +  * The <php>SplObjectStorage::attach()</php> method is deprecated in favour of <php>SplObjectStorage::offsetSet()</php>
 +
 +For details on each of those, refer to their relevant sections.
  
 ===== Version ===== ===== Version =====
rfc/container-offset-behaviour.1720100986.txt.gz · Last modified: 2025/04/03 13:08 (external edit)