数组:使用点表示法设置值?

时间:2011-10-21 15:21:36

标签: php arrays multidimensional-array eval

查看Kohana文档,我发现this really usefull function用于从多维数组中使用点表示法获取值,例如:

$foo = array('bar' => array('color' => 'green', 'size' => 'M'));
$value = path($foo, 'bar.color', NULL , '.');
// $value now is 'green'

我想知道是否有办法以相同的方式设置数组值:

set_value($foo, 'bar.color', 'black');

我发现这样做的唯一方法是重新构建数组表示法($ array ['bar'] ['color'])然后使用eval设置值。

任何想要避免评估的想法?

7 个答案:

答案 0 :(得分:38)

function set_val(array &$arr, $path,$val)
{
   $loc = &$arr;
   foreach(explode('.', $path) as $step)
   {
     $loc = &$loc[$step];
   }
   return $loc = $val;
}

答案 1 :(得分:9)

当然可以。

代码

function set_value(&$root, $compositeKey, $value) {
    $keys = explode('.', $compositeKey);
    while(count($keys) > 1) {
        $key = array_shift($keys);
        if(!isset($root[$key])) {
            $root[$key] = array();
        }
        $root = &$root[$key];
    }

    $key = reset($keys);
    $root[$key] = $value;
}

如何使用

$foo = array();
set_value($foo, 'bar.color', 'black');
print_r($foo);

输出

Array
(
    [bar] => Array
        (
            [color] => black
        )

)

<强> See it in action

答案 2 :(得分:5)

查看https://gist.github.com/elfet/4713488

$dn = new DotNotation(['bar'=>['baz'=>['foo'=>true]]]);

$value = $dn->get('bar.baz.foo'); // $value == true

$dn->set('bar.baz.foo', false); // ['foo'=>false]

$dn->add('bar.baz', ['boo'=>true]); // ['foo'=>false,'boo'=>true]

答案 3 :(得分:4)

这样,您可以多次将以下值设置为同一个变量。

你可以通过这两种方式(通过静态变量和引用变量):

<?php



    function static_dot_notation($string, $value)
    {
        static $return;

        $token = strtok($string, '.');

        $ref =& $return;     

        while($token !== false)
        {  
            $ref =& $ref[$token];
            $token = strtok('.');
        }


        $ref = $value;

        return $return;
    }

    $test = static_dot_notation('A.1', 'A ONE');
    $test = static_dot_notation('A.2', 'A TWO');
    $test = static_dot_notation('B.C1', 'C ONE');
    $test = static_dot_notation('B.C2', 'C TWO');
    $test = static_dot_notation('B.C.D', 'D ONE');

    var_export($test);

    /**
        array (
          'A' => 
              array (
                1 => 'A ONE',
                2 => 'A TWO',
              ),
          'B' => 
              array (
                'C1' => 'C ONE',
                'C2' => 'C TWO',
                'C' => 
                array (
                  'D' => 'D ONE',
                ),
          ),

    */



    function reference_dot_notation($string, $value, &$array)
    {
        static $return;

        $token = strtok($string, '.');

        $ref =& $return;     

        while($token !== false)
        {  

            $ref =& $ref[$token];
            $token = strtok('.');
        }

        $ref = $value;
        $array = $return;
    }

    reference_dot_notation('person.name', 'Wallace', $test2);
    reference_dot_notation('person.lastname', 'Maxters', $test2);

    var_export($test2);

    /**
        array (
          'person' => 
          array (
            'name' => 'Wallace',
            'lastname' => 'Maxters',
          ),
        )

    */

答案 4 :(得分:3)

我为此创建了一个小班级!

http://github.com/projectmeta/Stingray

$stingray = new StingRay();

//To Get value
$stingray->get($array, 'this.that.someother'):

//To Set value
$stingray->get($array, 'this.that.someother', $newValue):

答案 5 :(得分:0)

更新了@hair树脂的答案,以满足:

  • 当子路径已存在或
  • 当子路径不是数组时

    function set_val(array &$arr, $path,$val)
    {
       $loc = &$arr;
       $path = explode('.', $path);
       foreach($path as $step)
       {
           if ( ! isset($loc[$step]) OR ! is_array($loc[$step]))
               $loc = &$loc[$step];
       }
       return $loc = $val;
    }
    

答案 6 :(得分:-1)

这里没有一个示例对我有用,因此我想出了一个使用eval()的解决方案(了解here的风险,但是如果您不使用用户数据,那应该就不多了问题)。设置方法中的if子句允许您将项目推到该位置($ location [] = $ item)上的新数组或现有数组中。

class ArrayDot {
    public static function get(array &$array, string $path, string $delimiter = '.') {
        return eval("return ".self::getLocationCode($array, $path, $delimiter).";");
    }

    public static function set(array &$array, string $path, $item, string $delimiter = '.') : void {
        //if the last character is a delimiter, allow pushing onto a new or existing array
        $add = substr($path, -1) == $delimiter ? '[]': '';
        eval(self::getLocationCode($array, $path, $delimiter).$add." = \$item;");
    }

    public static function unset(array &$array, $path, string $delimiter = '.') : void {
        if (is_array($path)) {
            foreach($path as $part) {
                self::unset($array, $part, $delimiter);
            }
        }
        else {
            eval('unset('.self::getLocationCode($array, $path, $delimiter).');');
        }
    }

    public static function isSet(array &$array, $path, string $delimiter = '.') : bool {
        if (is_array($path)) {
            foreach($path as $part) {
                if (!self::isSet($array, $part, $delimiter)) {
                    return false;
                }
            }
            return true;
        }
        return eval("return isset(".self::getLocationCode($array, $path, $delimiter).");");
    }

    private static function getLocationCode(array &$array, string $path, string $delimiter) : string {
        $path = rtrim($path, $delimiter);           //Trim trailing delimiters
        $escapedPathParts = array_map(function ($s) { return str_replace('\'', '\\\'', $s); }, explode($delimiter, $path));
        return "\$array['".implode("']['", $escapedPathParts)."']";
    }
}

用法示例:

echo '<pre>';
$array = [];
ArrayDot::set($array, 'one.two.three.', 'one.two.three.');
ArrayDot::set($array, 'one.two.three.four.', 'one.two.three.four.');
ArrayDot::set($array, 'one.two.three.four.', 'one.two.three.four. again');
ArrayDot::set($array, 'one.two.three.five.', 'one.two.three.five.');
ArrayDot::set($array, 'one.two.three.direct set', 'one.two.three.direct set');

print_r($array);
echo "\n";
echo "one.two.three.direct set: ".print_r(ArrayDot::get($array, 'one.two.three.direct set'), true)."\n";
echo "one.two.three.four: ".print_r(ArrayDot::get($array, 'one.two.three.four'), true)."\n";

输出:

Array
(
    [one] => Array
        (
            [two] => Array
                (
                    [three] => Array
                        (
                            [0] => one.two.three.
                            [four] => Array
                                (
                                    [0] => one.two.three.four.
                                    [1] => one.two.three.four. again
                                )

                            [five] => Array
                                (
                                    [0] => one.two.three.five.
                                )

                            [direct set] => one.two.three.direct set
                        )

                )

        )

)

one.two.three.direct set: one.two.three.direct set
one.two.three.four: Array
(
    [0] => one.two.three.four.
    [1] => one.two.three.four. again
)