Операторы логического AND и OR всегда обрабатывают свой левый операнд перед правым. Кроме того, правый операнд обрабатывается, если и только если левый операнд не определил результат. Эта стратегия известна как вычисление по сокращенной схеме (short-circuit evaluation).
• Правая сторона оператора && вычисляется, если и только если левая сторона истинна.
• Правая сторона оператора || вычисляется, если и только если левая сторона ложна.
Оператор логического AND использовался в некоторых из программ главы 3. Эти программы использовали левый операнд для проверки, безопасно ли выполнять правый операнд. Например, условие цикла for в разд 3.2.3: сначала проверялось, что index не достиг конца строки:
index != s.size() && ! isspace(s[index])
Это гарантировало, что правый операнд не будет выполнен, если индекс уже вышел из диапазона.
Рассмотрим пример применения оператора логического OR. Предположим, что в векторе строк имеется некоторый текст, который необходимо вывести, добавляя символ новой строки после каждой пустой строки или после строки, завершающейся точкой. Для отображения каждого элемента используем серийный оператор for (раздел 3.2.3):
// обратите внимание, s - ссылка на константу; элементы не копируются и
// не могут быть изменены
for (const auto &s : text) { // для каждого элемента text
cout << s; // вывести текущий элемент
// пустые строки и строки, завершающиеся точкой, требуют новой строки
if (s.empty() || s[s.size() - 1] == '.')
cout << endl;
else
cout << " "; // в противном случае отделить пробелом
}
После вывода текущего элемента выясняется, есть ли необходимость выводить новую строку. Условие оператора if сначала проверяет, не пуста ли строка s. Если это так, то необходимо вывести новую строку независимо от значения правого операнда. Только если строка не пуста, обрабатывается второе выражение, которое проверяет, не заканчивается ли строка точкой. Это выражение полагается на вычисление по сокращенной схеме оператора ||, гарантирующего индексирование строки s, только если она не пуста.
Следует заметить, что переменная s объявлена как ссылка на константу (см. раздел 2.5.2). Элементами вектора text являются строки, и они могут быть очень большими, а использование ссылки позволяет избежать их копирования. Поскольку запись в элементы не нужна, объявляем s ссылкой на константу.
Оператор логического NOT (!) возвращает инверсию исходного значения своего операнда. Этот оператор уже использовался в разделе 3.2.2. В следующем примере подразумевается, что vec — это вектор целых чисел, для проверки наличия значений в элементах которого используется оператор логического NOT для значения, возвращенного функцией empty().
// отобразить первый элемент вектора vec, если он есть
if (!vec.empty())
cout << vec[0];
Подвыражение !vec.empty() возвращает значение true, если вызов функции empty() возвращает значение false.
Операторы отношения (<, <=, >, <=) имеют свой обычный смысл и возвращают значение типа bool. Эти операторы имеют левосторонний порядок.
Поскольку операторы отношения возвращают логическое значение, их сцепление может дать удивительный результат:
// Упс! это условие сравнивает k с результатом сравнения i < j
if (i < j < k) // true, если k больше 1!
Условие группирует i и j в первый оператор <. Результат этого выражения (типа bool) является левым операндом второго оператора <. Таким образом, переменная k сравнивается с результатом (true или false) первого оператора сравнения! Для реализации той проверки, которая и предполагалась, выражение нужно переписать следующим образом:
// условие истинно, если i меньше, чем j, и j меньше, чем k