我正在尝试创建一个新对象,为其成员之一分配一个值,然后在单个语句中将该对象分配给数组键。
即:
class MyObj {
public $member = 'sad';
}
$myArray = array(
'key' => ((new MyObj())->member = 'happy')
);
问题是这给了我一个语法错误: “语法错误,意外的‘=’”
以下代码是一种解决方法:
$obj = new MyObj();
$obj->member = 'happy';
$myArray = array(
'key' => $obj
);
问题是为什么单个语句不起作用?为什么会产生语法错误?最后,我如何创建一条语句来创建一个新对象,为其中一个成员变量赋值,然后将该对象与数组中的键相关联?
最佳答案
首先我想说这是一个很好的问题。我不知道为什么它不起作用,但有一个简单的解决方法,即使用带有参数的构造函数来设置成员变量。事实上,构造函数是为在初始化对象时触发代码而创建的,而这正是您正在尝试的。
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
Class member access on instantiation has been added, e.g. (new Foo)->bar().
编辑: 经过一些测试后,我想到了这个实际有效的示例。当您尝试初始化一个新对象并同时访问其属性时,您是 -per definition- 链接方法。这在 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;
}
}
如何使用:可链接的方法必须是私有(private)的或 protected 。父类 __call() 方法(Chainable 类)不会调用公共(public)方法,因为它们超出范围。
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();
关于php - 为什么在单个语句中分配动态对象的成员变量会导致 PHP 语法错误?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21639078/
类classAprivatedeffooputs:fooendpublicdefbarputs:barendprivatedefzimputs:zimendprotecteddefdibputs:dibendendA的实例a=A.new测试a.foorescueputs:faila.barrescueputs:faila.zimrescueputs:faila.dibrescueputs:faila.gazrescueputs:fail测试输出failbarfailfailfail.发送测试[:foo,:bar,:zim,:dib,:gaz].each{|m|a.send(m)resc
我有一个模型:classItem项目有一个属性“商店”基于存储的值,我希望Item对象对特定方法具有不同的行为。Rails中是否有针对此的通用设计模式?如果方法中没有大的if-else语句,这是如何干净利落地完成的? 最佳答案 通常通过Single-TableInheritance. 关于ruby-on-rails-Rails-子类化模型的设计模式是什么?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.co
我脑子里浮现出一些关于一种新编程语言的想法,所以我想我会尝试实现它。一位friend建议我尝试使用Treetop(Rubygem)来创建一个解析器。Treetop的文档很少,我以前从未做过这种事情。我的解析器表现得好像有一个无限循环,但没有堆栈跟踪;事实证明很难追踪到。有人可以指出入门级解析/AST指南的方向吗?我真的需要一些列出规则、常见用法等的东西来使用像Treetop这样的工具。我的语法分析器在GitHub上,以防有人希望帮助我改进它。class{initialize=lambda(name){receiver.name=name}greet=lambda{IO.puts("He
大约一年前,我决定确保每个包含非唯一文本的Flash通知都将从模块中的方法中获取文本。我这样做的最初原因是为了避免一遍又一遍地输入相同的字符串。如果我想更改措辞,我可以在一个地方轻松完成,而且一遍又一遍地重复同一件事而出现拼写错误的可能性也会降低。我最终得到的是这样的:moduleMessagesdefformat_error_messages(errors)errors.map{|attribute,message|"Error:#{attribute.to_s.titleize}#{message}."}enddeferror_message_could_not_find(obje
我正在使用的第三方API的文档状态:"[O]urAPIonlyacceptspaddedBase64encodedstrings."什么是“填充的Base64编码字符串”以及如何在Ruby中生成它们。下面的代码是我第一次尝试创建转换为Base64的JSON格式数据。xa=Base64.encode64(a.to_json) 最佳答案 他们说的padding其实就是Base64本身的一部分。它是末尾的“=”和“==”。Base64将3个字节的数据包编码为4个编码字符。所以如果你的输入数据有长度n和n%3=1=>"=="末尾用于填充n%
我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i
为什么4.1%2返回0.0999999999999996?但是4.2%2==0.2。 最佳答案 参见此处:WhatEveryProgrammerShouldKnowAboutFloating-PointArithmetic实数是无限的。计算机使用的位数有限(今天是32位、64位)。因此计算机进行的浮点运算不能代表所有的实数。0.1是这些数字之一。请注意,这不是与Ruby相关的问题,而是与所有编程语言相关的问题,因为它来自计算机表示实数的方式。 关于ruby-为什么4.1%2使用Ruby返
它不等于主线程的binding,这个toplevel作用域是什么?此作用域与主线程中的binding有何不同?>ruby-e'putsTOPLEVEL_BINDING===binding'false 最佳答案 事实是,TOPLEVEL_BINDING始终引用Binding的预定义全局实例,而Kernel#binding创建的新实例>Binding每次封装当前执行上下文。在顶层,它们都包含相同的绑定(bind),但它们不是同一个对象,您无法使用==或===测试它们的绑定(bind)相等性。putsTOPLEVEL_BINDINGput
我可以得到Infinity和NaNn=9.0/0#=>Infinityn.class#=>Floatm=0/0.0#=>NaNm.class#=>Float但是当我想直接访问Infinity或NaN时:Infinity#=>uninitializedconstantInfinity(NameError)NaN#=>uninitializedconstantNaN(NameError)什么是Infinity和NaN?它们是对象、关键字还是其他东西? 最佳答案 您看到打印为Infinity和NaN的只是Float类的两个特殊实例的字符串
如果您尝试在Ruby中的nil对象上调用方法,则会出现NoMethodError异常并显示消息:"undefinedmethod‘...’fornil:NilClass"然而,有一个tryRails中的方法,如果它被发送到一个nil对象,它只返回nil:require'rubygems'require'active_support/all'nil.try(:nonexisting_method)#noNoMethodErrorexceptionanymore那么try如何在内部工作以防止该异常? 最佳答案 像Ruby中的所有其他对象