为什么在单个语句中分配动态对象的成员变量会导致 PHP 中的语法错误?

Why does assigning a dynamic object's member variable in a single statement cause a syntax error in PHP?

提问人:Drellgor 提问时间:2/8/2014 更新时间:2/11/2014 访问量:295

问:

我正在尝试创建一个新对象,为其成员之一分配一个值,然后在单个语句中将该对象分配给数组键。
IE浏览器:

class MyObj {
  public $member = 'sad';
}

$myArray = array(
  'key' => ((new MyObj())->member = 'happy')
);

问题是这给了我一个语法错误: “语法错误,意外的'='”

以下代码是一种解决方法:

$obj = new MyObj();
$obj->member = 'happy';
$myArray = array(
  'key' => $obj
);

问题是为什么单一语句不起作用?为什么会产生语法错误?最后,如何创建一个语句来创建一个新对象,为其中一个成员变量分配一个值,然后将该对象与数组中的键相关联?

php 动态 变量赋值 值运算符

评论

0赞 chiliNUT 2/8/2014
ha javascript 让你毫无怨言地做到这一点,现在我有点困惑console.log(new Array().length=5); //5

答:

1赞 kasper Taeymans 2/8/2014 #1

首先,我想说这是一个很好的问题。我不知道为什么它不起作用,但是有一个简单的解决方法,即使用带有参数的构造函数来设置成员变量。事实上,构造函数是用来在初始化对象时触发代码的,而这正是您正在尝试的。

class MyObj {
  public $member;
  public function __construct($x){
    $this->member=$x;
  }
}

$myArray = array(
  'key' => new MyObj('happy')
);

print_r($myArray);

编辑:我发现在 php 版本 5.4.0 之前无法在初始化期间访问类。我正在使用 php 5.4.12,也无法使其工作,但它应该可以工作,因为它是自 5.4.0 版以来添加的新功能。

http://docs.php.net/manual/en/migration54.new-features.php

添加了实例化的类成员访问,例如(新 Foo)->bar()。

编辑:经过一些测试,我想出了这个实际有效的例子。当您尝试初始化一个新对象并同时访问其属性时,您是 - 每个定义 - 链接方法。这在 php 版本 5.4.0 之前是不可能的。如果你使用的是 php >= 5.4,你可以这样做。若要初始化对象并同时访问其属性,需要确保前面的方法调用返回该对象(new Obj 自动返回该对象)。PHP 不会自动执行此操作,因此您需要手动执行此操作。请参阅下一次编辑以获取解决方法。return $this

class MyObj {
  public $member;
  public function setMember($x){
    $this->member=$x;
    return $this;
  }
}

$myArray = array(
  'key' => (new MyObj)->setMember('happy')
);

print_r($myArray);

请注意,成员变量由返回对象的方法设置。

编辑: (11/02/14)

我创建了一个类,使子类中的方法可链接,而无需手动返回。它还可以启用和禁用方法链接,因此您可以在不需要时执行这两项操作,而无需返回$this(请参阅和方法)$thisstart()stop()

Class Chainable{
    private  $chaining;//boolean

    public function __call($method,$arguments) {
        if(method_exists($this, $method)) {
            //$res=call_user_func_array(array($this,$method),$arguments);

            if($this->chaining ){
                if(!isset(call_user_func_array(array($this,$method),$arguments))){
                    return $this;
                }
                else{
                    echo "you can't chain methods that return something";
                            //however you could add the returned values to an array 
                            //(ie $this->returnedvalues)
                }   
            }

        }
    }
    private function setChaining($x){
        $this->chaining=$x;
    }
    public function start(){
        $this->setChaining(1);
        return $this;
    }
    public function stop(){
        $this->setChaining(0);
        return $this;
    }
}

如何使用:可链接的方法必须是私有的或受保护的。父方法(可链接类)不会调用公共方法,因为它们超出了范围。__call()

Class Foo extends Chainable{
    public $par1;
    public $par2;

    protected function setPar1($x){
        $this->par1=$x;
    }
    protected function setPar2($y){
        $this->par2=$y;
    }
    public function getPar1(){//public functions are not visible to the __call method from parent class.
        return $this->par1;
    }
    public function getPar2(){
        return $this->par2;
    }
    protected function printPar1(){
        echo $this->par1;
    }
    protected function printPar2(){
        echo $this->par2;
    }
}

$test=(new Foo())->start()//start chaining, set boolean true
                 ->setPar1('hello')//chained method calls...
                 ->setPar2('world')
                 ->printPar1()
                 ->printPar2()
                 ->stop();//stop chaining

$test->setPar1('hi');//normal method calls...
$test->setPar2('you');
echo '<br>'.$test->getPar1().' '.$test->getPar2();
$test->start()->setPar1('bonjour')->setPar2('soleil');//start chaining again
$test->stop();
0赞 Cody Caughlan 2/8/2014 #2

似乎是语法上的限制,只是不支持。或者可能是因为是一种语言结构,而不是表达式。类似于为什么在 PHP 5.5.0 之前,您不能将表达式传递给,而只能传递纯变量。newempty()

https://www.php.net/empty

5.5.0 empty() now supports expressions, rather than only variables.

0赞 chiliNUT 2/8/2014 #3

我不希望这能奏效。关键字返回类的新对象实例,而 php 不允许直接写入返回值。new

1赞 vascowhite 2/8/2014 #4

您犯的错误是您试图在实例化时设置属性。PHP >= 5.4 仅支持在实例化时访问方法。

以你的班级为例:-

class MyObj {
  public $member = 'sad';
}

这:-

(new MyObj())->member = 'happy' 

不支持,并且会产生意外结果。

然而,正如 kasper Taeymans 所指出的那样,方法将起作用。但是,它们必须返回一些有用的东西,因为实例化的类被丢弃,除非被调用的函数返回,这就是 kasper Taeymans 示例工作的原因。

但是,您不限于退货$this,您可以退货任何您想要的东西:-

class MyObj {
  public $member = 'sad';

  public function getMember()
  {
     return $this->member;
  }
}

$member = (new MyObj())->getMember();//$member now === 'sad';

我只是觉得重要的是要澄清任何事情都可以以这种方式从实例化调用的方法中返回。

评论

0赞 kasper Taeymans 2/11/2014
是的,你可以退回别的东西,然后$this,但随后你的链条停止了。
0赞 vascowhite 2/12/2014
@kasperTaeymans 对不起,我一定错过了问题中关于链接方法的部分。流畅的接口并不是想要调用构造方法的唯一原因。你的答案很好,但我不相信它回答了被问到的问题。但是,它在回答您想回答的问题方面做得非常出色:)