如果值为空数组,http_build_query将忽略该键。这怎么不是一个bug?

时间:2010-05-28 16:05:50

标签: php

今天我遇到了一个问题,我将一个设置为空数组的键传递给http_build_query()。 E.g:

$args = array("foo", "bar", array(), "baz");
$qs = http_build_query($args);
echo $qs; // outputs 0=foo&1=bar&3=baz, I expected 0=foo&1=bar&2=&3=baz

这给我带来了一个问题,因为我通过http将一些数据传递给内部API,我需要将所有参数拉出另一端。

好吧,我用Google搜索了这个并提出了以下错误报告:http://bugs.php.net/bug.php?id=50407。来自管理员的简短回复是,“不设置与设置为空相同。没有错误。”

有人可以向我解释为什么这不是一个错误吗?有没有人对解决方法有任何想法,除了在一方设置任意值并将该值解释为另一方的空值的蹩脚黑客?

修改

这就是为什么我认为这是一个错误:

$args = array("foo", "bar", array(), "baz");
$qs = http_build_query($args);
parse_str($qs, $query);
echo ($args == $query); // false, I expect it to be true

我想也许我认为parse_str()http_build_query()是彼此的反转是天真的。

我正在发布我目前的“蹩脚黑客”解决方案作为下面的答案。

5 个答案:

答案 0 :(得分:3)

我重新实现了http_build_query以在返回的查询字符串中留下空对象/数组(后缀为'='符号)。我也从默认功能中增强了一点,所以总而言之:

  • 维护空对象和数组
  • 将默认enc_type更改为RFC3986(与年龄相关)
  • 添加了键值分隔符参数(覆盖默认值'='的能力)
  • 删除数字索引键值对的数字索引

我没有在生产环境中对此进行过测试(不了解性能或错误),并且没有进行优化,但是非常清楚。

function buildQuery($input,$numeric_prefix='',
        $arg_separator='&',$enc_type=2,
        $keyvalue_separator='=',$prefix='') {
    if ( is_array($input) ) {
        $arr = array();
        foreach ( $input as $key => $value ) {
            $name = $prefix;
            if ( strlen($prefix) ) {
                $name .= '[';
                if ( !is_numeric($key) ) {
                    $name .= $key;
                }
                $name .= ']';
            } else {
                if ( is_numeric($key) ) {
                    $name .= $numeric_prefix;
                }
                $name .= $key;
            }
            if ( (is_array($value) || is_object($value)) && count($value) ) {
                $arr[] = buildQuery($value,$numeric_prefix,
                        $arg_separator,$enc_type,
                        $keyvalue_separator,$name);
            } else {
                if ( $enc_type === 2 ) {
                    $arr[] = rawurlencode($name)
                        .$keyvalue_separator
                        .rawurlencode($value?:'');
                } else {
                    $arr[] = urlencode($name)
                        .$keyvalue_separator
                        .urlencode($value?:'');
                }
            }
        }
        return implode($arg_separator,$arr);
    } else {
        if ( $enc_type === 2 ) {
            return rawurlencode($input);
        } else {
            return urlencode($input);
        }
    }
}

示例:

$arr = array(
        'hello' => 'world',
        'colors' => array('red','green','blue'),
        'emptyArr' => array(),
        'nested' => array(
            'empty' => array(),
            'fruits' => array('orange','banana','apple'),
            'curly' => 'sue',
            'assoc' => array('a'=>'alpha','b'=>'bravo','c'=>'charlie')
        )
    );

echo buildQuery($arr);

输出:hello=world&colors%5B%5D=red&colors%5B%5D=green&colors%5B%5D=blue&emptyArr=&nested%5Bempty%5D=&nested%5Bfruits%5D%5B%5D=orange&nested%5Bfruits%5D%5B%5D=banana&nested%5Bfruits%5D%5B%5D=apple&nested%5Bcurly%5D=sue&nested%5Bassoc%5D%5Ba%5D=alpha&nested%5Bassoc%5D%5Bb%5D=bravo&nested%5Bassoc%5D%5Bc%5D=charlie 我希望这能找到一个好人。

答案 1 :(得分:1)

  

有人可以向我解释为什么会这样   不是一个错误?

从技术上讲,我认为它不应该被标记为错误。相反,它只是他们如何设计行为功能,无论其他人是否不同意该决定。

您的API只能查看if (empty($_POST['2']))

答案 2 :(得分:1)

这是我目前的“蹩脚黑客”解决方案。注意我必须考虑嵌套数组的可能性,所以我的示例原始数组与我在问题中发布的略有不同:

$args = array("foo", "bar", array("red", "blue", array(), "green"), "baz");
$original_array = $args; // save it to compare later
function replace_empty_array_with_fake_string(&$value, $key) {
    if (is_array($value)) {
        if (empty($value)) {
            $value = 'array()';
        } else {
            array_walk($value, 'replace_empty_array_with_fake_string');
        }

    }
}
array_walk($args, 'replace_empty_array_with_fake_string');
$qs = http_build_query($args);

// convert the query string back to an array, this would happen on the "other side"
parse_str($qs, $query);
function replace_fake_string_with_empty_array(&$value, $key) {
    if ($value == 'array()') {
        $value = array();
    }
    if (is_array($value)) {
        array_walk($value, 'replace_fake_string_with_empty_array');
    }
}
array_walk($query, 'replace_fake_string_with_empty_array');
echo ($original_array == $query); // true

据推测,我可以想出一个比“array()”更多的字符串用作占位符。

拉姆,我知道。

答案 3 :(得分:0)

你可以简单地走查询参数,如果是空数组,请改用“[]”,如下所示:

function walkCriteria(&$criteria) {
    array_walk($criteria, function (&$val) {
        if ($val === []) {
            $val = "[]";
        } else if (is_array($val)) {
            walkCriteria($val);
        }
    });
}

不要使用 array_walk_recursive 。因为它会将 转换为 空数组并且什么都不做。

答案 4 :(得分:0)

基于@anyx解决方案:如果您想保留原始数组并处理NULL值,则可以使用以下版本:

function empty2blank(array $arr) {
 array_walk($arr, function(&$val, $key) {
  if (empty($val)) { 
   $val = is_array($val) ? '[]' : '';
  } elseif (is_array($val)) {
   $val = empty2blank($val);
  }
 });
 return $arr;
}