Как преобразовать массив в объект в PHP?
Как преобразовать такой массив в object?
[128] => Array ( [status] => Figure A. Facebook's horizontal scrollbars showing up on a 1024x768 screen resolution. ) [129] => Array ( [status] => The other day at work, I had some spare time ) )
30 ответов
в простейшем случае, вероятно, достаточно "бросить" массив в качестве объекта:
$object = (object) $array;
другой вариант-создать экземпляр стандартного класса в качестве переменной и выполнить цикл через массив при повторном назначении значений:
$object = new stdClass();
foreach ($array as $key => $value)
{
$object->$key = $value;
}
As Эдсон Медина указал, действительно чистым решением является использование встроенного json_
функции:
$object = json_decode(json_encode($array), FALSE);
Это также (рекурсивно) преобразует все ваши суб-массивы в объекты, которые вы можете или может не хочу. К сожалению, он имеет 2-3x производительность над циклическим подходом.
предупреждение! (спасибо Ultra за комментарий):
json_decode в разных средах преобразует данные UTF-8 по-разному. Я заканчиваю тем, что получаю значения "240.00" локально и " 240 " на производстве - massive dissaster. Morover если преобразование не удается строка get возвращается как NULL
вы можете просто использовать литье типов для преобразования массива в объект.
// *convert array to object* Array([id]=> 321313[username]=>shahbaz)
$object = (object) $array_name;
//now it is converted to object and you can access it.
echo $object->username;
вот три способа:
-
поддельные реальный объект:
class convert { public $varible; public function __construct($array) { $this = $array; } public static function toObject($array) { $array = new convert($array); return $array; } }
-
преобразовать массив в объект, приведя его к объекту:
$array = array( // ... ); $object = (object) $array;
-
вручную преобразовать массив в объект:
$object = object; foreach ($arr as $key => $value) { $object->{$key} = $value; }
быстрый Хак:
// assuming $var is a multidimensional array
$obj = json_decode (json_encode ($var), FALSE);
Не очень, но работает.
простой способ был бы
$object = (object)$array;
но это не то, что вы хотите. Если вам нужны объекты, вы хотите чего-то достичь, но этого не хватает в этом вопросе. Использование объектов только по причине использования объектов не имеет смысла.
свой путь к простому, это создаст объект для рекурсивных массивов также:
$object = json_decode(json_encode((object) $yourArray), FALSE);
В зависимости от того, где вам это нужно и как получить доступ к объекту, есть разные способы сделать это.
например: просто введите его
$object = (object) $yourArray;
однако наиболее совместимым является использование метода утилиты (еще не являющегося частью PHP), который реализует стандартный PHP-кастинг на основе строки, указывающей тип (или игнорируя его, просто отменяя ссылку на значение):
/**
* dereference a value and optionally setting its type
*
* @param mixed $mixed
* @param null $type (optional)
*
* @return mixed $mixed set as $type
*/
function rettype($mixed, $type = NULL) {
$type === NULL || settype($mixed, $type);
return $mixed;
}
пример использования в вашем случае (онлайн Демо):
$yourArray = Array('status' => 'Figure A. ...');
echo rettype($yourArray, 'object')->status; // prints "Figure A. ..."
это сработало для меня
function array_to_obj($array, &$obj)
{
foreach ($array as $key => $value)
{
if (is_array($value))
{
$obj->$key = new stdClass();
array_to_obj($value, $obj->$key);
}
else
{
$obj->$key = $value;
}
}
return $obj;
}
function arrayToObject($array)
{
$object= new stdClass();
return array_to_obj($array,$object);
}
использование :
$myobject = arrayToObject($array);
print_r($myobject);
возвращает :
[127] => stdClass Object
(
[status] => Have you ever created a really great looking website design
)
[128] => stdClass Object
(
[status] => Figure A.
Facebook's horizontal scrollbars showing up on a 1024x768 screen resolution.
)
[129] => stdClass Object
(
[status] => The other day at work, I had some spare time
)
как обычно, вы можете зациклить его так:
foreach($myobject as $obj)
{
echo $obj->status;
}
нет встроенного метода, чтобы сделать это, насколько мне известно, но это так же просто, как простой цикл:
$obj= new stdClass();
foreach ($array as $k=> $v) {
$obj->{$k} = $v;
}
вы можете объяснить это, если вам нужно рекурсивно построить свой объект.
на самом деле, если вы хотите использовать это с многомерными массивами, вы хотели бы использовать некоторую рекурсию.
static public function array_to_object(array $array)
{
foreach($array as $key => $value)
{
if(is_array($value))
{
$array[$key] = self::array_to_object($value);
}
}
return (object)$array;
}
Я бы определенно пошел с чистым способом, как это:
<?php
class Person {
private $name;
private $age;
private $sexe;
function __construct ($payload)
{
if (is_array($payload))
$this->from_array($payload);
}
public function from_array($array)
{
foreach(get_object_vars($this) as $attrName => $attrValue)
$this->{$attrName} = $array[$attrName];
}
public function say_hi ()
{
print "hi my name is {$this->name}";
}
}
print_r($_POST);
$mike = new Person($_POST);
$mike->say_hi();
?>
Если вы представите:
вы получите:
Я нашел это более логичным, сравнивая приведенные выше ответы от объектов, следует использовать для цели, для которой они были сделаны (инкапсулированные милые маленькие объекты).
также с помощью get_object_vars убедитесь, что в управляемом объекте не создаются дополнительные атрибуты (вы не хотите автомобиль, имеющий фамилию, ни человек, ведущий себя 4 колеса).
рекурсия-ваш друг:
function __toObject(Array $arr) {
$obj = new stdClass();
foreach($arr as $key=>$val) {
if (is_array($val)) {
$val = __toObject($val);
}
$obj->$key = $val;
}
return $obj;
}
просто:
$object = json_decode(json_encode($array));
пример:
$array = array(
'key' => array(
'k' => 'value',
),
'group' => array('a', 'b', 'c')
);
$object = json_decode(json_encode($array));
тогда верно следующее:
$object->key->k === 'value';
$object->group === array('a', 'b', 'c')
вы также можете использовать ArrayObject, например:
<?php
$arr = array("test",
array("one"=>1,"two"=>2,"three"=>3),
array("one"=>1,"two"=>2,"three"=>3)
);
$o = new ArrayObject($arr);
echo $o->offsetGet(2)["two"],"\n";
foreach ($o as $key=>$val){
if (is_array($val)) {
foreach($val as $k => $v) {
echo $k . ' => ' . $v,"\n";
}
}
else
{
echo $val,"\n";
}
}
?>
//Output:
2
test
one => 1
two => 2
three => 3
one => 1
two => 2
three => 3
тот, который я использую (это член класса):
const MAX_LEVEL = 5; // change it as needed
public function arrayToObject($a, $level=0)
{
if(!is_array($a)) {
throw new InvalidArgumentException(sprintf('Type %s cannot be cast, array expected', gettype($a)));
}
if($level > self::MAX_LEVEL) {
throw new OverflowException(sprintf('%s stack overflow: %d exceeds max recursion level', __METHOD__, $level));
}
$o = new stdClass();
foreach($a as $key => $value) {
if(is_array($value)) { // convert value recursively
$value = $this->arrayToObject($value, $level+1);
}
$o->{$key} = $value;
}
return $o;
}
используйте эту функцию, что я сделал:
function buildObject($class,$data){
$object = new $class;
foreach($data as $key=>$value){
if(property_exists($class,$key)){
$object->{'set'.ucfirst($key)}($value);
}
}
return $object;
}
использование:
$myObject = buildObject('MyClassName',$myArray);
немного сложно, но легко расширить технику:
Предположим у вас есть массив
$a = [
'name' => 'ankit',
'age' => '33',
'dob' => '1984-04-12'
];
Предположим, у вас есть класс person, который может иметь больше или меньше атрибутов из этого массива. например
class Person
{
private $name;
private $dob;
private $age;
private $company;
private $city;
}
если вы все еще хотите изменить свой массив на объект person. Вы можете использовать класс ArrayIterator.
$arrayIterator = new \ArrayIterator($a); // Pass your array in the argument.
теперь у вас есть объект iterator.
создать расширение класса FilterIterator; где нужно определите абстрактный метод accept. Следуйте примеру
class PersonIterator extends \FilterIterator
{
public function accept()
{
return property_exits('Person', parent::current());
}
}
выше impelmentation будет привязать свойство, только если он существует в классе.
добавить еще один метод в класс PersonIterator
public function getObject(Person $object)
{
foreach ($this as $key => $value)
{
$object->{'set' . underscoreToCamelCase($key)}($value);
}
return $object;
}
убедитесь, что в вашем классе определены мутаторы. Теперь вы готовы вызвать эти функции, где вы хотите создать объект.
$arrayiterator = new \ArrayIterator($a);
$personIterator = new \PersonIterator($arrayiterator);
$personIterator->getObject(); // this will return your Person Object.
вы также можете сделать это путем добавления (объект) слева от переменной, чтобы создать новый объект.
<?php
$a = Array
( 'status' => " text" );
var_dump($a);
$b = (object)$a;
var_dump($b);
var_dump($b->status);
используя json_encode
проблематично из-за того, как он обрабатывает данные, отличные от UTF-8. Стоит отметить, что json_encode
/json_encode
метод также оставляет неассоциативные массивы в виде массивов. Это может быть или не быть тем, чего вы хотите. Недавно мне пришлось воссоздать функциональность этого решения, но без использования json_
функции. Вот что я придумал:--6-->
/**
* Returns true if the array has only integer keys
*/
function isArrayAssociative(array $array) {
return (bool)count(array_filter(array_keys($array), 'is_string'));
}
/**
* Converts an array to an object, but leaves non-associative arrays as arrays.
* This is the same logic that `json_decode(json_encode($arr), false)` uses.
*/
function arrayToObject(array $array, $maxDepth = 10) {
if($maxDepth == 0) {
return $array;
}
if(isArrayAssociative($array)) {
$newObject = new \stdClass;
foreach ($array as $key => $value) {
if(is_array($value)) {
$newObject->{$key} = arrayToObject($value, $maxDepth - 1);
} else {
$newObject->{$key} = $value;
}
}
return $newObject;
} else {
$newArray = array();
foreach ($array as $value) {
if(is_array($value)) {
$newArray[] = arrayToObject($value, $maxDepth - 1);
} else {
$newArray[] = $value;
}
}
return $newArray;
}
}
лучший метод в мире :)
function arrayToObject($conArray)
{
if(is_array($conArray)){
/*
* Return array converted to object
* Using __FUNCTION__ (Magic constant)
* for recursive call
*/
return (object) array_map(__FUNCTION__, $conArray);
}else{
// Return object
return $conArray;
}
}
Если вы используете разные методы, у вас будут проблемы. Это лучший метод. Вы когда-нибудь видели.
для этого требуется PHP7, потому что я решил использовать лямбда-функцию для блокировки "innerfunc" в основной функции. Лямбда-функция вызывается рекурсивно, отсюда необходимость: "use (&$innerfunc )". Вы могли бы сделать это в PHP5, но не могли скрыть innerfunc.
function convertArray2Object($defs) {
$innerfunc = function ($a) use ( &$innerfunc ) {
return (is_array($a)) ? (object) array_map($innerfunc, $a) : $a;
};
return (object) array_map($innerfunc, $defs);
}
очевидно, просто экстраполяция ответов некоторых других людей, но вот рекурсивная функция, которая преобразует любой многомерный массив в объект:
function convert_array_to_object($array){
$obj= new stdClass();
foreach ($array as $k=> $v) {
if (is_array($v)){
$v = convert_array_to_object($v);
}
$obj->{strtolower($k)} = $v;
}
return $obj;
}
и помните, что если массив имел числовые ключи, на них все еще можно ссылаться в результирующем объекте, используя {}
(например: $obj->prop->{4}->prop
)
вдохновленный всеми этими кодами, я попытался создать расширенную версию с поддержкой: конкретного имени класса, избежать метода конструктора, шаблона "бобов" и строгого режима (установить только существующие свойства):
class Util {
static function arrayToObject($array, $class = 'stdClass', $strict = false) {
if (!is_array($array)) {
return $array;
}
//create an instance of an class without calling class's constructor
$object = unserialize(
sprintf(
'O:%d:"%s":0:{}', strlen($class), $class
)
);
if (is_array($array) && count($array) > 0) {
foreach ($array as $name => $value) {
$name = strtolower(trim($name));
if (!empty($name)) {
if(method_exists($object, 'set'.$name)){
$object->{'set'.$name}(Util::arrayToObject($value));
}else{
if(($strict)){
if(property_exists($class, $name)){
$object->$name = Util::arrayToObject($value);
}
}else{
$object->$name = Util::arrayToObject($value);
}
}
}
}
return $object;
} else {
return FALSE;
}
}
}
код
эта функция работает так же, как json_decode(json_encode($arr), false)
.
function arrayToObject(array $arr)
{
$flat = array_keys($arr) === range(0, count($arr) - 1);
$out = $flat ? [] : new \stdClass();
foreach ($arr as $key => $value) {
$temp = is_array($value) ? $this->arrayToObject($value) : $value;
if ($flat) {
$out[] = $temp;
} else {
$out->{$key} = $temp;
}
}
return $out;
}
тестирование
CakePHP имеет рекурсивный класс Set:: map, который в основном отображает массив в объект. Возможно, вам придется изменить внешний вид массива, чтобы объект выглядел так, как вы хотите.
http://api.cakephp.org/view_source/set/#line-158
в худшем случае вы можете получить несколько идей из этой функции.
Я сделал это довольно простым способом,
$list_years = array();
$object = new stdClass();
$object->year_id = 1 ;
$object->year_name = 2001 ;
$list_years[] = $object;
function object_to_array($data)
{
if (is_array($data) || is_object($data))
{
$result = array();
foreach ($data as $key => $value)
{
$result[$key] = object_to_array($value);
}
return $result;
}
return $data;
}
function array_to_object($data)
{
if (is_array($data) || is_object($data))
{
$result= new stdClass();
foreach ($data as $key => $value)
{
$result->$key = array_to_object($value);
}
return $result;
}
return $data;
}
используя (array) и (object) в качестве префикса, вы можете просто преобразовать массив объектов в стандартный массив и vice-verse
<?php
//defining an array
$a = array('a'=>'1','b'=>'2','c'=>'3','d'=>'4');
//defining an object array
$obj = new stdClass();
$obj->a = '1';
$obj->b = '2';
$obj->c = '3';
$obj->d = '4';
print_r($a);echo '<br>';
print_r($obj);echo '<br>';
//converting object array to array
$b = (array) $obj;
print_r($b);echo '<br>';
//converting array to object
$c = (object) $a;
print_r($c);echo '<br>';
?>
Я использую следующее для разбора ассоциативных массивов файлов Yaml в состояние объекта.
это проверяет все поставляемые массивы, если есть объекты, скрывающиеся там, и превращает их также в объекты.
/**
* Makes a config object from an array, making the first level keys properties a new object.
* Property values are converted to camelCase and are not set if one already exists.
* @param array $configArray Config array.
* @param boolean $strict To return an empty object if $configArray is not an array
* @return stdObject The config object
*/
public function makeConfigFromArray($configArray = [],$strict = true)
{
$object = new stdClass();
if (!is_array($configArray)) {
if(!$strict && !is_null($configArray)) {
return $configArray;
}
return $object;
}
foreach ($configArray as $name => $value) {
$_name = camel_case($name);
if(is_array($value)) {
$makeobject = true;
foreach($value as $key => $val) {
if(is_numeric(substr($key,0,1))) {
$makeobject = false;
}
if(is_array($val)) {
$value[$key] = $this->makeConfigFromArray($val,false);
}
}
if($makeobject) {
$object->{$name} = $object->{$_name} = $this->makeConfigFromArray($value,false);
}
else {
$object->{$name} = $object->{$_name} = $value;
}
}
else {
$object->{$name} = $object->{$_name} = $value;
}
}
return $object;
}
это превращает yaml, настроенный как
fields:
abc:
type: formfield
something:
- a
- b
- c
- d:
foo:
bar
массиву, состоящему из:
array:1 [
"fields" => array:1 [
"abc" => array:2 [
"type" => "formfield"
"something" => array:4 [
0 => "a"
1 => "b"
2 => "c"
3 => array:1 [
"d" => array:1 [
"foo" => "bar"
]
]
]
]
]
]
на объект:
{#325
+"fields": {#326
+"abc": {#324
+"type": "formfield"
+"something": array:4 [
0 => "a"
1 => "b"
2 => "c"
3 => {#328
+"d": {#327
+"foo": "bar"
}
}
]
}
}
}