Оператор отношения - Relational operator

В Информатика, а реляционный оператор это язык программирования построить или оператор который проверяет или определяет какой-то связь между два объекта. К ним относятся числовые равенство (например, 5 = 5) и неравенство (например, 4 ≥ 3).

В языках программирования, которые включают логический тип данных в их система типов, подобно Паскаль, Ада, или же Ява, эти операторы обычно оцениваются как истинные или ложные, в зависимости от того, существует ли условная связь между двумя операнды держит или нет. На таких языках, как C, операторы отношения возвращают целые числа 0 или 1, где 0 означает ложь, а любое ненулевое значение означает истину.

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

Равенство

использование

Равенство используется во многих конструкциях языков программирования и типах данных. Он используется для проверки, существует ли уже элемент в набор или для доступа к значению с помощью ключа. Он используется в операторы переключения для отправки потока управления в правильную ветвь и во время процесса унификации в логическом программировании.

Одно из возможных значений равенства состоит в том, что «если а равно б, то либо а или же б могут использоваться взаимозаменяемо в любом контексте, не замечая никакой разницы ». Но это утверждение не обязательно верно, особенно когда учитывается изменчивость вместе с равенством содержимого.

Сходство (идентичность объекта) против равенства содержимого

Иногда, особенно в объектно-ориентированного программирования, сравнение вызывает вопросы типы данных и наследование, равенство, и личность. Часто бывает необходимо различать:

  • два разных объекта одного типа, например, две руки
  • два объекта равны, но различны, например, две банкноты по 10 долларов
  • два объекта равны, но имеют разное представление, например, банкнота в 1 доллар и монета в 1 доллар
  • две разные ссылки на один и тот же объект, например, два прозвища для одного и того же человека

Во многих современных языках программирования доступ к объектам и структурам данных осуществляется через Рекомендации. На таких языках возникает необходимость проверки двух разных типов равенства:

  • Физическое равенство: если две ссылки (A и B) ссылаются на один и тот же объект. Взаимодействия с объектом через A неотличимы от тех же взаимодействий через B, и, в частности, изменения объекта через A отражаются через B. Физическая идентичность не применима, когда речь идет о ценностях, а не об объектах.
  • Семантическое равенство: если объекты, на которые ссылаются две ссылки или два значения, эквивалентны в некотором смысле:
  • Структурное равенство (т.е., их содержание такое же). который может быть либо поверхностным (тестирование только непосредственных частей), либо глубоким (проверка равенства частей рекурсивно). Простым способом добиться этого является репрезентативное равенство: проверка того, что значения имеют одинаковое представление.
  • Другое индивидуальное равенство, сохраняющее внешнее поведение. Например, 1/2 и 2/4 считаются равными, если рассматривать их как рациональные числа. Возможное требование: «A = B тогда и только тогда, когда все операции с объектами A и B будут иметь одинаковый результат», в дополнение к рефлексивность, симметрия, и транзитивность.

Первый тип равенства обычно подразумевает второй (за исключением таких вещей, как не число (NaN ), которые неравны сами по себе), но обратное не обязательно верно. Например, два нить объекты могут быть разными объектами (неравными в первом смысле), но содержать одну и ту же последовательность символов (равными во втором смысле). Видеть личность для получения дополнительной информации по этому вопросу.

Реальные числа, включая множество простых фракции, не могут быть точно представлены в арифметика с плавающей запятой, и может потребоваться проверка на равенство в пределах заданного допуска. Однако такая толерантность может легко нарушить желаемые свойства, такие как транзитивность, в то время как рефлексивность также нарушается: IEEE с плавающей точкой стандарт требует, чтобы NaN ≠ NaN держит.

Другие элементы программирования, такие как вычислимые функции, могут либо не иметь чувства равенства, либо равенства, которое невозможно вычислить. По этим причинам некоторые языки определяют явное понятие «сопоставимого» в форме базового класса, интерфейса, признака или протокола, которое используется либо явно, путем объявления в исходном коде, либо неявно через структуру вовлеченного типа.

Сравнение значений разных типов

В JavaScript, PHP, VBScript и еще несколько динамически типизированный языках стандартный оператор равенства оценивается как истинный если два значения равны, даже если они имеют разные типы, например, число 4 при сравнении равно текстовой строке «4». Типизированный оператор равенства также часто доступен на таких языках, возвращающий истину только для значений с идентичными или эквивалентными типами (в PHP, 4 === "4" ложно, хотя 4 == "4" правда).[1][2] Для языков, в которых число 0 можно интерпретировать как ложный, этот оператор может упростить такие вещи, как проверка нуля (как х == 0 будет истинным для x равным 0 или "0" с использованием оператора равенства, не зависящего от типа).

Заказ

Лучше чем и меньше, чем сравнение нечисловых данных выполняется в соответствии с соглашением о сортировке (например, для текстовых строк, лексикографический порядок ), который может быть встроен в язык программирования и / или настраиваться программистом.

Когда требуется связать числовое значение с результатом сравнения двух элементов данных, например а и б, обычное соглашение заключается в присвоении −1, если a b. Например, функция C strcmp выполняет трехстороннее сравнение и возвращает -1, 0 или 1 в соответствии с этим соглашением, и qsort ожидает, что функция сравнения вернет значения в соответствии с этим соглашением. В алгоритмы сортировки эффективность кода сравнения имеет решающее значение, поскольку это один из основных факторов, влияющих на производительность сортировки.

Сравнение определенных программистом типы данных (типы данных, для которых язык программирования не имеет встроенного понимания) могут выполняться с помощью специально написанных или библиотечных функций (таких как strcmp упомянутые выше), или, на некоторых языках, перегрузка оператор сравнения, то есть присвоение значения, определенного программистом, которое зависит от сравниваемых типов данных. Другой альтернативой является использование некоторого соглашения, такого как поэлементное сравнение.

Логическая эквивалентность

Хотя поначалу это может быть неочевидно, как логический логические операторы Операторы отношения XOR, AND, OR и NOT могут иметь логическая эквивалентность, так что все они могут быть определены в терминах друг друга. Следующие четыре условных оператора имеют одинаковую логическую эквивалентность E (либо все верно, либо все ложно) для любого данного Икс и у значения:

Это зависит от того, что домен хорошо заказанный.

Стандартные реляционные операторы

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

Общие реляционные операторы
соглашениеравноне равнолучше чемменьше, чемлучше чем
или равно
меньше, чем
или равно
В печати=><
FORTRAN[примечание 1].EQ..NE..GT..LT..GE..LE.
АЛГОЛ 68[заметка 2]=><
/=>=<=
эквnegtltgeле
APL=><
БАЗОВЫЙ, ML, Паскаль[заметка 3]=<>><>=<=
МАМПЫ='=><'<'>
Lua==~=><>=<=
C-подобный[примечание 4]==!=><>=<=
Erlang==/=><>==<
=:==/=
Борноподобный снаряды[примечание 5]-eq-ne-gt-lt-ge-le
Пакетный файлEQUNEQGTRLSSGEQLEQ
MATLAB[примечание 6]==~=><>=<=
уравнение (х, у)ne (x, y)gt (x, y)lt (х, у)ge (x, y)ле (х, у)
Фортран 90[примечание 7], Haskell==/=><>=<=
Mathematica[3]==!=><>=<=
Равно [x, y]Неравно [x, y]Больше [x, y]Меньше [x, y]GreaterEqual [x, y]LessEqual [x, y]
  1. ^ Включая FORTRAN II, III, IV, 66 и 77.
  2. ^ АЛГОЛ 68: строппинг режимы используются в коде на платформах с ограниченными наборами символов (например, использовать >= или же GE вместо ), платформы без смелый акцент (использовать 'ge') или платформы только с ВЕРХНИЙ РЕГИСТР (использовать .GE или же 'GE').
  3. ^ Включая АЛГОЛ, Симула, Модула-2, Эйфель, SQL, формулы электронных таблиц, и другие.
  4. ^ Включая C, C ++, C #, Идти, Ява, JavaScript, Perl (только численное сравнение), PHP, Python, Рубин, и р.
  5. ^ Включая Оболочка Борна, Баш, KornShell, и Windows PowerShell. Символы < и > обычно используются в оболочке для перенаправление, поэтому необходимо использовать другие символы. Без дефиса используется в Perl для сравнения строк.
  6. ^ MATLAB, хотя в других отношениях использует синтаксис, аналогичный C, не использует !=, так как ! в MATLAB отправляет следующий текст как командную строку в Операционная система. Первая форма также используется в Болтовня, за исключением равенства, которое =.
  7. ^ Включая FORTRAN 95, 2003, 2008 и 2015 гг.

Другие условности встречаются реже: Common Lisp и Macsyma /Максима используйте операторы типа Basic, за исключением неравенства, которое /= в Common Lisp и # в Macsyma / Maxima. Старшая Лиспы использовал равный, больше, и менее; и отверг их, используя нет для остальных операторов.

Синтаксис

Операторы отношения также используются в технической литературе вместо слов. Операторы отношения обычно записываются на инфиксная запись, если они поддерживаются языком программирования, что означает, что они появляются между своими операндами (двумя связанными выражениями). Например, выражение в Python напечатает сообщение, если Икс меньше чем у:

если Икс < у:    Распечатать("x меньше y в этом примере")

Другие языки программирования, такие как Лисп, использовать префиксная запись, следующее:

(>= Икс Y)

Цепочка операторов

В математике обычной практикой является объединение операторов отношения в цепочку, например, в 3 и х <у и у <20). Синтаксис ясен, поскольку эти операторы отношения в математике транзитивны.

Однако многие современные языки программирования увидят выражение типа 3 (3 <х) <у. Если мы скажем, что x = 4, то получим (3 <4) <у, и оценка даст правда <у что вообще не имеет смысла. Однако он компилируется на C / C ++ и некоторых других языках, давая удивительный результат (как истинный будет представлен здесь цифрой 1).

Можно дать выражение х <у <г его знакомое математическое значение, а также некоторые языки программирования, такие как Python и Раку сделай это. Другие, такие как C # и Java, этого не делают, отчасти потому, что это будет отличаться от того, как большинство других инфиксных операторов работают в C-подобных языках. В Язык программирования D не делает этого, поскольку поддерживает некоторую совместимость с C, и «разрешение выражений C, но с несколько иной семантикой (хотя, возможно, в правильном направлении) добавило бы больше путаницы, чем удобства».[4]

Некоторые языки, например Common Lisp используйте для этого предикаты с несколькими аргументами. В Лиспе (<= 1 х 10) истинно, когда x находится между 1 и 10.

Путаница с операторами присваивания

Ранний FORTRAN (1956–57) был ограничен сильно ограниченными наборами символов, где = был единственным доступным оператором отношения. Не было < или же > (и уж точно нет или же ). Это заставило дизайнеров определить такие символы, как .GT., .LT., .GE., .EQ. и т. д., и впоследствии возник соблазн использовать оставшиеся = символ для копирования, несмотря на очевидную несогласованность с математическим употреблением (Х = Х + 1 должно быть невозможно).

Международный алгебраический язык (IAL, АЛГОЛ 58 ) и АЛГОЛ (1958 и 1960) таким образом представили := по назначению, оставив стандарт = доступны для равенства, соглашение, за которым следует CPL, АЛГОЛ W, АЛГОЛ 68, Базовый комбинированный язык программирования (BCPL ), Симула, УСТАНОВИТЬ язык (SETL ), Паскаль, Болтовня, Модула-2, Ада, Стандартный ML, OCaml, Эйфель, Object Pascal (Delphi ), Оберон, Дилан, Язык описания оборудования VHSIC (VHDL ) и несколько других языков.

B и C

Этот единый де-факто стандарт среди большинства языков программирования в конечном итоге был косвенно изменен минималистским компилируемым языком под названием B. Его единственное предполагаемое применение было как средство передвижения для первого порта (тогда еще очень примитивного) Unix, но он также превратился в очень влиятельную C язык.

B начинался как синтаксически измененный вариант языка системного программирования. BCPL, упрощенная (и безтиповая) версия CPL. В процессе того, что было описано как «разборка», и и или же операторы BCPL[5] были заменены на & и | (который позже станет && и ||, соответственно.[6]). Таким же образом стиль АЛГОЛ := BCPL был заменен на = в Б. Причина всего этого неизвестна.[7] Поскольку обновления переменных не имели специального синтаксиса в B (например, позволять или подобное) и допускались в выражениях, это нестандартное значение знака равенства означало, что традиционная семантика знака равенства теперь должна была быть связана с другим символом. Кен Томпсон использовал специальный == комбинация для этого.

Поскольку позже была введена небольшая система типов, B затем стал C. Популярность этого языка вместе с его ассоциацией с Unix привела к появлению Java, C # и многих других языков, которые следовали этому примеру, синтаксически, несмотря на этот ненужный конфликт с математическим значением знак равенства.

Языки

Присвоения в C имеют ценить и поскольку любое ненулевое скалярное значение интерпретируется как истинный в условные выражения,[8] код если (x = y) законно, но имеет совсем другое значение, чем если (х == у). Прежний фрагмент кода означает «назначить у к Икс, и если новое значение Икс не равно нулю, выполните следующую инструкцию ". Последний фрагмент означает"если и только если Икс равно у, выполните следующую инструкцию ".[9]

  int Икс = 1;  int у = 2;  если (Икс = у) {      / * Этот код всегда будет выполняться, если y не равно 0 * /      printf("x равно% d, а y равно% d п", Икс, у);  }

Хотя Ява и C # имеют те же операторы, что и C, эта ошибка обычно вызывает ошибку компиляции на этих языках, потому что условие if должно иметь тип логический, и не существует неявного способа преобразования из других типов (например, числа) в логическийс. Поэтому, если переменная, присвоенная переменной, не имеет типа логический (или тип оболочки Булево), произойдет ошибка компиляции.

В АЛГОЛ-подобных языках, таких как Pascal, Delphi и Ada (в том смысле, что они позволяют определения вложенных функций ), И в Python, и многие функциональные языки, среди прочего, операторы присваивания не могут появляться в выражение (включая если пункты), что исключает этот класс ошибок. Некоторые компиляторы, например Коллекция компиляторов GNU (GCC), выдает предупреждение при компиляции кода, содержащего оператор присваивания внутри оператора if, хотя есть некоторые законные способы использования присваивания внутри условия if. В таких случаях назначение должно быть явно заключено в дополнительную пару круглых скобок, чтобы избежать предупреждения.

Точно так же некоторые языки, такие как БАЗОВЫЙ используйте только = символ для обоих назначений и равенство, поскольку они синтаксически разделены (как в Паскале, Аде, Python и т. д., операторы присваивания не могут появляться в выражениях).

Некоторые программисты имеют привычку писать сравнения с константой в обратном обычному порядке:

  если (2 == а) {   / * Ошибочное использование = по сравнению с == будет ошибкой времени компиляции * /  }

Если = используется случайно, полученный код недействителен, потому что 2 не является переменной. Компилятор выдаст сообщение об ошибке, в котором можно заменить соответствующий оператор. Этот стиль кодирования называется левосторонним сравнением или Условия Йоды.

В этой таблице перечислены различные механизмы для проверки этих двух типов равенства на разных языках:

ЯзыкФизическое равенствоСтруктурное равенствоПримечания
АЛГОЛ 68а: =: б или же а является ба = бкогда а и б указатели
C, C ++а == б* а == * бкогда а и б указатели
C #объект.ReferenceEquals (a, b)а) равны (б)В == оператор по умолчанию Ссылка, но может быть перегружен выполнять Равно вместо.
Common Lisp(уравнение а б)(равно a b)
Erlangа =: = ба == бкогда a и b числа
Идтиа == ботражать.DeepEqual (* a, * b)когда a и b указатели
Яваа == бa.equals (b)
JavaScriptа === ба == бкогда a и b являются двумя строковыми объектами, содержащими эквивалентные символы, оператор === все равно вернет true.
OCaml, Болтовняа == ба = б
Паскальа ^ = Ь ^а = б
Perl$ a == $ b$$ a == $$ bкогда $ а и $ млрд ссылки на скаляры
PHP$ a === $ b$ a == $ bкогда $ а и $ млрд объекты
Pythonа это ба == б
Рубина. равно? (б)а == б
Схема(ур? а б)(равно? a b)
Быстрыйа === ба == бкогда a и b имеют тип класса
Visual Basic .NET[неравенство 1]а есть б или же объект.ReferenceEquals (a, b)а = б или же а) равны (б)То же, что и C #
Цель-C (Какао, GNUstep )а == б[a isEqual: b]когда а и б указатели на объекты, которые являются экземплярами NSObject
  1. ^ Заявка на патент: 14 мая 2003 г. Заявка США 20,040,230,959  «НЕ ОПЕРАТОР» было подано на НЕ ЯВЛЯЕТСЯ оператора сотрудниками Microsoft. Этот патент был выдан 18 ноября 2004 г.

Ruby использует а === б означать, что «b является членом множества a», хотя детали того, что значит быть членом, значительно различаются в зависимости от задействованных типов данных. === здесь известен как оператор «равенство случаев» или «подчинение случаев».

Смотрите также

Примечания и ссылки

  1. ^ Авторы. «Сравнение объектов». Руководство по PHP. Группа PHP. Получено 29 июня, 2014.
  2. ^ «PHP: операторы сравнения - Руководство». Получено 31 июля, 2008.
  3. ^ Реляционные и логические операторы из Mathematica
  4. ^ Александреску, Андрей. Язык программирования D. Эддисон Уэсли. п. 58. ISBN  978-0-321-63536-5.
  5. ^ Используется не только в АЛГОЛ-подобных языках, но также в FORTRAN и BASIC.
  6. ^ Поскольку некоторые программисты были сбиты с толку двойным значением (побитовый оператор и логическая связка) этих новых символов (согласно Деннис Ричи ). Только побитовое значение & и | были сохранены.
  7. ^ Несмотря на то что Деннис Ричи предположил, что это могло быть связано с «экономией на вводе текста», поскольку обновления переменных могут происходить чаще, чем сравнения в определенных типах программ.
  8. ^ Нулевое скалярное значение интерпретируется как ложное, а любое ненулевое скалярное значение интерпретируется как истинное; это обычно используется с целочисленными типами, аналогично язык ассемблера идиомы.
  9. ^ Брайан Керниган и Деннис Ричи (1988) [1978]. Язык программирования C (Второе изд.). Прентис Холл., 19