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

Posted

技术标签:

【中文标题】为啥在单个语句中分配动态对象的成员变量会导致 PHP 中的语法错误?【英文标题】:Why does assigning a dynamic object's member variable in a single statement cause a syntax error in PHP?为什么在单个语句中分配动态对象的成员变量会导致 PHP 中的语法错误? 【发布时间】:2014-03-05 12:48:26 【问题描述】:

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

class MyObj 
  public $member = 'sad';


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

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

以下代码可以解决:

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

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

【问题讨论】:

ha javascript 让你做到不抱怨,现在我有点困惑console.log(new Array().length=5); //5 【参考方案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。它还可以启用和禁用方法链接,因此您可以在不需要时不返回 $this 来执行这两项操作(请参阅start()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() 方法(Chainable 类)不会调用公共方法,因为它们超出了范围。

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();

【讨论】:

【参考方案2】:

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

以你的班级为例:-

class MyObj 
  public $member = 'sad';

这个:-

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

不支持,会产生意想不到的结果。

然而,方法会起作用,as pointed out by kasper Taeymans。但是,它们必须返回一些有用的东西,因为实例化的类被丢弃,除非被调用的函数返回,这就是 kasper Taeymans 的示例有效的原因。

但是,你不限于返回 $this,你可以返回任何你想要的东西:-

class MyObj 
  public $member = 'sad';

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


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

我只是觉得有必要澄清任何东西都可以通过这种方式从调用实例化的方法返回。

【讨论】:

是的,你可以在 $this 之后返回其他东西,但是你的链会停止。 @kasperTaeymans 抱歉,我一定错过了问题中关于链接方法的部分。流畅的接口并不是想要在构造时调用方法的唯一原因。您的回答很好,但我不相信它回答了所提出的问题。但是,它在回答您想回答的问题方面做得很好:)【参考方案3】:

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

【讨论】:

【参考方案4】:

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

http://us2.php.net/empty

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

【讨论】:

以上是关于为啥在单个语句中分配动态对象的成员变量会导致 PHP 中的语法错误?的主要内容,如果未能解决你的问题,请参考以下文章

当堆中分配的对象变大时会发生啥?

在单个代码行 C# 中分配多个变量

java在循环语句的执行语句赋予变量值循环结束后会释放吗?

Vue2:当我从响应中分配属性时,为啥会收到无限的 $http.get 请求?

java基础学习堆和栈的区别详解

PHP:在if语句中分配多个变量