103

因此,当我偶然发现新的JsonSerializable Interface时,我在php.net上闲逛以获取有关将 PHP 对象序列化为 JSON 的信息。虽然它只是PHP >= 5.4,但我在 5.3.x 环境中运行。

这种功能是如何实现PHP < 5.4的?

我还没有对 JSON 进行过多的工作,但我正在尝试在应用程序中支持 API 层,并且将数据对象(否则将被发送到视图)转储到 JSON 中将是完美的。

如果我尝试直接序列化对象,它会返回一个空的 JSON 字符串;这是因为我假设json_encode()不知道该怎么处理这个对象。我是否应该递归地将对象简化为数组,然后对其进行编码


例子

$data = new Mf_Data();
$data->foo->bar['hello'] = 'world';

echo json_encode($data)产生一个空对象:

{}

var_dump($data)但是,按预期工作:

object(Mf_Data)#1 (5) {
  ["_values":"Mf_Data":private]=>
  array(0) {
  }
  ["_children":"Mf_Data":private]=>
  array(1) {
    [0]=>
    array(1) {
      ["foo"]=>
      object(Mf_Data)#2 (5) {
        ["_values":"Mf_Data":private]=>
        array(0) {
        }
        ["_children":"Mf_Data":private]=>
        array(1) {
          [0]=>
          array(1) {
            ["bar"]=>
            object(Mf_Data)#3 (5) {
              ["_values":"Mf_Data":private]=>
              array(1) {
                [0]=>
                array(1) {
                  ["hello"]=>
                  string(5) "world"
                }
              }
              ["_children":"Mf_Data":private]=>
              array(0) {
              }
              ["_parent":"Mf_Data":private]=>
              *RECURSION*
              ["_key":"Mf_Data":private]=>
              string(3) "bar"
              ["_index":"Mf_Data":private]=>
              int(0)
            }
          }
        }
        ["_parent":"Mf_Data":private]=>
        *RECURSION*
        ["_key":"Mf_Data":private]=>
        string(3) "foo"
        ["_index":"Mf_Data":private]=>
        int(0)
      }
    }
  }
  ["_parent":"Mf_Data":private]=>
  NULL
  ["_key":"Mf_Data":private]=>
  NULL
  ["_index":"Mf_Data":private]=>
  int(0)
}

附录

1)

所以这是toArray()我为Mf_Data班级设计的功能:

public function toArray()
{
    $array = (array) $this;
    array_walk_recursive($array, function (&$property) {
        if ($property instanceof Mf_Data) {
            $property = $property->toArray();
        }
    });
    return $array;
}

但是,由于Mf_Data对象也有对其父(包含)对象的引用,因此递归失败。_parent当我删除参考时,它就像一个魅力。

2)

只是为了跟进,我使用的转换复杂树节点对象的最终函数是:

// class name - Mf_Data
// exlcuded properties - $_parent, $_index
public function toArray()
{
    $array = get_object_vars($this);
    unset($array['_parent'], $array['_index']);
    array_walk_recursive($array, function (&$property) {
        if (is_object($property) && method_exists($property, 'toArray')) {
            $property = $property->toArray();
        }
    });
    return $array;
}

3)

我再次跟进,实施更清晰。使用接口进行instanceof检查似乎比method_exists()但是method_exists()交叉继承/实现)更干净。

使用unset()似乎也有点乱,似乎应该将逻辑重构为另一种方法。但是,此实现确实复制了属性数组(由于array_diff_key),因此需要考虑。

interface ToMapInterface
{

    function toMap();

    function getToMapProperties();

}

class Node implements ToMapInterface
{

    private $index;
    private $parent;
    private $values = array();

    public function toMap()
    {
        $array = $this->getToMapProperties();
        array_walk_recursive($array, function (&$value) {
            if ($value instanceof ToMapInterface) {
                $value = $value->toMap();
            }
        });
        return $array;
    }

    public function getToMapProperties()
    {
        return array_diff_key(get_object_vars($this), array_flip(array(
            'index', 'parent'
        )));
    }

}
4

11 回答 11

93

在最简单的情况下,类型提示应该起作用:

$json = json_encode( (array)$object );
于 2011-07-26T21:43:32.610 回答
46

编辑:目前是 2016-09-24,PHP 5.4 已于 2012-03-01 发布,支持已于2015-09-01结束。尽管如此,这个答案似乎还是获得了支持。如果您仍在使用 PHP < 5.4,那么您将产生安全风险并危及您的项目。如果您没有令人信服的理由留在 <5.4,甚至已经使用版本 >= 5.4,请不要使用此答案,而只需使用 PHP>= 5.4(或者,您知道,最近的一个)并实现JsonSerializable 接口


您将定义一个函数,例如 named getJsonData();,它将返回一个数组、stdClass对象或其他具有可见参数而不是私有/受保护参数的对象,然后执行json_encode($data->getJsonData());. 本质上是从 5.4 开始实现函数,但是手动调用它。

像这样的东西会起作用,就像get_object_vars()从类内部调用的那样,可以访问私有/受保护的变量:

function getJsonData(){
    $var = get_object_vars($this);
    foreach ($var as &$value) {
        if (is_object($value) && method_exists($value,'getJsonData')) {
            $value = $value->getJsonData();
        }
    }
    return $var;
}
于 2011-07-26T21:09:16.610 回答
20

json_encode()只会对公共成员变量进行编码。所以如果你想在你必须自己做的时候包括私人(正如其他人所建议的那样)

于 2011-07-26T21:14:57.707 回答
11

以下代码正在使用反射完成这项工作。它假设您有要序列化的属性的吸气剂

    <?php

    /**
     * Serialize a simple PHP object into json
     * Should be used for POPO that has getter methods for the relevant properties to serialize
     * A property can be simple or by itself another POPO object
     *
     * Class CleanJsonSerializer
     */
    class CleanJsonSerializer {

    /**
     * Local cache of a property getters per class - optimize reflection code if the same object appears several times
     * @var array
     */
    private $classPropertyGetters = array();

    /**
     * @param mixed $object
     * @return string|false
     */
    public function serialize($object)
    {
        return json_encode($this->serializeInternal($object));
    }

    /**
     * @param $object
     * @return array
     */
    private function serializeInternal($object)
    {
        if (is_array($object)) {
            $result = $this->serializeArray($object);
        } elseif (is_object($object)) {
            $result = $this->serializeObject($object);
        } else {
            $result = $object;
        }
        return $result;
    }

    /**
     * @param $object
     * @return \ReflectionClass
     */
    private function getClassPropertyGetters($object)
    {
        $className = get_class($object);
        if (!isset($this->classPropertyGetters[$className])) {
            $reflector = new \ReflectionClass($className);
            $properties = $reflector->getProperties();
            $getters = array();
            foreach ($properties as $property)
            {
                $name = $property->getName();
                $getter = "get" . ucfirst($name);
                try {
                    $reflector->getMethod($getter);
                    $getters[$name] = $getter;
                } catch (\Exception $e) {
                    // if no getter for a specific property - ignore it
                }
            }
            $this->classPropertyGetters[$className] = $getters;
        }
        return $this->classPropertyGetters[$className];
    }

    /**
     * @param $object
     * @return array
     */
    private function serializeObject($object) {
        $properties = $this->getClassPropertyGetters($object);
        $data = array();
        foreach ($properties as $name => $property)
        {
            $data[$name] = $this->serializeInternal($object->$property());
        }
        return $data;
    }

    /**
     * @param $array
     * @return array
     */
    private function serializeArray($array)
    {
        $result = array();
        foreach ($array as $key => $value) {
            $result[$key] = $this->serializeInternal($value);
        }
        return $result;
    }  
} 
于 2014-08-28T06:48:11.800 回答
7

只需实现 PHP JsonSerializable给出的接口即可。

于 2016-01-25T14:08:15.007 回答
2

由于您的对象类型是自定义的,我倾向于同意您的解决方案 - 使用编码方法(如 JSON 或序列化内容)将其分解为更小的部分,并且在另一端有相应的代码来重新构建对象。

于 2011-07-26T21:08:35.143 回答
2

我的版本:

json_encode(self::toArray($ob))

执行:

private static function toArray($object) {
    $reflectionClass = new \ReflectionClass($object);

    $properties = $reflectionClass->getProperties();

    $array = [];
    foreach ($properties as $property) {
        $property->setAccessible(true);
        $value = $property->getValue($object);
        if (is_object($value)) {
            $array[$property->getName()] = self::toArray($value);
        } else {
            $array[$property->getName()] = $value;
        }
    }
    return $array;
}

JsonUtils:GitHub

于 2017-06-04T09:43:51.473 回答
1

尝试使用它,这对我来说效果很好。

json_encode(unserialize(serialize($array)));
于 2018-07-28T05:28:00.867 回答
1

更改为您的变量private类型public

这更简单且更具可读性。

例如

不工作;

class A{
   private $var1="valuevar1";
   private $var2="valuevar2";
   public function tojson(){
    return json_encode($this)
   }
}

这是工作;

class A{
   public $var1="valuevar1";
   public $var2="valuevar2";
   public function tojson(){
    return json_encode($this)
   }
}
于 2019-09-23T23:00:27.470 回答
0

我做了一个很好的帮助类,它将带有 get 方法的对象转换为数组。它不依赖于属性,只依赖于方法。

所以我有一个包含两种方法的以下评论对象:

审查

  • getAmountReviews : 整数
  • getReviews : 评论数组

评论

  • 获取主题
  • 获取描述

我编写的脚本会将其转换为具有如下属性的数组:

    {
      amount_reviews: 21,
      reviews: [
        {
          subject: "In een woord top 1!",
          description: "Lorem ipsum dolor sit amet, consectetur adipiscing elit. Pellentesque laoreet lacus quis eros venenatis, sed tincidunt mi rhoncus. Aliquam ut pharetra diam, nec lobortis dolor."
        },
        {
          subject: "En een zwembad 2!",
          description: "Maecenas et aliquet mi, a interdum mauris. Donec in egestas sem. Sed feugiat commodo maximus. Pellentesque porta consectetur commodo. Duis at finibus urna."
        },
        {
          subject: "In een woord top 3!",
          description: "Lorem ipsum dolor sit amet, consectetur adipiscing elit. Pellentesque laoreet lacus quis eros venenatis, sed tincidunt mi rhoncus. Aliquam ut pharetra diam, nec lobortis dolor."
        },
        {
          subject: "En een zwembad 4!",
          description: "Maecenas et aliquet mi, a interdum mauris. Donec in egestas sem. Sed feugiat commodo maximus. Pellentesque porta consectetur commodo. Duis at finibus urna."
       },
       {
          subject: "In een woord top 5!",
          description: "Lorem ipsum dolor sit amet, consectetur adipiscing elit. Pellentesque laoreet lacus quis eros venenatis, sed tincidunt mi rhoncus. Aliquam ut pharetra diam, nec lobortis dolor."
    }
]}

来源:PHP Serializer,它将对象转换为可以编码为 JSON 的数组。

您所要做的就是将 json_encode 包裹在输出周围。

关于脚本的一些信息:

  • 仅添加以 get 开头的方法
  • 私有方法被忽略
  • 构造函数被忽略
  • 方法名称中的大写字符将替换为下划线和小写字符
于 2016-06-11T21:38:52.363 回答
-7

我在同一个问题上花了几个小时。我要转换的对象包含许多其他我不应该触及的定义(API),所以我想出了一个我猜可能很慢的解决方案,但我将它用于开发目的。

这个将任何对象转换为数组

function objToArr($o) {
$s = '<?php
class base {
    public static function __set_state($array) {
        return $array;
    }
}
function __autoload($class) {
    eval("class $class extends base {}");
}
$a = '.var_export($o,true).';
var_export($a);
';
$f = './tmp_'.uniqid().'.php';
file_put_contents($f,$s);
chmod($f,0755);
$r = eval('return '.shell_exec('php -f '.$f).';');
unlink($f);
return $r;
}

这会将任何对象转换为 stdClass

class base {
    public static function __set_state($array) {
        return (object)$array;
    }
}
function objToStd($o) {
$s = '<?php
class base {
    public static function __set_state($array) {
        $o = new self;
        foreach($array as $k => $v) $o->$k = $v;
        return $o;
    }
}
function __autoload($class) {
    eval("class $class extends base {}");
}
$a = '.var_export($o,true).';
var_export($a);
';
$f = './tmp_'.uniqid().'.php';
file_put_contents($f,$s);
chmod($f,0755);
$r = eval('return '.shell_exec('php -f '.$f).';');
unlink($f);
return $r;
}
于 2012-08-15T17:08:29.670 回答