Skip to content

Commit 0943b8b

Browse files
authored
Improve performance of array_find() etc (#18157)
This avoids destruction logic for the common case, avoids some copy, and adds an optimization hint. For this script: ```php $array = range(1, 10000); $result = 0; for ($i = 0; $i < 5000; $i++) { $result += array_find($array, static function ($item) { return $item === 5000; }); } var_dump($result); ``` On an intel i7 1185G7: ``` Benchmark 1: ./sapi/cli/php x.php Time (mean ± σ): 543.7 ms ± 3.8 ms [User: 538.9 ms, System: 4.4 ms] Range (min … max): 538.4 ms … 552.9 ms 10 runs Benchmark 2: ./sapi/cli/php_old x.php Time (mean ± σ): 583.0 ms ± 4.2 ms [User: 578.4 ms, System: 3.4 ms] Range (min … max): 579.3 ms … 593.9 ms 10 runs Summary ./sapi/cli/php x.php ran 1.07 ± 0.01 times faster than ./sapi/cli/php_old x.php ``` On an intel i7 4790: ``` Benchmark 1: ./sapi/cli/php x.php Time (mean ± σ): 828.6 ms ± 4.8 ms [User: 824.4 ms, System: 1.6 ms] Range (min … max): 822.8 ms … 839.0 ms 10 runs Benchmark 2: ./sapi/cli/php_old x.php Time (mean ± σ): 940.1 ms ± 26.4 ms [User: 934.4 ms, System: 2.5 ms] Range (min … max): 918.0 ms … 981.1 ms 10 runs Summary ./sapi/cli/php x.php ran 1.13 ± 0.03 times faster than ./sapi/cli/php_old x.php ```
1 parent 370f242 commit 0943b8b

File tree

1 file changed

+27
-16
lines changed

1 file changed

+27
-16
lines changed

ext/standard/array.c

+27-16
Original file line numberDiff line numberDiff line change
@@ -6509,6 +6509,22 @@ PHP_FUNCTION(array_reduce)
65096509
}
65106510
/* }}} */
65116511

6512+
/* Consumes `zv` */
6513+
static bool php_is_true(zval *zv)
6514+
{
6515+
switch (Z_TYPE_P(zv)) {
6516+
case IS_TRUE:
6517+
return true;
6518+
case IS_FALSE:
6519+
return false;
6520+
default: {
6521+
bool rv = zend_is_true(zv);
6522+
zval_ptr_dtor(zv);
6523+
return rv;
6524+
}
6525+
}
6526+
}
6527+
65126528
/* {{{ Filters elements from the array via the callback. */
65136529
PHP_FUNCTION(array_filter)
65146530
{
@@ -6571,9 +6587,7 @@ PHP_FUNCTION(array_filter)
65716587
RETURN_THROWS();
65726588
}
65736589

6574-
bool retval_true = zend_is_true(&retval);
6575-
zval_ptr_dtor(&retval);
6576-
if (!retval_true) {
6590+
if (!php_is_true(&retval)) {
65776591
continue;
65786592
}
65796593
} else if (!zend_is_true(operand)) {
@@ -6597,7 +6611,7 @@ enum php_array_find_result {
65976611
PHP_ARRAY_FIND_SOME = 1,
65986612
};
65996613

6600-
static enum php_array_find_result php_array_find(const HashTable *array, zend_fcall_info fci, zend_fcall_info_cache fci_cache, zval *result_key, zval *result_value, bool negate_condition)
6614+
static enum php_array_find_result php_array_find(const HashTable *array, zend_fcall_info fci, zend_fcall_info_cache *fci_cache, zval *result_key, zval *result_value, bool negate_condition)
66016615
{
66026616
zend_ulong num_key;
66036617
zend_string *str_key;
@@ -6620,25 +6634,22 @@ static enum php_array_find_result php_array_find(const HashTable *array, zend_fc
66206634
if (!str_key) {
66216635
ZVAL_LONG(&args[1], num_key);
66226636
} else {
6637+
/* Allows copying the numeric branch, without this branch, into the iteration code
6638+
* that checks for the packed array flag. */
6639+
ZEND_ASSUME(!HT_IS_PACKED(array));
66236640
ZVAL_STR(&args[1], str_key);
66246641
}
66256642

66266643
ZVAL_COPY_VALUE(&args[0], operand);
66276644

6628-
zend_result result = zend_call_function(&fci, &fci_cache);
6645+
zend_result result = zend_call_function(&fci, fci_cache);
66296646
ZEND_ASSERT(result == SUCCESS);
66306647

66316648
if (UNEXPECTED(Z_ISUNDEF(retval))) {
66326649
return PHP_ARRAY_FIND_EXCEPTION;
66336650
}
66346651

6635-
bool retval_true = zend_is_true(&retval);
6636-
zval_ptr_dtor(&retval);
6637-
6638-
/* This negates the condition, if negate_condition is true. Otherwise it does nothing with `retval_true`. */
6639-
retval_true ^= negate_condition;
6640-
6641-
if (retval_true) {
6652+
if (php_is_true(&retval) ^ negate_condition) {
66426653
if (result_value != NULL) {
66436654
ZVAL_COPY_DEREF(result_value, &args[0]);
66446655
}
@@ -6667,7 +6678,7 @@ PHP_FUNCTION(array_find)
66676678
Z_PARAM_FUNC(fci, fci_cache)
66686679
ZEND_PARSE_PARAMETERS_END();
66696680

6670-
php_array_find(array, fci, fci_cache, NULL, return_value, false);
6681+
php_array_find(array, fci, &fci_cache, NULL, return_value, false);
66716682
}
66726683
/* }}} */
66736684

@@ -6683,7 +6694,7 @@ PHP_FUNCTION(array_find_key)
66836694
Z_PARAM_FUNC(fci, fci_cache)
66846695
ZEND_PARSE_PARAMETERS_END();
66856696

6686-
php_array_find(array, fci, fci_cache, return_value, NULL, false);
6697+
php_array_find(array, fci, &fci_cache, return_value, NULL, false);
66876698
}
66886699
/* }}} */
66896700

@@ -6699,7 +6710,7 @@ PHP_FUNCTION(array_any)
66996710
Z_PARAM_FUNC(fci, fci_cache)
67006711
ZEND_PARSE_PARAMETERS_END();
67016712

6702-
RETURN_BOOL(php_array_find(array, fci, fci_cache, NULL, NULL, false) == PHP_ARRAY_FIND_SOME);
6713+
RETURN_BOOL(php_array_find(array, fci, &fci_cache, NULL, NULL, false) == PHP_ARRAY_FIND_SOME);
67036714
}
67046715
/* }}} */
67056716

@@ -6715,7 +6726,7 @@ PHP_FUNCTION(array_all)
67156726
Z_PARAM_FUNC(fci, fci_cache)
67166727
ZEND_PARSE_PARAMETERS_END();
67176728

6718-
RETURN_BOOL(php_array_find(array, fci, fci_cache, NULL, NULL, true) == PHP_ARRAY_FIND_NONE);
6729+
RETURN_BOOL(php_array_find(array, fci, &fci_cache, NULL, NULL, true) == PHP_ARRAY_FIND_NONE);
67196730
}
67206731
/* }}} */
67216732

0 commit comments

Comments
 (0)