Операторы сравнения

Операторы сравнения, как это видно из их названия, позволяют сравнивать между собой два значения. Возможно вам будет интересно также ознакомиться с разделом Сравнение типов, в котором приведено большое количество соответствующих примеров.

Операторы сравнения
Пример Название Результат
$a == $b Равно TRUE если $a равно $b после преобразования типов.
$a === $b Тождественно равно TRUE если $a равно $b и имеет тот же тип.
$a != $b Не равно TRUE если $a не равно $b после преобразования типов.
$a <> $b Не равно TRUE если $a не равно $b после преобразования типов.
$a !== $b Тождественно не равно TRUE если $a не равно $b или они разных типов.
$a < $b Меньше TRUE если $a строго меньше $b.
$a > $b Больше TRUE если $a строго больше $b.
$a <= $b Меньше или равно TRUE если $a меньше или равно $b.
$a >= $b Больше или равно TRUE если $a больше или равно $b.

В случае, если вы сравниваете число со строкой или две строки, содержащие числа, каждая строка будет преобразована в число, и сравниваться они будут как числа. Эти правила также распространяются на оператор switch. Преобразование типов не происходит при использовании === или !== так как в этом случае кроме самих значений сравниваются еще и типы.

<?php
var_dump
(== "a"); // 0 == 0 -> true
var_dump("1" == "01"); // 1 == 1 -> true
var_dump("10" == "1e1"); // 10 == 10 -> true
var_dump(100 == "1e2"); // 100 == 100 -> true

switch ("a") {
case 
0:
    echo 
"0";
    break;
case 
"a"// Эта ветка никогда не будет достигнута, так как "a" уже сопоставленно с 0
    
echo "a";
    break;
}
?>

Для различных типов сравнение происходит в соответствии со следующей таблицей (по порядку).

Сравнение различных типов
Тип операнда 1 Тип операнда 2 Результат
null или string string NULL преобразуется в "", числовое или лексическое сравнение
bool или null что угодно Оба операнда преобразуются в bool, FALSE < TRUE
object object Встроенные классы могут определять свои собственные правила сравнения, объекты разных классов не сравниваются, объекты одного класса - сравниваются свойства тем же способом, что и в массивах (PHP 4), в PHP 5 есть свое собственное объяснение
string, resource или number string, resource или number Строки и ресурсы переводятся в числа, обычная математика
array array Массивы с меньшим числом элементов считаются меньше, если ключ из первого операнда не найден во втором операнде - массивы не могут сравниваться, иначе идет сравнение соответствующих значений (смотри пример ниже)
object что угодно object всегда больше
array что угодно array всегда больше

Пример #1 Сравнение булево/null

<?php
// Булево и null всегда сравниваются как булево значение 
var_dump(== TRUE);  // TRUE - тоже что и (bool)1 == TRUE
var_dump(== FALSE); // TRUE - тоже что и (bool)0 == FALSE
var_dump(100 TRUE); // FALSE - тоже что и (bool)100 < TRUE
var_dump(-10 FALSE);// FALSE - тоже что и (bool)-10 < FALSE
var_dump(min(-100, -10NULL10100)); // NULL - (bool)NULL < (bool)-100 , тоже что и FALSE < TRUE
?>

Пример #2 Алгоритм сравнения обычных массивов

<?php
// Так сравниваются массивы при сравнении стандартными операторами
function standard_array_compare($op1$op2)
{
    if (
count($op1) < count($op2)) {
        return -
1// $op1 < $op2
    
} elseif (count($op1) > count($op2)) {
        return 
1// $op1 > $op2
    
}
    foreach (
$op1 as $key => $val) {
        if (!
array_key_exists($key$op2)) {
            return 
null// не могут быть сравнимы
        
} elseif ($val $op2[$key]) {
            return -
1;
        } elseif (
$val $op2[$key]) {
            return 
1;
        }
    }
    return 
0// $op1 == $op2
}
?>

Смотрите также strcasecmp(), strcmp(), операторы массивов, и раздел руководства Типы.

Внимание

Сравнение чисел с плавающей точкой

Из-за особого внутреннего представления float, не нужно проверять на равенство два float-числа.

Для более подробной информации смотрите документацию по типу float.

Тернарный оператор

Еще одним условным оператором является тернарный оператор "?:".

Пример #3 Присваивание значения по умолчанию

<?php
// Пример использования тернарного оператора
$action = (empty($_POST['action'])) ? 'default' $_POST['action'];

// Приведенный выше код аналогичен следующему блоку с использованием if/else
if (empty($_POST['action'])) {
    
$action 'default';
} else {
    
$action $_POST['action'];
}

?>
Выражение (expr1) ? (expr2) : (expr3) интерпретируется как expr2, если expr1 имеет значение TRUE, или как expr3 если expr1 имеет значение FALSE.

Начиная с версии PHP 5.3 также стало возможным не писать среднюю часть тернарного оператора. Выражение expr1 ?: expr3 возвращает expr1 если expr1 имеет значение TRUE, и expr3 в другом случае.

Замечание: Пожалуйста учтите, что тернарный оператор является выражением и трактуется не как переменная, а как результат выражения. Это важно знать, если вы хотите вернуть переменную по ссылке. Выражение return $var == 42 ? $a : $b; не будет работать в функции, возвращающей значение по ссылке, а в более поздних версиях PHP также будет выдано предупреждение.

Замечание:

Рекомендуется избегать "нагромождения" тернарных выражений. Поведение PHP неочевидно при использовании нескольких тернарных операторов в одном выражении:

Пример #4 Неочевидное поведение тернарного оператора

<?php
// на первый взгляд, следующий код должен вывести 'true'
echo (true?'true':false?'t':'f');

// однако, он выводит 't'
// это происходит потому, что тернарные выражения вычисляются слева направо

// это намного более очевидная версия вышеприведенного кода
echo ((true 'true' false) ? 't' 'f');

// здесь вы можете видеть, что первое выражение вычисляется в 'true', которое
// в свою очередь вычисляется в (bool)true, таким образом возвращая истинную ветвь
// второго тернарного выражения.
?>

Коментарии

You can't just compare two arrays with the === operator
like you would think to find out if they are equal or not.  This is more complicated when you have multi-dimensional arrays.  Here is a recursive comparison function.

<?php
/**
 * Compares two arrays to see if they contain the same values.  Returns TRUE or FALSE.
 * usefull for determining if a record or block of data was modified (perhaps by user input)
 * prior to setting a "date_last_updated" or skipping updating the db in the case of no change.
 *
 * @param array $a1
 * @param array $a2
 * @return boolean
 */
function array_compare_recursive($a1$a2)
{
   if (!(
is_array($a1) and (is_array($a2)))) { return FALSE;}   
   
   if (!
count($a1) == count($a2)) 
      {
       return 
FALSE// arrays don't have same number of entries
     
}
     
   foreach (
$a1 as $key => $val
   {
       if (!
array_key_exists($key$a2)) 
           {return 
FALSE// uncomparable array keys don't match
             

       elseif (
is_array($val) and is_array($a2[$key]))  // if both entries are arrays then compare recursive 
           
{if (!array_compare_recursive($val,$a2[$key])) return FALSE;
           } 
       elseif (!(
$val === $a2[$key])) // compare entries must be of same type.
           
{return FALSE;
           }
   }
   return 
TRUE// $a1 === $a2
}
?>
2006-01-18 13:36:34
http://php5.kiev.ua/manual/ru/language.operators.comparison.html
Note: according to the spec, PHP's comparison operators are not transitive.  For example, the following are all true in PHP5:

"11" < "a" < 2 < "11"

As a result, the outcome of sorting an array depends on the order the elements appear in the pre-sort array.  The following code will dump out two arrays with *different* orderings:

<?php
$a 
= array(2,    "a""11"2);
$b = array(2,    "11""a"2);
sort($a);
var_dump($a);
sort($b);
var_dump($b);
?>

This is not a bug report -- given the spec on this documentation page, what PHP does is "correct".  But that may not be what was intended...
2006-05-09 01:49:59
http://php5.kiev.ua/manual/ru/language.operators.comparison.html
Автор:
When you want to know if two arrays contain the same values, regardless of the values' order, you cannot use "==" or "===".  In other words:

<?php
(array(1,2) == array(2,1)) === false;
?>

To answer that question, use:

<?php
function array_equal($a$b) {
    return (
is_array($a) && is_array($b) && array_diff($a$b) === array_diff($b$a));
}
?>

A related, but more strict problem, is if you need to ensure that two arrays contain the same key=>value pairs, regardless of the order of the pairs.  In that case, use:

<?php
function array_identical($a$b) {
    return (
is_array($a) && is_array($b) && array_diff_assoc($a$b) === array_diff_assoc($b$a));
}
?>

Example:
<?php
$a 
= array (21);
$b = array (12);
// true === array_equal($a, $b);
// false === array_identical($a, $b);

$a = array ('a' => 2'b' => 1);
$b = array ('b' => 1'a' => 2);
// true === array_identical($a, $b)
// true === array_equal($a, $b)
?>

(See also the solution "rshawiii at yahoo dot com" posted)
2006-10-26 18:49:53
http://php5.kiev.ua/manual/ru/language.operators.comparison.html
I couldn't find much info on stacking the new ternary operator, so I ran some tests:

<?php
echo ?: ?: ?: 3//1
echo ?: ?: ?: 2//1
echo ?: ?: ?: 3//2
echo ?: ?: ?: 0//3

echo ?: ?: ?: 3//1
echo ?: ?: ?: 3//2
echo ?: ?: ?: 3//3
?>

It works just as expected, returning the first non-false value within a group of expressions.
2010-02-02 00:32:04
http://php5.kiev.ua/manual/ru/language.operators.comparison.html
Автор:
In the table "Comparison with Various Types", please move the last line about "Object" to be above the line about "Array", since Object is considered to be greater than Array (tested on 5.3.3)

(Please remove my "Anonymous" post of the same content before. You could check IP to see that I forgot to type my name)
2011-07-10 10:48:55
http://php5.kiev.ua/manual/ru/language.operators.comparison.html
Автор:
A < B and still B < A...

$A = [1 => 1, 2 => 0, 3 => 1];
$B = [1 => 1, 3 => 0, 2 => 1];

var_dump($A < $B);  // TRUE
var_dump($B < $A);  // TRUE

var_dump($A > $B);  // TRUE
var_dump($B > $A);  // TRUE

Next - C and D are comparable, but neither C < D nor D < C (and still C != D)...

$C = [1 => 1, 2 => 1, 3 => 0];
$D = [1 => 1, 3 => 1, 2 => 0];

var_dump($C < $D); // FALSE
var_dump($D < $C); // FALSE

var_dump($C > $D); // FALSE
var_dump($D > $C); // FALSE

var_dump($D == $C); // FALSE
2015-06-21 23:12:37
http://php5.kiev.ua/manual/ru/language.operators.comparison.html
Care must be taken when using the spaceship operator with arrays that do not have the same keys:

- Contrary to the notes above ("Example #2 Transcription of standard array comparison"), it does *not* return null if the left-hand array contains a key that the right-hand array does not.
- Because of this, the result depends on the order you do the comparison in.

For example:

<?php
$a 
= ['a' => 1'b' => 2'c' => 3'e' => 4];
$b = ['a' => 1'b' => 2'd' => 3'e' => 4];

var_dump($a <=> $b);        // int(1) : $a > $b because $a has the 'c' key and $b doesn't.

var_dump($b <=> $a);        // int(1) : $b > $a because $b has the 'd' key and $a doesn't.
?>
2017-11-23 14:16:51
http://php5.kiev.ua/manual/ru/language.operators.comparison.html
Автор:
Searching for "double question mark" operator should find this page (and hopefully after this comment the crawlers will agree)
2019-09-16 20:00:43
http://php5.kiev.ua/manual/ru/language.operators.comparison.html
Автор:
Extending from here: https://www.php.net/manual/en/language.operators.comparison.php#121907

$a = ['a' => 1, 'b' => 2, 'c' => 3, 'e' => 4]; 
$b = ['a' => 1, 'b' => 2, 'd' => 3, 'e' => 4];

echo $a > $b; // 0
echo $b > $a; // 0
echo $a <$b; // 0
echo $b < $a; // 0

If using spaceship operator then it is returning true like : 

echo $a <=> $b; //1
echo $b <=> $a; //1
echo $a <=> $b; //1
echo $b <=> $a; //1
2020-01-23 08:59:59
http://php5.kiev.ua/manual/ru/language.operators.comparison.html
Автор:
Very careful when reading PHP documentation, Here's a lot of miss information. 

According to documentation, They say's (int) 0 == (string) "a" is true. But it is not in PHP 8.

var_dump(0 == "a"); // 0 == 0 -> true

Now In PHP 8 it's False.
2021-04-25 13:56:46
http://php5.kiev.ua/manual/ru/language.operators.comparison.html
Please be careful when you try to compare strings that have a plus sign `+` at the beginning (such as phone number, etc). When you use the Equal operator `==` PHP will ignore the plus sign. Use Identical operator `===` instead

Example:

$str1 = "62";
$str2 = "+62";

var_dump($str1 == $str2); // bool(true)
var_dump($str1 === $str2); // bool(false)
2022-08-06 16:05:56
http://php5.kiev.ua/manual/ru/language.operators.comparison.html
Please note that using the null coalescing operator to check properties on a class that has the __get magic method (without an __isset magic method) invokes the magic method. 

For example:

<?php

class A
{
    public function 
__get($property)
    {
        echo 
'Called __get for ' $property PHP_EOL;
    }
}

$a = new A();

echo 
'Trying null coalescing operator' PHP_EOL;
$b $a->test ?? 5;

echo 
'Trying isset()' PHP_EOL;
if (isset(
$a->test)) {
   
$b $a->test;
} else {
   
$b 5;
}

?>
2023-03-24 23:50:58
http://php5.kiev.ua/manual/ru/language.operators.comparison.html
Автор:
Between the "shortcut ternary" (aka "elvis") and "spaceship" operators, you can write some quite compact comparison functions for usort and its ilk.

If you want to sort an array of associative arrays by several different keys you can chain them in the same way that you can list column names in an SQL ORDER BY clause.

<?php
usort
($array, fn($a$b) => $a['a'] <=> $b['a']
                         ?: 
$b['b'] <=> $a['b']
                         ?: 
$a['c'] <=> $b['c']);
?>
Will sort the array by column 'a', then by column 'b' descending, then by column 'c'; or in SQL-speak 'ORDER BY a, b DESC, c".
2023-08-15 09:15:45
http://php5.kiev.ua/manual/ru/language.operators.comparison.html

    Поддержать сайт на родительском проекте КГБ