Как взорвать массив с ключом и значением без 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);

IDEONE


изменить

-    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;
    }

}