PHP 7.4 类型化属性迭代

PHP 7.4 Typed properties iteration

我刚刚发现了一些关于 PHP 7.4 的东西 "kinda strange",我不确定这是否只是我遗漏了什么,或者它是否是一个真正的错误。主要是我对您的 opinion/confirmation 感兴趣。

因此在 PHP 中,您可以像这样迭代对象属性:

class DragonBallClass 
{
    public $goku;
    public $bulma = 'Bulma';
    public $vegeta = 'Vegeta';
}

$dragonBall = new DragonBallClass();

foreach ($dragonBall as $character) {
  var_dump($character);

}

RESULT

NULL
string(5) "Bulma"
string(6) "Vegeta"

现在,如果我们开始使用这样的强类型属性:

class DragonBallClass 
{
    public string $goku;
    public string $bulma = 'Bulma';
    public string $vegeta = 'Vegeta';
}

$dragonBall = new DragonBallClass();

foreach ($dragonBall as $character) {
  var_dump($character);

}

我们会得到不同的结果:

string(5) "Bulma"
string(6) "Vegeta"

现在有什么不同:

当您不将默认值分配给强类型 属性 时,它将是 Uninitialized 类型。这当然是有道理的。问题是,如果它们以这样的方式结束,你就不能遍历它们,它们将被简单地省略——没有错误,没有任何东西,正如你在第二个例子中看到的那样。所以我无法访问它们。

这是有道理的,但想象一下您有一个像这样的自定义 Request/Data class:

namespace App\Request\Request\Post;

use App\Request\Request\Request;

class GetPostsRequest extends Request
{
    public string $title = '';
}

你看到那个难看的字符串赋值了吗?如果我想让 class 上的属性可迭代,那么我必须:

我可能想要一个具有类型化属性但其中没有任何值的对象来遍历它们并在有意义的情况下填充它们。

有更好的方法吗?是否有任何选项可以保留类型并保持 em 可迭代而不必执行此虚拟值可憎?

如果你想让一个类型化的属性可以为空,你可以简单地在类型前添加一个 ? 并将 NULL 作为默认值,如下所示:

class DragonBallClass 
{
    public ?string $goku = NULL;
    public string $bulma = 'Bulma';
    public string $vegeta = 'Vegeta';
}

在这种情况下,NULL 是一个完全合法的值(而不是虚拟值)。

demo


同样不使用 ?,您始终可以将 class 属性与对象属性列表合并:

class DragonBallClass 
{
    public string $goku;
    public string $bulma = 'Bulma';
    public string $vegeta = 'Vegeta';
}

$dragonBall = new DragonBallClass();

$classProperties = get_class_vars(get_class($dragonBall));
$objectProperties = get_object_vars($dragonBall);

var_dump(array_merge($classProperties, $objectProperties));

// array(3) {
//  ["goku"]=>
//  NULL
//  ["bulma"]=>
//  string(5) "Bulma"
//  ["vegeta"]=>
//  string(6) "Vegeta"
// }

在我们开始之前 - 我认为我接受并由 Casimir 提供的答案比我想出的答案更好更正确(这也适用于评论)。

我只是想分享我的想法,因为这在某种程度上是一个可行的解决方案,至少我们可以称之为答案。

这是我为满足我的特定需求而想出的,只是为了好玩。我很好奇我可以做些什么来让它更像我想要的那样,所以不要为此感到害怕;P 我认为这是一个非常干净的解决方法 - 我知道它并不完美。

class MyRequest
{
    public function __construct()
    {    
        $reflectedProperties = (new \ReflectionClass($this))->getProperties();
        foreach ($reflectedProperties as $property) {
            !$property->isInitialized($this) ??
            $property->getType()->allowsNull() ? $property->setValue($this, null) : null;
        }
    }

}


class PostRequest extends MyRequest 
{
    public ?string $title;

}

$postRequest = new PostRequest();

// works fine - I have access here!
foreach($postRequest as $property) {
    var_dump($property);
}

此解决方案的缺点是您总是必须在 class 中使类型可为空。 但是对于我和我的特定需求这完全没问题。我不介意,无论如何它们最终都可以为空,如果有人赶时间,这可能是一个很好的解决方法,可以解决截止日期很短的情况。

它仍然保留原始的 PHP not initialized 错误,尽管当类型不可为 null 时。我认为现在这真的很酷。您可以保留所有的东西:苗条和精益 classes,PHP 错误表明问题的真实性质以及如果您同意保持类型化属性可为 null 则循环遍历类型化属性的可能性。全部由本机 PHP 7 可空运算符控制。

当然,如果有意义的话,这可以更改或扩展为更特定于类型。

可能不是你想要的,但你可以使用反射:

<?php

class DragonBallClass 
{
    public string $goku;
    public string $bulma = 'Bulma';
    public string $vegeta = 'Vegeta';
}
$ob        = new DragonBallClass;
$reflector = new ReflectionClass($ob);
foreach($reflector->getProperties(ReflectionProperty::IS_PUBLIC) as $prop) {
    echo $prop->name, ':', ($ob->{$prop->name} ?? 'NOT INITIALISED'), "\n";
}

输出:

goku:NOT INITIALISED
bulma:Bulma
vegeta:Vegeta

更新:这个答案可能已经过时,但评论中包含一个有趣的讨论。

@Robert 的解决方法有问题;在这部分:

        foreach ($reflectedProperties as $property) {
            !$property->isInitialized($this) ??
            $property->getType()->allowsNull() ? $property->setValue($this, null) : null;
        }

必须将 ?? 更正为 &&

此外,这是对三元条件的误用;只需使用经典 if:

        foreach ($reflectedProperties as $property) {
            if (!$property->isInitialized($this)
                && $property->getType()->allowsNull()
            ) {
                $property->setValue($this, null);
            }
        }

或:

        foreach ($reflectedProperties as $property) {
            if (!$property->isInitialized($this) && $property->getType()->allowsNull()) {
                $property->setValue($this, null);
            }
        }