演算子の優先順位は、二つの式が"緊密に"結合している度合いを指定します。
例えば、式 1 + 5 * 3
の答えは
16
になり、18
とはなりません。
これは乗算演算子("*")は、加算演算子("+")より高い優先順位を有するか
らです。必要に応じて強制的に優先順位を設定するために括弧を使用する
ことが可能です。例えば、18
と評価するためには、
(1 + 5) * 3
とします。
演算子の優先順位が等しい場合は、その結合性によって評価順
(右から評価するのか、あるいは左から評価するのか) が決まります。
たとえば "-" は左結合なので、
1 - 2 - 3
は (1 - 2) - 3
とグループ分けされて、
評価結果は -4
になります。
一方、"=" は右結合です。つまり $a = $b = $c
のグループ分けは
$a = ($b = $c)
となります。
優先順位が同じで結合しない演算子を並べることはできません。つまり、たとえば
1 < 2 > 1
は PHP では無効になります。
一方 1 <= 1 == 1
は問題ありません。
==
演算子の優先順位が
<=
演算子より低いからです。
厳密には不要な場所であっても、括弧をつけておけばコードの可読性があがります。 明示的にグループ分けをしておくことで、演算子の優先順位や結合性による暗黙のグループ分けに頼らずに済むからです。
次の表では、優先順位が高い順に演算子を挙げています。 同じ行にある演算子は優先順位が等しくなります。そのような場合は、 結合時の評価にしたがってグループ分けが決まります。
結合時の評価 | 演算子 | 追加情報 |
---|---|---|
(n/a) |
clone
new
|
clone および new |
right | ** |
代数演算子 |
(n/a) |
++
--
~
(int)
(float)
(string)
(array)
(object)
(bool)
@
|
型 および 加算子/減算子 |
left | instanceof |
型 |
(n/a) | ! |
論理演算子 |
left |
*
/
%
|
代数演算子 |
left |
+
-
.
|
代数演算子 そして 文字列演算子 |
left |
<<
>>
|
ビット演算子 |
結合しない |
<
<=
>
>=
|
比較演算子 |
結合しない |
==
!=
===
!==
<>
<=>
|
比較演算子 |
left | & |
ビット演算子 そして リファレンス |
left | ^ |
ビット演算子 |
left | | |
ビット演算子 |
left | && |
論理演算子 |
left | || |
論理演算子 |
right | ?? |
NULL合体演算子 |
(n/a) | ? : |
三項演算子 |
right |
=
+=
-=
*=
**=
/=
.=
%=
&=
|=
^=
<<=
>>=
??=
|
代入演算子 |
(n/a) | yield from |
yield from |
(n/a) | yield |
yield |
(n/a) | print |
|
left | and |
論理演算子 |
left | xor |
論理演算子 |
left | or |
論理演算子 |
例1 結合時の評価
<?php
$a = 3 * 3 % 5; // (3 * 3) % 5 = 4
$a = true ? 0 : true ? 1 : 2; // (true ? 0 : true) ? 1 : 2 = 2
$a = 1;
$b = 2;
$a = $b += 3; // $a = ($b += 3) -> $a = 5, $b = 5
?>
演算子の優先順位や結合性は、あくまでも式のグループ分けだけを決めるものであり、評価順を決めるものではありません。 PHP では一般に、式をどの順番で評価するかは決めていません。 そのため、特定の順序で式が評価されることを前提としたコードを書いてはいけません。 PHP のバージョンが変わったり前後のコードが変わったりしたときに、評価順が変わる可能性があるからです。
例2 評価順序は未定義
<?php
$a = 1;
echo $a + $a++; // 2 になるかもしれないし、3 になるかもしれません
$i = 1;
$array[$i] = $i++; // インデックス 1 をセットするかもしれないし、インデックス 2 をセットするかもしれません
?>
例3 +
、-
、.
の優先順位は同じ
<?php
$x = 4;
// 次の行は、予期せぬ結果になることでしょう
echo "x minus one equals " . $x-1 . ", or so I hope\n";
// なぜなら、これは次のように評価されるからです
echo (("x minus one equals " . $x) - 1) . ", or so I hope\n";
// 期待どおりの結果を得るには、括弧を使って優先順位を指定します
echo "x minus one equals " . ($x-1) . ", or so I hope\n";
?>
上の例の出力は以下となります。
-1, or so I hope -1, or so I hope x minus one equals 3, or so I hope
注意:
=
は他のほとんどの演算子よりも優先順位が低いはずなのにもかかわらず、 PHP は依然としてif (!$a = foo())
のような式も許します。この場合はfoo()
の返り値が $a に代入されます。
バージョン | 説明 |
---|---|
8.0.0 |
三項演算子 (? : ) は、
どの演算とも結合しなくなりました。
これより前のバージョンでは、左結合でした。
|
7.4.0 |
三項演算子(? : ) が左結合であることに依存すること、
つまり、括弧で囲わずに三項演算子をネストすることは推奨されなくなりました。
|