Как взорвать массив с ключом и значением без foreach в PHP
без foreach, как я могу превратить такой массив
array("item1"=>"object1", "item2"=>"object2",......."item-n"=>"object-n");
для такой строки
item1='object1', item2='object2',.... item-n='object-n'
Я думал о implode()
уже, но он не взрывает ключ с ним.
Если foreach это необходимо, можно ли не вложить foreach?
EDIT: Я изменил строку
EDIT2 / UPDATE: Этот вопрос был задан довольно давно. В то время я хотел написать все в одной строке, чтобы я использовал троичные операторы и встроенные вызовы функций в пользу foreach. Это была плохая практика! Напишите код, который читается, является ли он кратким или нет, не имеет большого значения.
в этом случае: размещение foreach в функции будет гораздо более читаемым и модульным, чем написание однострочного(хотя все ответы велики!).
10 ответов
и еще:
$input = array(
'item1' => 'object1',
'item2' => 'object2',
'item-n' => 'object-n'
);
$output = implode(', ', array_map(
function ($v, $k) {
if(is_array($v)){
return $k.'[]='.implode('&'.$k.'[]=', $v);
}else{
return $k.'='.$v;
}
},
$input,
array_keys($input)
));
или:
$output = implode(', ', array_map(
function ($v, $k) { return sprintf("%s='%s'", $k, $v); },
$input,
array_keys($input)
));
можно использовать http_build_query, например:
<?php
$a=array("item1"=>"object1", "item2"=>"object2");
echo http_build_query($a,'',', ');
?>
выход:
item1=object1, item2=object2
я провел измерения (100000 итераций), какой самый быстрый способ склеить ассоциативный массив?
цель: чтобы получить строку из 1000 элементов, в этом формате: "key: value, key2: value2"
у нас есть массив (например):
$array = [
'test0' => 344,
'test1' => 235,
'test2' => 876,
...
];
тест номер один:
использовать http_build_query и как str_replace:
str_replace('=', ':', http_build_query($array, null, ','));
среднее время взрыва 1000 элементов: 0.00012930955084904
тест номер два:
использовать использование array_map и лопаются:
implode(',', array_map(
function ($v, $k) {
return $k.':'.$v;
},
$array,
array_keys($array)
));
среднее время взрыва 1000 элементов: 0.0004890081976675
тест номер три:
использовать array_walk и лопаются:
array_walk($array,
function (&$v, $k) {
$v = $k.':'.$v;
}
);
implode(',', $array);
среднее время взрыва 1000 элементов: 0.0003874126245348
количество теста четыре:
использовать foreach:
$str = '';
foreach($array as $key=>$item) {
$str .= $key.':'.$item.',';
}
rtrim($str, ',');
среднее время взрыва 1000 элементов: 0.00026632803902445
Я могу сделать вывод, что лучший способ склеить массив - использовать http_build_query и как str_replace
Я хотел бы использовать serialize()
или json_encode()
.
хотя это не даст вам точную строку результата, которую вы хотите, было бы намного проще кодировать/хранить/извлекать/декодировать позже.
С помощью array_walk
$a = array("item1"=>"object1", "item2"=>"object2","item-n"=>"object-n");
$r=array();
array_walk($a, create_function('$b, $c', 'global $r; $r[]="$c=$b";'));
echo implode(', ', $r);
изменить
- return substr($result, (-1 * strlen($glue)));
+ return substr($result, 0, -1 * strlen($glue));
Если вы хотите использовать всю строку без последнего $ glue
function key_implode(&$array, $glue) {
$result = "";
foreach ($array as $key => $value) {
$result .= $key . "=" . $value . $glue;
}
return substr($result, (-1 * strlen($glue)));
}
и использование:
$str = key_implode($yourArray, ",");
для целей отладки. Рекурсивная запись массива вложенных массивов в строку. Использовать циклы foreach. Функция хранит символы национального языка.
function q($input)
{
$glue = ', ';
$function = function ($v, $k) use (&$function, $glue) {
if (is_array($v)) {
$arr = [];
foreach ($v as $key => $value) {
$arr[] = $function($value, $key);
}
$result = "{" . implode($glue, $arr) . "}";
} else {
$result = sprintf("%s=\"%s\"", $k, var_export($v, true));
}
return $result;
};
return implode($glue, array_map($function, $input, array_keys($input))) . "\n";
}
вы можете использовать PHP array_reduce также,
$a = ['Name' => 'Last Name'];
function acc($acc,$k)use($a){ return $acc .= $k.":".$a[$k].",";}
$imploded = array_reduce(array_keys($a), "acc");
для создания MySQL, где условия из массива
$sWheres = array('item1' => 'object1',
'item2' => 'object2',
'item3' => 1,
'item4' => array(4,5),
'item5' => array('object3','object4'));
$sWhere = '';
if(!empty($sWheres)){
$sWhereConditions = array();
foreach ($sWheres as $key => $value){
if(!empty($value)){
if(is_array($value)){
$value = array_filter($value); // For remove blank values from array
if(!empty($value)){
array_walk($value, function(&$item){ $item = sprintf("'%s'", $item); }); // For make value string type 'string'
$sWhereConditions[] = sprintf("%s in (%s)", $key, implode(', ', $value));
}
}else{
$sWhereConditions[] = sprintf("%s='%s'", $key, $value);
}
}
}
if(!empty($sWhereConditions)){
$sWhere .= "(".implode(' AND ', $sWhereConditions).")";
}
}
echo $sWhere; // (item1='object1' AND item2='object2' AND item3='1' AND item4 in ('4', '5') AND item5 in ('object3', 'object4'))
вот простой пример использования класса:
$input = array(
'element1' => 'value1',
'element2' => 'value2',
'element3' => 'value3'
);
echo FlatData::flatArray($input,', ', '=');
class FlatData
{
public static function flatArray(array $input = array(), $separator_elements = ', ', $separator = ': ')
{
$output = implode($separator_elements, array_map(
function ($v, $k, $s) {
return sprintf("%s{$s}%s", $k, $v);
},
$input,
array_keys($input),
array_fill(0, count($input), $separator)
));
return $output;
}
}