54

在 CoffeeScript、Clojure、ES6 和许多其他语言中,我们有对象/映射/等的解构,有点像这样:

obj = {keyA: 'Hello from A', keyB: 'Hello from B'}
{keyA, keyB} = obj

我在 php 中找到了可以让您像这样解构数组的list函数:

$info = array('coffee', 'brown', 'caffeine');
list($drink, $color, $power) = $info;

有没有办法在 PHP 中解构对象或关联数组?如果不在核心库中,也许有人写了一些智能辅助函数?

4

3 回答 3

68

对于 PHP 7.0 及更低版本,这超出了list. 文档状态:

list 仅适用于数字数组,并假定数字索引从 0 开始。

可能适合您的目的之一是将extract()变量从数组导入当前符号表的函数。虽然list您可以明确定义变量名,extract()但不会给您这种自由。

提取关联数组

extract可以这样做:

<?php

$info = [ 'drink' => 'coffee', 'color' => 'brown', 'power' => 'caffeine' ];
extract($info);

var_dump($drink); // string(6) "coffee"
var_dump($color); // string(5) "brown"
var_dump($power); // string(8) "caffeine"

提取对象

提取对象的工作方式几乎相同。由于extract仅将数组作为参数,我们需要将对象属性作为数组获取。get_object_vars为你做。它返回一个关联数组,其中所有公共属性作为键,它们的值作为值。

<?php

class User {

    public $name = 'Thomas';

}

$user = new User();
extract( get_object_vars($user) );

var_dump($name); // string(6) "Thomas"

陷阱

extract()与它不同,list因为它不允许您显式定义导出到符号表的变量名称。默认情况下,变量名称对应于数组键。

  • list是语言结构,extract()而是函数
  • 您可能会无意中覆盖您事先定义的变量
  • 您的数组键可能作为变量名无效

使用$flags可以作为第二个参数传递给您的参数,extract()您可以在发生冲突或无效变量的情况下影响行为。但是,了解其extract()工作原理并将其与 cauton 一起使用仍然很重要。

编辑:从 PHP 7.1 开始,这是可能的:

http://php.net/manual/en/migration71.new-features.php#migration71.new-features.support-for-keys-in-list

您现在可以在 list() 或其新的简写 [] 语法中指定键。这可以解构具有非整数或非顺序键的数组。

https://php.net/manual/en/migration71.new-features.php#migration71.new-features.symmetric-array-destructuring

速记数组语法 ([]) 现在可用于解构数组以进行赋值(包括在 foreach 中),作为现有 list() 语法的替代方案,该语法仍受支持。

例如这个:

$test_arr = ['a' => 1, 'b' => 2];
list('a' => $a, 'b' => $b) = $test_arr;
var_dump($a);
var_dump($b);

从 7.1.0 开始将输出以下内容

int(1) 
int(2)
于 2015-01-30T10:09:51.213 回答
45

我注意到接受的答案错过了使用速记符号、使用提取的安全问题和 IDE 问题的示例。

数值数组解构 (PHP 7.1)

自 PHP 7.1 起支持数值数组解构(对称数组解构),如下所示:

<?php
$data = [55, 'John', 'UK'];
[$id, $name] = $data; // short-hand (recommended)
list($id, $name) = $data; // long-hand

请注意,如果您不想要它们,您可能会错过它们。

关联数组解构 (PHP 7.1)

您还可以像这样解构关联数组(支持列表中的键):

<?php
$data = ['id' => 55, 'firstName' => 'John', 'country' => 'UK']
['id' => $id, 'firstName' => $name] = $data; // short-hand (recommended)
list('id' => $id, 'firstName' => $name) = $data; // long-hand

请注意,如果您不想要它们,您可能会错过它们。此外,变量名称可以与属性名称不同。

对象解构 (PHP 7.1)

不幸的是,没有对象解构。但是,您可以使用 将对象转换为关联数组get_object_vars,然后使用关联数组解构。

<?php
class User {
    public $id;
    public $name;
    public $country;
}

$user = new User();
$user->id = 55;
$user->name = 'John';
$user->country = 'UK';

['id' => $id, 'firstName' => $name] = get_object_vars($user)

但是,这可能会破坏某些 IDE 功能。以下是我在使用 PHPStorm 2019.1 时注意到的一些问题:

  • IDE 可能不再理解变量的类型,因此您需要添加一些@var TypePHPDocs 来维护自动完成功能
  • 不适用于重构工具。例如,如果您重命名属性之一,则数组解构部分也不会自动重命名。

所以我建议只做正常的方式:

$id = $user->id
$name = $user->firstName

不使用extract

使用提取,始终设置所有变量。使用它是一个非常糟糕的主意,因为:

  • 它可能导致安全问题。即使你小心,它也可能在未来导致不明显的安全漏洞。如果您确实使用它,请不要在用户输入时使用它(例如$_GET, $_POST),除非您想成为恶意黑客的一天。
  • 可能导致难以检测到错误
  • 如果将来类或数组发生更改,通过引入新属性,如果它与已使用的变量重合,它可能会破坏您的代码,除非您使用EXTR_SKIP标志或类似的
于 2019-08-02T14:14:38.397 回答
0

变量变量是实现此目的的一种方法:

$args = ['a' => 1, 'b' => 2, 'c' => 3];
foreach (['a', 'c'] as $v) $$v = $args[$v];
// $a is 1, $b is undefined, $c is 3

它真的不漂亮,谢天谢地,这已在 7.1 中由https://wiki.php.net/rfc/short_list_syntax解决。这会让你['a' => $a, 'c' => $c] = $args;在上面的例子中说。

由于 7.1 包含一种方法,可以为您的 var 使用与 assoc 数组键不同的名称。在这里使用变量变量也很简单:

foreach (['a' => 'eh', 'b' => 'bee'] as $k => $v) $$v = $args[$k];
// $eh is 1, $bee is 2

一些开发人员和一些编码风格,定义为类似于直接使用、、和魔术变量$$var的反模式。这是因为使用可变变量会使代码更难理解,从而直接导致错误并阻止静态代码分析工具发挥作用。evalextractGPR

如果您确实采用$$var,则改用该${$var}表单可能会有所帮助,这表明作者并没有简单地键入一个太多的 $,并且可以避免作者在其代码被审计时立即得到负面反馈。

于 2017-01-05T20:36:58.103 回答