Boolean Expressions

The parenthesized condition of the if statement is a Boolean expression. In Listing 4.29, the condition is highlighted.

Listing 4.29: Boolean Expression
1. if (input < 9)
2. {
3.     // Input is less than 9
4.     Console.WriteLine(
5.         $"Tic-tac-toe has more than {input}" +
6.          " maximum turns.");
7. }
8. // ...

Boolean expressions appear within many control flow statements. Their key characteristic is that they always evaluate to true or false. For input < 9 to be allowed as a Boolean expression, it must result in a bool. The compiler disallows x = 42, for example, because this expression assigns x and results in the value that was assigned instead of checking whether the value of the variable is 42.

Language Contrast: C++—Mistakenly Using = in Place of ==

C# eliminates a coding error commonly found in C and C++. In C++, Listing 4.30 is allowed.

Listing 4.30: C++, but Not C#, Allows Assignment as a Condition
1. if (input = 9) // Allowed in C++, not in C#
2.   System.Console.WriteLine(
3.     "Correct, tic-tac-toe has a maximum of 9 turns.");

Although at first glance this code appears to check whether input equals 9, Chapter 1 showed that = represents the assignment operator, not a check for equality. The return from the assignment operator is the value assigned to the variable—in this case, 9. However, 9 is an int, so it does not qualify as a Boolean expression and is not allowed by the C# compiler. The C and C++ languages treat integers that are nonzero as true and integers that are zero as false. C#, by contrast, requires that the condition actually be of a Boolean type; integers are not allowed.

Relational and Equality Operators

Relational and equality operators determine whether a value is greater than, less than, or equal to another value. Table 4.2 lists all the relational and equality operators. All are binary operators.

Table 4.2: Relational and Equality Operators

Operator

Description

Example

<

Less than

input < 9;

>

Greater than

input > 9;

<=

Less than or equal to

input <= 9;

>=

Greater than or equal to

input >= 9;

==

Equality operator

input == 9;

!=

Inequality operator

input != 9;

The C# syntax for equality uses ==, just as many other programming languages do. For example, to determine whether input equals 9, you use input == 9. The equality operator uses two equal signs to distinguish it from the assignment operator, =. The exclamation point signifies NOT in C#, so to test for inequality you use the inequality operator, !=.

Relational and equality operators always produce a bool value, as shown in Listing 4.31.

Listing 4.31: Assigning the Result of a Relational Operator to a bool Variable
1. bool result = 70 > 7;

In the full tic-tac-toe program listing, you use the equality operator to determine whether a user has quit. The Boolean expression in Listing 4.32 includes an OR (||) logical operator, which the next section discusses in detail.

Listing 4.32: Using the Equality Operator in a Boolean Expression
1. if (input.Length == 0 || input == "quit")
2. {
3.     Console.WriteLine($"Player {currentPlayer} quit!!");
4. }
Logical Boolean Operators

The logical operators have Boolean operands and produce a Boolean result. Logical operators allow you to combine multiple Boolean expressions to form more complex Boolean expressions. The logical operators are |, ||, &, &&, and ^, corresponding to OR, AND, and exclusive OR. The | and & versions of OR and AND are rarely used for Boolean logic, for reasons which we discuss in this section.

OR Operator (||)

In Listing 4.32, if the user enters quit or presses the Enter key without typing in a value, it is assumed that they want to exit the program. To enable two ways for the user to resign, you can use the logical OR operator, ||. The || operator evaluates Boolean expressions and results in a true value if either operand is true (see Listing 4.33).

Listing 4.33: Using the OR Operator
1. if ((hourOfTheDay > 23) || (hourOfTheDay < 0))
2.     Console.WriteLine("The time you entered is invalid.");

It is not necessary to evaluate both sides of an OR expression, because if one operand is true, the result is known to be true regardless of the value of the other operand. Like all operators in C#, the left operand is evaluated before the right one, so if the left portion of the expression evaluates to true, the right portion is ignored. In the example in Listing 4.33, if hourOfTheDay has the value 33, then (hourOfTheDay > 23) evaluates to true and the OR operator ignores the second half of the expression, short-circuiting it. Short-circuiting an expression also occurs with the Boolean AND operator. (Note that the parentheses are not necessary here; the logical operators are of lower precedence than the relational operators. However, it is clearer to the novice reader when the subexpressions are parenthesized for clarity.)

AND Operator (&&)

The Boolean AND operator, &&, evaluates to true only if both operands evaluate to true. If either operand is false, the result will be false. Listing 4.34 writes a message if the given variable is both greater than 10 and less than 24.3 Similarly to the OR operator, the AND operator will not always evaluate the right side of the expression. If the left operand is determined to be false, the overall result will be false regardless of the value of the right operand, so the runtime skips evaluating the right operand.

Listing 4.34: Using the AND Operator
1. if ((10 < hourOfTheDay) && (hourOfTheDay < 24))
2.     Console.WriteLine(
3.         "Hi-Ho, Hi-Ho, it's off to work we go.");
Exclusive OR Operator (^)

The caret symbol, ^, is the exclusive OR (XOR) operator. When applied to two Boolean operands, the XOR operator returns true only if exactly one of the operands is true, as shown in Table 4.3.

Table 4.3: Conditional Values for the XOR Operator

Left Operand

Right Operand

Result

True

True

False

True

False

True

False

True

True

False

False

False

Unlike the Boolean AND and Boolean OR operators, the Boolean XOR operator does not short-circuit: It always checks both operands, because the result cannot be determined unless the values of both operands are known. Note that the XOR operator is exactly the same as the Boolean inequality operator.

Logical Negation Operator (!)

The logical negation operator, or NOT operator, !, inverts a bool value. This operator is a unary operator, meaning it requires only one operand. Listing 4.35 demonstrates how it works, and Output 4.16 shows the result.

Listing 4.35: Using the Logical Negation Operator
1. bool valid = false;
2. bool result = !valid;
3. // Displays "result = True"
4. Console.WriteLine($"result = { result }");
Output 4.16
result = True

At the beginning of Listing 4.35, valid is set to false. You then use the negation operator on valid and assign the value to result.

Conditional Operator (?:)

As an alternative to using an if-else statement to select one of two values, you can use the conditional operator. The conditional operator uses both a question mark and a colon; the general format is as follows:

condition ? consequent : alternative

The conditional operator is a ternary operator because it has three operands: condition, consequent, and alternative. (As it is the only ternary operator in C#, it is often called the ternary operator, but it is clearer to refer to it by its name than by the number of operands it takes.) Like the logical operators, the conditional operator uses a form of short-circuiting. If the condition operator evaluates to true, the conditional operator evaluates only consequent. If the conditional evaluates to false, it evaluates only alternative. The result of the operator is the evaluated expression.

Listing 4.36 illustrates the use of the conditional operator. The full listing of this program appears in of the source code.

Listing 4.36: Conditional Operator
1. public class TicTacToe
2. {
3.     public static void Main()
4.     {
5.         // Initially set the currentPlayer to Player 1
6.         int currentPlayer = 1;
7.  
8.         // ...
9.  
10.         for(int turn = 1; turn <= 10; turn++)
11.         {
12.             // ...
13.  
14.             // Switch players
15.             currentPlayer = (currentPlayer == 2) ? 1 : 2;
16.         }
17.     }
18. }

The program swaps the current player. To do so, it checks whether the current value is 2. This is the conditional portion of the conditional expression. If the result of the condition is true, the conditional operator results in the consequent value, 1. Otherwise, it results in the alternative value, 2. Unlike in an if statement, the result of the conditional operator must be assigned (or passed as a parameter); it cannot appear as an entire statement on its own.

Guidelines
CONSIDER using an if-else statement instead of an overly complicated conditional expression.

Prior to C# 9.0, the language required that the output (the consequent and alternative) expressions in a conditional operator be consistently typed and that the consistent type be determined without examination of the surrounding context of the expression, including the resulting type assigned. However, C# 9.0 added support for target-typed conditional expressions so that even if the consequent and alternative expressions are of different types (such as string and int) and neither is convertible to the other, the statement is still allowed if both types will implicitly cast to the targeted type. For example, even if you have object result = condition ? "abc" : 123;, the C# compiler will allow it because both the potential conditional output types will implicitly cast to object.

________________________________________

3. The typical hours that programmers work each day.
{{ snackbarMessage }}
;