Answers:
这是字符串插值的复杂(弯曲)语法。从手册中:
复杂(卷曲)语法
因为语法复杂,所以不将其称为复杂,而是因为它允许使用复杂的表达式。
可以通过此语法包括具有字符串表示形式的任何标量变量,数组元素或对象属性。只需以与出现在字符串外部相同的方式编写表达式,然后将其包装在
{
和中即可}
。由于{
无法转义,因此仅在$
紧随其后的才会识别此语法{
。使用{\$
得到文字{$
。一些示例可以使您清楚:<?php // Show all errors error_reporting(E_ALL); $great = 'fantastic'; // Won't work, outputs: This is { fantastic} echo "This is { $great}"; // Works, outputs: This is fantastic echo "This is {$great}"; echo "This is ${great}"; // Works echo "This square is {$square->width}00 centimeters broad."; // Works, quoted keys only work using the curly brace syntax echo "This works: {$arr['key']}"; // Works echo "This works: {$arr[4][3]}"; // This is wrong for the same reason as $foo[bar] is wrong outside a string. // In other words, it will still work, but only because PHP first looks for a // constant named foo; an error of level E_NOTICE (undefined constant) will be // thrown. echo "This is wrong: {$arr[foo][3]}"; // Works. When using multi-dimensional arrays, always use braces around arrays // when inside of strings echo "This works: {$arr['foo'][3]}"; // Works. echo "This works: " . $arr['foo'][3]; echo "This works too: {$obj->values[3]->name}"; echo "This is the value of the var named $name: {${$name}}"; echo "This is the value of the var named by the return value of getName(): {${getName()}}"; echo "This is the value of the var named by the return value of \$object->getName(): {${$object->getName()}}"; // Won't work, outputs: This is the return value of getName(): {getName()} echo "This is the return value of getName(): {getName()}"; ?>
通常,此语法是不必要的。例如,这:
$a = 'abcd';
$out = "$a $a"; // "abcd abcd";
行为与此完全相同:
$out = "{$a} {$a}"; // same
因此花括号是不必要的。但是这个:
$out = "$aefgh";
由于没有命名为的变量,根据您的错误级别,将不起作用或产生错误$aefgh
,因此您需要执行以下操作:
$out = "${a}efgh"; // or
$out = "{$a}efgh";
$out = '$aefgh'
,(如果您确实需要$ aefgh)
SimpleXMLElement
:{}
用于访问节点本身,例如$element[0]->{0}
。由于属性“ 0”不存在,因此将触发__get
/ __set
方法。本质上,这使您可以替代ArrayAccess
索引访问,例如3v4l.org/1F254。
$min=1;$max=5; echo ".{{$min},{$max}}"
yields .{1,5}
(我很难知道@BobStein的评论中提到的“将[大括号]加倍”)
对于我来说,花括号可以代替串联,它们输入起来更快,代码看起来更简洁。请记住使用双引号(“”),因为它们的内容由PHP 解析,因为在单引号('')中,您将获得提供的变量的字面名称:
<?php
$a = '12345';
// This works:
echo "qwe{$a}rty"; // qwe12345rty, using braces
echo "qwe" . $a . "rty"; // qwe12345rty, concatenation used
// Does not work:
echo 'qwe{$a}rty'; // qwe{$a}rty, single quotes are not parsed
echo "qwe$arty"; // qwe, because $a became $arty, which is undefined
?>
SHIFT
按键以获取双引号和大括号。但是,如果严格使用双引号会更快。
evaluate
,而不是parse
我还发现访问对象属性很有用,其中属性名称因某些迭代器而有所不同。例如,我将以下模式用于一组时间段:小时,天,月。
$periods=array('hour', 'day', 'month');
foreach ($periods as $period)
{
$this->{'value_'.$period}=1;
}
同样的模式也可以用于访问类方法。只需使用字符串和字符串变量以相同的方式构建方法名称即可。
您可以轻松地争辩说只使用一个数组来存储按期存储的值。如果此应用程序仅是PHP,我会同意。当类属性映射到数据库表中的字段时,我使用此模式。尽管可以使用序列化将数组存储在数据库中,但是效率低下,而且如果必须为各个字段建立索引也毫无意义。出于两全其美的考虑,我经常添加由迭代器键入的字段名称数组。
class timevalues
{
// Database table values:
public $value_hour; // maps to values.value_hour
public $value_day; // maps to values.value_day
public $value_month; // maps to values.value_month
public $values=array();
public function __construct()
{
$this->value_hour=0;
$this->value_day=0;
$this->value_month=0;
$this->values=array(
'hour'=>$this->value_hour,
'day'=>$this->value_day,
'month'=>$this->value_month,
);
}
}