Есть ли лучший способ PHP для получения значения по умолчанию из массива (словарь)?
В Python можно сделать:
foo = {}
assert foo.get('bar', 'baz') == 'baz'
В PHP можно использовать тройной оператор, например:
$foo = array();
assert( (isset($foo['bar'])) ? $foo['bar'] : 'baz' == 'baz');
Я ищу версию для гольфа. Могу ли я сделать это короче/лучше в PHP?
Ответы
Ответ 1
Я просто придумал эту небольшую вспомогательную функцию:
function get(&$var, $default=null) {
return isset($var) ? $var : $default;
}
Это не только работает для словарей, но и для всех переменных:
$test = array('foo'=>'bar');
get($test['foo'],'nope'); // bar
get($test['baz'],'nope'); // nope
get($test['spam']['eggs'],'nope'); // nope
get($undefined,'nope'); // nope
Передача предыдущей переменной undefined для каждой ссылки не вызывает ошибки NOTICE
. Вместо этого передача $var
по ссылке определяет его и устанавливает его в null
. Значение по умолчанию также будет возвращено, если переданная переменная null
. Также обратите внимание на неявно сгенерированный массив в примере спама/яйца:
json_encode($test); // {"foo":"bar","baz":null,"spam":{"eggs":null}}
$undefined===null; // true (got defined by passing it to get)
isset($undefined) // false
get($undefined,'nope'); // nope
Обратите внимание, что хотя $var
передается по ссылке, результат get($var)
будет копией $var
, а не ссылкой. Надеюсь, это поможет!
Ответ 2
Время идет, и PHP развивается. PHP-теперь поддерживает оператор коалесцирующего нуля, ??
:
// Fetches the value of $_GET['user'] and returns 'nobody'
// if it does not exist.
$username = $_GET['user'] ?? 'nobody';
// This is equivalent to:
$username = isset($_GET['user']) ? $_GET['user'] : 'nobody';
// Coalescing can be chained: this will return the first
// defined value out of $_GET['user'], $_POST['user'], and
// 'nobody'.
$username = $_GET['user'] ?? $_POST['user'] ?? 'nobody';
Ответ 3
Используйте оператор контроля ошибок @
с краткой версией PHP 5.3 для троичного оператора:
$bar = @$foo['bar'] ?: 'defaultvalue';
Ответ 4
PHP 5.3 имеет сокращенную версию троичного оператора:
$x = $foo ?: 'defaultvaluehere';
что в основном
if (isset($foo)) {
$x = $foo;
else {
$x = 'defaultvaluehere';
}
В противном случае нет, нет более короткого метода.
Ответ 5
Мне полезно создать такую функцию:
function array_value($array, $key, $default_value = null) {
return is_array($array) && array_key_exists($key, $array) ? $array[$key] : $default_value;
}
И используйте его следующим образом:
$params = array('code' => 7777, 'name' => "Cloud Strife");
$code = array_value($params, 'code');
$name = array_value($params, 'name');
$weapon = array_value($params, 'weapon', "Buster Sword");
$materia = array_value($params, 'materia');
echo "{ code: $code, name: $name, weapon: $weapon, materia: $materia }";
Значение по умолчанию в этом случае равно null
, но вы можете настроить его на все, что вам нужно.
Надеюсь, это полезно.
Ответ 6
"Немного" хакерский способ сделать это:
<?php
$foo = array();
var_dump('baz' == $tmp = &$foo['bar']);
$foo['bar'] = 'baz';
var_dump('baz' == $tmp = &$foo['bar']);
http://codepad.viper-7.com/flXHCH
Очевидно, это не очень хороший способ сделать это. Но это удобно в других ситуациях. Например. Я часто объявляю ярлыки для переменных GET и POST:
<?php
$name =& $_GET['name'];
// instead of
$name = isset($_GET['name']) ? $_GET['name'] : null;
PS: Можно было бы назвать это "встроенным ==$_=&
специальным оператором сравнения":
<?php
var_dump('baz' ==$_=& $foo['bar']);
PPS: Ну, вы могли бы просто использовать
<?php
var_dump('baz' == @$foo['bar']);
но это еще хуже, чем оператор ==$_=&
. Знаете, людям не нравится оператор подавления ошибок.
Ответ 7
Если вы перечисляете значения по умолчанию по ключу в массиве, это можно сделать следующим образом:
$foo = array('a' => 1, 'b' => 2);
$defaults = array('b' => 55, 'c' => 44);
$foo = array_merge($defaults, $foo);
print_r($foo);
Результат:
Array
(
[b] => 2
[c] => 44
[a] => 1
)
Чем больше пар ключ/значение, которое вы перечисляете по умолчанию, тем лучше становится код-гольф.
Ответ 8
"Marc B" предложил решение использовать троичный ярлык $x = $foo?: 'defaultvaluehere';
но это все еще дает уведомления. Наверное это опечатка, может он имел ввиду?? или это было написано до выпуска PHP 7. Согласно тройному описанию:
Начиная с PHP 5.3, можно опустить среднюю часть троичного оператора. Выражение expr1?: expr3
возвращает expr1
если expr1
оценивается как TRUE
, и expr3
противном случае.
Но он не использует isset
внутри и производит уведомления. Во избежание уведомлений лучше использовать Null Coalescing Operator ??
который использует isset
внутри него. Доступно в PHP 7.
Выражение (expr1) (expr2) оценивается как expr2, если expr1 равен NULL, и expr1 в противном случае. В частности, этот оператор не выдает уведомление, если левое значение не существует, как и isset(). Это особенно полезно для ключей массива.
Пример # 5 Назначение значения по умолчанию
<?php
// Example usage for: Null Coalesce Operator
$action = $_POST['action'] ?? 'default';
// The above is identical to this if/else statement
if (isset($_POST['action'])) {
$action = $_POST['action'];
} else {
$action = 'default';
}
?>