Использование json_encode для объектов в PHP (независимо от области видимости)
Я пытаюсь вывести списки объектов как json и хотел бы узнать, есть ли способ сделать объекты пригодными для json_encode
? Код, который я получил, выглядит примерно так
$related = $user->getRelatedUsers();
echo json_encode($related);
Прямо сейчас я просто перебираю массив пользователей и по отдельности экспортирую их в массивы для json_encode
превратить в пригодный для использования JSON для меня. Я уже пытался сделать объекты повторяемыми, но json_encode
просто, кажется, пропустить их в любом случае.
редактировать: вот var_dump();
php > var_dump($a);
object(RedBean_OODBBean)#14 (2) {
["properties":"RedBean_OODBBean":private]=>
array(11) {
["id"]=>
string(5) "17972"
["pk_UniversalID"]=>
string(5) "18830"
["UniversalIdentity"]=>
string(1) "1"
["UniversalUserName"]=>
string(9) "showforce"
["UniversalPassword"]=>
string(32) ""
["UniversalDomain"]=>
string(1) "0"
["UniversalCrunchBase"]=>
string(1) "0"
["isApproved"]=>
string(1) "0"
["accountHash"]=>
string(32) ""
["CurrentEvent"]=>
string(4) "1204"
["userType"]=>
string(7) "company"
}
["__info":"RedBean_OODBBean":private]=>
array(4) {
["type"]=>
string(4) "user"
["sys"]=>
array(1) {
["idfield"]=>
string(2) "id"
}
["tainted"]=>
bool(false)
["model"]=>
object(Model_User)#16 (1) {
["bean":protected]=>
*RECURSION*
}
}
}
и вот что дает мне json_encode:
php > echo json_encode($a);
{}
Я закончил только с этим:
function json_encode_objs($item){
if(!is_array($item) && !is_object($item)){
return json_encode($item);
}else{
$pieces = array();
foreach($item as $k=>$v){
$pieces[] = "\"$k\":".json_encode_objs($v);
}
return '{'.implode(',',$pieces).'}';
}
}
Он берет массивы, полные этих объектов или только отдельные экземпляры, и превращает их в json - я использую его вместо json_encode. Я уверен, что есть места, где я мог бы сделать это лучше, но я надеялся, что json_encode сможет определить, когда нужно перебирать объект, основываясь на его открытых интерфейсах.
9 ответов
В RedBeanPHP 2.0 есть функция массового экспорта, которая превращает всю коллекцию бинов в массивы. Это работает с кодировщиком JSON.
json_encode( R::exportAll( $beans ) );
Все свойства вашего объекта являются частными. ака... не доступно за пределами их класса.
Решение для PHP < 5.4
Если вы хотите сериализовать свои свойства закрытых и защищенных объектов, вы должны реализовать функцию кодирования JSON внутри вашего класса, которая использует json_encode()
на структуру данных, которую вы создаете для этой цели.
class Thing {
...
public function to_json() {
return json_encode(array(
'something' => $this->something,
'protected_something' => $this->get_protected_something(),
'private_something' => $this->get_private_something()
));
}
...
}
Решение для PHP >= 5.4
Используйте новый JsonSerializable
Интерфейс для предоставления собственного представления json для использования json_encode
class Thing implements JsonSerializable {
...
public function jsonSerialize() {
return [
'something' => $this->something,
'protected_something' => $this->get_protected_something(),
'private_something' => $this->get_private_something()
];
}
...
}
В PHP >= 5.4.0 появился новый интерфейс для сериализации объектов в JSON: JsonSerializable
Просто реализуйте интерфейс в вашем объекте и определите JsonSerializable
метод, который будет вызываться при использовании json_encode
,
Поэтому решение для PHP >= 5.4.0 должно выглядеть примерно так:
class JsonObject implements JsonSerializable
{
// properties
// function called when encoded with json_encode
public function jsonSerialize()
{
return get_object_vars($this);
}
}
Следующий код работал для меня:
public function jsonSerialize()
{
return get_object_vars($this);
}
Я еще не видел упомянутого, но у бобов есть встроенный метод, называемый getProperties()
,
Итак, чтобы использовать это:
// What bean do we want to get?
$type = 'book';
$id = 13;
// Load the bean
$post = R::load($type,$id);
// Get the properties
$props = $post->getProperties();
// Print the JSON-encoded value
print json_encode($props);
Это выводит:
{
"id": "13",
"title": "Oliver Twist",
"author": "Charles Dickens"
}
Теперь сделайте шаг вперед. Если у нас есть массив бобов...
// An array of beans (just an example)
$series = array($post,$post,$post);
... тогда мы могли бы сделать следующее:
Перебрать массив с
foreach
петля.Замените каждый элемент (компонент) массивом свойств компонента.
Так что это...
foreach ($series as &$val) {
$val = $val->getProperties();
}
print json_encode($series);
... выводит это:
[
{
"id": "13",
"title": "Oliver Twist",
"author": "Charles Dickens"
},
{
"id": "13",
"title": "Oliver Twist",
"author": "Charles Dickens"
},
{
"id": "13",
"title": "Oliver Twist",
"author": "Charles Dickens"
}
]
Надеюсь это поможет!
Я обычно включаю небольшую функцию в мои объекты, которая позволяет мне создавать дамп в массив или json или xml. Что-то вроде:
public function exportObj($method = 'a')
{
if($method == 'j')
{
return json_encode(get_object_vars($this));
}
else
{
return get_object_vars($this);
}
}
в любом случае, get_object_vars()
вероятно, полезно для вас.
Вот мой путь:
function xml2array($xml_data)
{
$xml_to_array = [];
if(isset($xml_data))
{
if(is_iterable($xml_data))
{
foreach($xml_data as $key => $value)
{
if(is_object($value))
{
if(empty((array)$value))
{
$value = (string)$value;
}
else
{
$value = (array)$value;
}
$value = xml2array($value);
}
$xml_to_array[$key] = $value;
}
}
else
{
$xml_to_array = $xml_data;
}
}
return $xml_to_array;
}
$products=R::findAll('products');
$string = rtrim(implode(',', $products), ',');
echo $string;
Для массива объектов я использовал что-то вроде этого, следуя пользовательскому методу для php < 5.4:
$jsArray=array();
//transaction is an array of the class transaction
//which implements the method to_json
foreach($transactions as $tran)
{
$jsArray[]=$tran->to_json();
}
echo json_encode($jsArray);