This is an old revision of the document!
PHP RFC: array_first() and array_last()
- Version: 0.9
- Date: 2025-03-31
- Author: Niels Dossche, nielsdos@php.net
- Status: Draft
- First Published at: https://wiki.php.net/rfc/array_first_last
Introduction
In PHP 7.3, we got array_key_first() and array_key_last() to get the first and last keys from an array. What we don't have yet is a way to get the first and last values of an array. This is harder than you may think because:
- the array keys are not necessarily integers, not necessarily starting with 0, etc...
- existing “tricks” like reset() and end() are semantically the wrong approach because they modify the “internal iterator” of the array
- Using
'$array[array_key_first($array)]
' is cumbersome
Proposal
If the array value to be returned is a reference, it is dereferenced automatically.
Behaviour on empty arrays
Should it throw on an empty array or should it return NULL? Theoretically NULL can be a valid value from an array, and therefore returning NULL does not allow distinguishing between an empty array and a NULL value. However, there are still strong arguments in favor of using NULL:
- Consistent with
'$array[array_key_first($array)]
' - Consistent with common framework helpers like
'Arr::first()
' - NULL is a rare legitimate value, so the potential for clashing is low
- Works nicely with “??” operator
- If it were to throw an exception instead, then you would need to check the size of the array upfront. Similarly, if NULL is a legit value then you can just check the array size upfront too.
Another interesting idea is to add an optional “$default” argument (i.e. if provided return the default; otherwise throw). However, since PHP arrays are untyped it's often hard to define something meaningful as a sentinel value. There is also no precedent for this design choice for the array functions.
Naming
Why array_first/array_last instead of array_value_first/array_value_last? First, this is consistent with how (most) array functions are named: the ones that work on keys have “key” in the name, and the ones that work on values have “value” in the name. Think about array_find, array_find_key, etc. Second, the shorter name is less verbose and intuitively at least I understood this is about the value.
Fibers
In the https://externals.io/message/121313 there was some confusion on how this interacted with Fibers. TL;DR: The concern was that if you have a call for array_key_first() followed by array_first() that the result could be inconsistent if a Fiber interrupted the execution. This is however based on the false premise that Fibers are threads, in fact they are not. This situation cannot happen.
Why include this in the engine rather than userland?
Three main reasons:
- Consistency with array_key_first()/array_key_last().
- The cost/maintenance burden is practically zero, the implementation is stupidly simple (just a few lines).
- Very fast implementation.
Backward Incompatible Changes
If anyone defined array_first() or array_last() in their global scope, then they need to guard it now with function_exists or throw it away because of the name clash. Although it should be noted that the global namespace is normally reserved for PHP's usage.
Proposed PHP Version(s)
Next PHP 8.x, i.e. PHP 8.5 at the time of writing.
RFC Impact
To Existing Extensions
These two new functions are added to ext-standard, where array_key_first() and array_key_last() also live.
Unaffected PHP Functionality
No changes to existing PHP functionality.
Proposed Voting Choices
2/3rd yes/no vote.
Patches and Tests
TODO
Implementation
After the project is implemented, this section should contain
- the version(s) it was merged into
- a link to the git commit(s)
- a link to the PHP manual entry for the feature
- a link to the language specification section (if any)
References
Current discussion: TODO
Prior Work
- Proposal from 2023 discussing these new functions: https://externals.io/message/121313
- RFC that introduced array_key_first() and array_key_last(): https://wiki.php.net/rfc/array_key_first_last
Rejected Features
None yet.