This is an old revision of the document!
Request for Comments: array_part
- Version: 1.0
- Date: 2012-05-14
- Author: Gustavo Lopes cataphract@php.net
- Status: Under Discussion
- First Published at: http://wiki.php.net/rfc/array_part
Introduces a new function array_part()
in ext/standard.
Introduction
This RFC proposes a new array function that can extract multidimensional slices for arrays.
Specification for the function
The function array_part()
shall have the following signature:
array array_part(array $originalArray, array $partSpecification[, bool $indexesAreKeys = false])
The parameter $originalArray
represents the original array from which a part is to be extracted.
The parameter $partSpecification
is a sequentially indexed numeric array that specifies the slice to extract at each “level” of the original array. A level n (with n >= 0) of an array $arr
is the set of array elements than can be fetched by “dereferencing” the array n times. For instance, the array [[1],[['a','b']]]
has at level 0 itself, at level 1 the elements [1]
and [['a','b']]
and at level 2 the elements 1
and ['a','b']
. $partSpecification[0]
shall give the part specification for level 1, and, in, general, $partSpecification[m]
for level m + 1.
Each part specification shall be of one the following forms:
- An sequentially indexed numeric array of indexes, which specifies that only the elements existing at those indexes will be kept.
- A single index, which specifies that only the element existing at that index will be kept. In this case the level will be collapsed onto the previous one, meaning all the arrays at that level will be replaced with its element the specified index.
- A span part specification is an associative array. The following keys are allowed start, end and step. At least start or end must be specified. start and end are an index or a special value --
null
. step is a non-zero integer. Ifstart
orend
are not specified, they default tonull
, which refer to either the first or last element depending on the sign of step. If step is +-1 and start and end are bothnull
(or, if$indexesAreKeys
isfalse
, start and end are 0 and -1 or vice-versa depending on the sign of step), then we say the span encompasses are all elements on that level (possibly 0).
Span parts extract any number of elements (possibly 0) starting at the index specified by start and advancing until end is reached, advancing in steps of step. The element at the index specified at end is included.
An index i has the following meaning:
- If
$indexesAreKeys
isfalse
and i is a non-negative integer (possibly after casting) then an element is at index i if it i is the the number of times one would have to callnext()
after callingreset()
forcurrent()
to return that element. - If
$indexesAreKeys
isfalse
and i is a negative number, then an element is at positive i if -i+1 is the number of times one would have to callprev()
after callingend()
forcurrent()
to return that element. - If
$indexesAreKeys
istrue
then an element is at i if its array key is i.
This function shall return false
upon finding error conditions. The following are error conditions:
- Giving arguments with different PHP types from those specified.
- Giving malformed part specifications (where the individual elements do not follow the syntactic rules given here).
- Specifying an part index than does not exist at at least of the elements at the level the part refers to: (e.g.
array_part([[1],[1,2]], [['start'=>0, 'end'=>-1], 1])
is an error condition, because while1
is a valid index for[1,2]
, it is not so for1
). However, span part specifications that comprise all elements are always accepted. - Giving a part specification with levels that do not exist in the input. This does not apply if, as a result of a span part that comprises all the elements of that level and at the level before existing only empty arrays, no elements were left for the next element. For instance 'array_part({}, [['start'=>0, 'end'=>-1]])' is valid, as is 'array_part([[]], [0, ['start'=>0, 'end'=>-1], 1, 2, 3])', which will return
[[]]
. - Encountering already visited array elements from the original array (recursion).
Sample implementation
A sample implementation, with tests exemplifying the use of this function, is available.
Objections
I want native array slicing through a new operator
That's fine, but it is not what this proposal is about. A change to the language presents many more complications which I very much want to avoid -- for instance the current array dereferencing syntax most likely would unusable because -1 represents the element with key -1, not the last element. Besides, the introduction of this function does not prevent native array slicing from being added in the future.
You seem to already have an implementation, why are you pushing this to us?
First, this function would benefit greatly from a native implementation. It's impossible to efficiently detect recursion in userland. The sample makes heavy use of references, which introduces a lot of separations. Traversing arrays without changing the internal pointer is very inefficient in userland.
Of course, the main reason is that this function is useful. Recently, functions like array_column()
, array_first()
and array_last()
have been proposed. This function satisfies all those needs: array_part($arr, [['start'=>null], 'column'], true)'',
array_part($arr, 0) and
array_part($arr, -1).
The $indexesAreKeys = false mode is inefficient because there's no constant time access to the n-th element
This is true. We must traverse the array from the start or the end to get to the n-th element. That's just the way PHP arrays are implemented. However, if you have numeric sequential arrays, you can use
$indexesAreKeys = true'' to access the n-th element without this penalty.
Changelog
- 2012-05-14 Initial version