it-swarm.com.ru

Создание анонимных объектов в php

Как мы знаем, создавать анонимные объекты в JavaScript легко, как в коде ниже:

var object = { 
    p : "value", 
    p1 : [ "john", "johnny" ] } ; 
alert(object.p1[1]) ;

Результат:

an alert is raised with value "johnny"

Можно ли применить эту же технику в случае PHP? Можем ли мы создавать анонимные объекты в PHP?

120
Sujit Agarwal

Прошло несколько лет, но я думаю, что мне нужно обновлять информацию!


В php-7 можно создавать анонимные классы, поэтому вы можете делать такие вещи:

<?php

    class Foo {}
    $child = new class extends Foo {};

    var_dump($child instanceof Foo); // true

?>

Вы можете прочитать больше об этом в RFC (это принято): https://wiki.php.net/rfc/anonymous_classes

Но я не знаю, насколько он похож на Javscript, поэтому между анонимными классами в javascript и php могут быть некоторые различия.

Правка:

Как и в комментариях, вот ссылка на руководство сейчас: http://php.net/manual/en/language.oop5.anonymous.php

31
Rizier123

«Аноним» не является правильной терминологией, когда речь идет об объектах. Было бы лучше сказать «объект анонимного type», но это не относится к PHP.

Все объекты в PHP имеют класс. Класс по умолчанию - stdClass , и вы можете создавать его объекты следующим образом:

$obj = new stdClass;
$obj->aProperty = 'value';

Вы также можете воспользоваться приведением массива к объекту для более удобного синтаксиса:

$obj = (object)array('aProperty' => 'value');
print_r($obj);

Однако имейте в виду, что приведение массива к объекту может привести к «интересным» результатам для тех ключей массива, которые не являются допустимыми PHP именами переменных - например, здесь мой ответ, который показывает что происходит, когда ключи начинаются с цифр.

214
Jon

Да, это возможно! Используя этот простой PHP анонимный объект class. Как это устроено:

// define by passing in constructor
$anonim_obj = new AnObj(array(
    "foo" => function() { echo "foo"; }, 
    "bar" => function($bar) { echo $bar; } 
));

$anonim_obj->foo(); // prints "foo"
$anonim_obj->bar("hello, world"); // prints "hello, world"

// define at runtime
$anonim_obj->Zoo = function() { echo "Zoo"; };
$anonim_obj->Zoo(); // prints "Zoo"

// mimic self 
$anonim_obj->prop = "abc";
$anonim_obj->propMethod = function() use($anonim_obj) {
    echo $anonim_obj->prop; 
};
$anonim_obj->propMethod(); // prints "abc"

Конечно, этот объект является экземпляром класса AnObj, поэтому он не является анонимным, но позволяет определять методы на лету, как это делает JavaScript.

19
Mihailoff

До недавнего времени я создавал объекты на лету.

$someObj = json_decode("{}");

Затем:

$someObj->someProperty = someValue;

Но теперь я иду с:

$someObj = (object)[];

Тогда как раньше:

$someObj->someProperty = someValue;

Конечно, если вы уже знаете свойства и значения, вы можете установить их внутри, как уже упоминалось:

$someObj = (object)['prop1' => 'value1','prop2' => 'value2'];

NB. Я не знаю, с какими версиями PHP это работает, поэтому вам следует помнить об этом. Но я думаю, что первый подход (который также является коротким, если нет никаких свойств для установки в конструкции) должен работать для всех версий, которые имеют json_encode/json_decode

13
Zuks

Если вы хотите имитировать JavaScript, вы можете создать класс Object и, таким образом, получить то же поведение. Конечно, это уже не совсем анонимно, но это сработает.

<?php 
class Object { 
    function __construct( ) { 
        $n = func_num_args( ) ; 
        for ( $i = 0 ; $i < $n ; $i += 2 ) { 
            $this->{func_get_arg($i)} = func_get_arg($i + 1) ; 
        } 
    } 
} 

$o = new Object( 
    'aProperty', 'value', 
    'anotherProperty', array('element 1', 'element 2')) ; 
echo $o->anotherProperty[1];
?>

Это выведет element 2. Это было украдено из комментарий к PHP: классы и объекты .

7
kba

Стенография:

$obj = (object)      ['property' => 'Here we go'];
                 //or//
$obj = (object) array('property' => 'Here we go');
5
T.Todua

Если вы хотите создать объект (как в javascript) с динамическими свойствами, без получения предупреждения о неопределенном свойстве, когда вы не установили значение для свойства 

class stdClass {

public function __construct(array $arguments = array()) {
    if (!empty($arguments)) {
        foreach ($arguments as $property => $argument) {
            if(is_numeric($property)):
                $this->{$argument} = null;
            else:
                $this->{$property} = $argument;
            endif;
        }
    }
}

public function __call($method, $arguments) {
    $arguments = array_merge(array("stdObject" => $this), $arguments); // Note: method argument 0 will always referred to the main class ($this).
    if (isset($this->{$method}) && is_callable($this->{$method})) {
        return call_user_func_array($this->{$method}, $arguments);
    } else {
        throw new Exception("Fatal error: Call to undefined method stdObject::{$method}()");
    }
}

public function __get($name){
    if(property_exists($this, $name)):
        return $this->{$name};
    else:
        return $this->{$name} = null;
    endif;
}

public function __set($name, $value) {
    $this->{$name} = $value;
}

}

$obj1 = new stdClass(['property1','property2'=>'value']); //assign default property
echo $obj1->property1;//null
echo $obj1->property2;//value

$obj2 = new stdClass();//without properties set
echo $obj2->property1;//null
0
fredtma

Для того, кто хочет рекурсивный объект:

$o = (object) array(
    'foo' => (object) array(
        'sub' => '...'
    )
);

echo $o->foo->sub;
0
Jonatas Walker

Можно ли применить эту же технику в случае PHP?

Нет - поскольку javascript использует прототипы/прямое объявление объектов - в PHP (и во многих других OO языках) объект может быть создан только из класса.

Поэтому возникает вопрос - можете ли вы создать анонимный класс.

Опять же, ответ «нет» - как бы вы создали экземпляр класса, не имея возможности ссылаться на него?

0
symcbean