有没有办法将数组转换为 class 属性?

Is there any way to convert an array to class properties?

有一种方法可以获取 class 属性并使用 (array) 转换进行转换。

$user = (array) get_object_vars($userObject)

但我想知道,我没有找到任何将数组转换为 class 属性的直接方法。

假设我的用户 class 具有以下实现:

class User
{
    private $id = 0;
    private $fname = '';
    private $lname = '';
    private $username = '';
}

我有以下数组:

$user = array(
    'id'=> 2,
    'fname' => 'FirstName',
    'lname' => 'LastName',
    'username' => 'UserName'
);

我正在寻找类型转换,就像我们 (object) 一样。问题是 (object) type cast converts array to stdClass object,但在这里我希望我的数组被转换成我指定的 class.

示例应该这样表述:(我的假设,PHP 应该有这个。或者它可能已经有这样的东西,我不知道。)

$userobj = (object User) $user;

以上语法的结果应该是这样的:

$userobj->id = $user['id'];
$userobj->fname = $user['fname'];
$userobj->lname = $user['lname'];
$userobj->username = $user['username'];

想知道有没有直接的方法,不是映射逻辑。寻找语言级别的技巧。

注意: 我知道使用 foreach 来完成上述工作。我正在寻找一些直接的方法,如果在 PHP 中可用,但还没有成为焦点(可能是!)。

以下没有解决我的问题: Convert/cast an stdClass object to another class

根据您对@AbraCadaver 评论的回复,我会考虑使用类似这样的方法将数据库结果加载到您选择的实例化 class 中:

// using MySQLi
$mysqli_result = ...; // your mysqli_result object
while($class_obj = $mysqli_result->fetch_object('your_class_name')) {
    // do something
}

// using PDO
$pdo_statement = ...; // your PDOStatement object
while($class_obj = $pdo_statement->fetch_object('your_class_name')) {
    // do something
}

数组不是对象。所以你不能用语言结构来转换它。即使在 PHP 中转换用户定义的对象也是不可能的。

所以你得自己写逻辑。 serialize() 对象有一些奇特的方法,更改 class 名称并再次 unserialize() 它以自动化该过程。

您也可以使用反射编写一个转换函数。但是一切都需要用户逻辑。

另见 post SO:Type casting for user defined objects

所以如果你真的喜欢投射对象,我最终得到了这个函数 (see it in action):

function cast($to, $obj)
{
    $toLength = strlen($to);
    $serializedObj = preg_replace('/^O:\d+:"[^"]+"/', 'O:' . $toLength . ':"' . $to . '"', serialize($obj));

    $serializedObj = preg_replace_callback('/s:(\d+):"([^"]+)";(.+?);/', function($m) use($to, $toLength) {
        $propertyName = $m[2];
        $propertyLength = $m[1];

        if(strpos($m[2], "[=10=]*[=10=]") === false && ($pos = strrpos($m[2], "[=10=]")) !== false) {
            $propertyName = "[=10=]" . $to . "[=10=]" . substr($m[2], $pos + 1);
            $propertyLength = ($m[1] + $toLength - $pos + 1);
        }

        return 's:' . $propertyLength . ':"' . $propertyName . '";' . $m[3] . ';';
    }, $serializedObj);

    return unserialize($serializedObj);
}

$user = array(
    'id'=> 2,
    'fname' => 'FirstName',
    'lname' => 'LastName',
    'username' => 'UserName'
);

$userObj = cast(User::class, (object)$user);

print_r($userObj);

但我永远不会在生产中使用这个功能。这只是一个实验性的东西。

更新

好吧,我重新考虑了 Java 中转换的处理方式。在 Java 中,您实际上可以从一个对象转换为它的子对象。您不能从任何对象转换到任何其他对象。

因此您可以在 Java 中执行此操作:

$apple = new Apple();
$macintoshApple = (MacintoshApple)$apple;

鉴于 MacintoshApple extends Apple.

但是你不能像这样投射两个不相关的对象:

$apple = new Apple();
$pear = (Pear)$apple;