Приоритет оператора

Приоритет оператора определяет, насколько "тесно" он связывает между собой два выражения. Например, выражение 1 + 5 * 3 вычисляется как 16, а не 18, поскольку оператор умножения ("*") имеет более высокий приоритет, чем оператор сложения ("+"). Круглые скобки могут использоваться для принудительного указания порядка выполнения операторов. Например, выражение (1 + 5) * 3 вычисляется как 18.

Если операторы имеют равный приоритет, то будут ли они выполняться справа налево или слева направо определяется их ассоциативностью. К примеру, "-" является лево-ассоциативным оператором. Следовательно 1 - 2 - 3 сгруппируется как (1 - 2) - 3 и пересчитается в -4. С другой стороны "=" - право-ассоциативный оператор, так что $a = $b = $c сгруппируется как $a = ($b = $c).

Неассоциативные операторы с одинаковым приоритетом не могут использоваться совместно. К примеру 1 < 2 > 1 не будет работать в PHP. Выражение 1 <= 1 == 1, с другой стороны, будет, поскольку == имеет более низкий приоритет чем <=.

Использование скобок, кроме случаев когда они строго необходимы, может улучшить читаемость кода, группируя явно, а не полагаясь на приоритеты и ассоциативность.

В следующей таблице приведен список операторов, отсортированный по убыванию их приоритетов. Операторы, размещенные в одной строке имеют одинаковый приоритет и порядок их выполнения определяется исходя из их ассоциативности.

Порядок выполнения операторов
Ассоциативность Оператор Дополнительная информация
неассоциативна clone new clone и new
левая [ array()
правая ** арифметические операторы
правая ++ -- ~ (int) (float) (string) (array) (object) (bool) @ типы и инкремент/декремент
неассоциативна instanceof типы
правая ! логические операторы
левая * / % арифметические операторы
левая + - . арифметические операторы и строковые операторы
левая << >> побитовые операторы
неассоциативна < <= > >= операторы сравнения
неассоциативна == != === !== <> <=> операторы сравнения
левая & побитовые операторы и ссылки
левая ^ побитовые операторы
левая | побитовые операторы
левая && логические операторы
левая || логические операторы
правая ?? операторы сравнения
левая ? : тернарный оператор
правая = += -= *= **= /= .= %= &= |= ^= <<= >>= операторы присваивания
левая and логические операторы
левая xor логические операторы
левая or логические операторы

Пример #1 Ассоциативность

<![CDATA[
<?php
$a 
5// (3 * 3) % 5 = 4
// ассоциативность тернарного оператора отличается от  C/C++
$a true true 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 минус 1 равно " $x-", ну я надеюсь\n";
// поскольку он вычисляется таким образом::
echo (("x минус один равно " $x) - 1) . ", ну я надеюсь\n";
// требуемый приоритет следует задать скобками:
echo "x минус 1 равно " . ($x-1) . ", ну я надеюсь\n";
?>

Результат выполнения данного примера:

-1, or so I hope
-1, or so I hope
x минус один равно 3, ну я надеюсь

Замечание:

Несмотря на то, что = имеет более низкий приоритет чем большинство других операторов, PHP все же позволяет делать так: if (!$a = foo()), в этом примере результат выполнения foo() будет присвоен $a.

add a note add a note

User Contributed Notes 7 notes

up
57
fabmlk
1 year ago
Watch out for the difference of priority between 'and vs &&' or '|| vs or':
<?php
$bool
= true && false;
var_dump($bool); // false, that's expected

$bool = true and false;
var_dump($bool); // true, ouch!
?>
Because 'and/or' have lower priority than '=' but '||/&&' have higher.
up
32
Carsten Milkau
4 years ago
Beware the unusual order of bit-wise operators and comparison operators, this has often lead to bugs in my experience. For instance:

<?php if ( $flags & MASK  == 1) do_something(); ?>

will not do what you might expect from other languages. Use

<?php if (($flags & MASK) == 1) do_something(); ?>

in PHP instead.
up
1
karlisd at gmail dot com
1 year ago
Sometimes it's easier to understand things in your own examples.
If you want to play around operator precedence and look which tests will be made, you can play around with this:

<?php
function F($v) {echo $v." "; return false;}
function
T($v) {echo $v." "; return true;}

IF (
F(0) || T(1) && F(2)  || F(3)  && ! F(4) ) {
  echo
"true";
} else echo
" false";
?>
Now put in IF arguments f for false and t for true, put in them some ID's. Play out by changing "F" to "T" and vice versa, by keeping your ID the same. See output and you will know which arguments  actualy were checked.
up
0
headden at karelia dot ru
7 years ago
Although example above already shows it, I'd like to explicitly state that ?: associativity DIFFERS from that of C++. I.e. convenient switch/case-like expressions of the form

$i==1 ? "one" :
$i==2 ? "two" :
$i==3 ? "three" :
"error";

will not work in PHP as expected
up
-4
leipie at gmail dot com
3 years ago
The precedence of the arrow operator (->) on objects seems to the highest of all, even higher then clone.

But you can't wrap (clone $foo)->bar() like this!
up
-1
kitchin
4 months ago
The precedence of '->' is less than '[' in this situation: object contains array, name of array is stored in string variable.
<?php
$farm
= new StdClass;
$farm->emu = array( 'name' => 'Henry', 'age' => 9 );
$farm->rabbit = array( 'name' => 'George', 'age' => 4 );

$animal = 'rabbit';
print_r( $farm->$animal ); // ok
// print( $farm->$animal[ 'name' ] );  // wrong, [ has precedence.
print( $farm->{$animal}[ 'name' ] ); // correct, prints George.

$farm->wash = 'Suds';
$jobs = array( 'morning' => 'feed', 'evening' => 'wash' );
print(
$farm->$jobs[ 'evening' ] ); // correct, prints Suds.
print( $farm->{$jobs[ 'evening' ]} ); // correct, prints Suds.
?>
up
-12
Anonymous
1 year ago
The following example will output false
$a = 1;
$b = 1;

$c = $a + $a++;
$d = 1 + $b++;

if($c == $d){
    echo 'true';
}else{
    echo 'false';
}
To Top