Java provides three operators that you can use to form logical expressions:
Table2.11.1.Java Logical Operators
Operator
Type
Description
&&
infix, binary
Logical conjunction (AND)
||
infix, binary
Logical disjunction (AND)
!
prefix, unary
Negation (NOT)
Logical expressions combine subexpressions using logical operators. All input operands for logical operators must be boolean in type, and logical expressions always evaluate to a boolean value. In short, when it comes to logical expressions, its booleans in and booleans out.
Logical operator precedence is lower than relational operators. That means that relational subexpressions involving inequalities used as operands for a logical operator, will be evaluated before the logical operator is evaluated. This is a good thing because relational expressions evaluate exclusively to booleans and logical expression operands are exclusively boolean, so the two work together nicely.
With the conjunction operator (&&), if both the first operand AND the second operand both evaluate to true, then the conjunction expression evaluates to true. If either the first or second operand evaluates to false, then the conjunction expression is false.
As an example, if day is a variable holding the number of the day in any given month following is a logical expression that tells us to pay the rent during the first week of the month: day >= 1 && day <=7.
The disjunction operator (||) is similar. If either the first operand OR the second operand evaluates to true, then the disjunction expression evaluates to true. If both the first and second operand evaluates to false, then the disjunction expression is false.
The above also can be summarized as a truth table.
Table2.11.3.Truth Table for Disjunction (||)
A
B
A || B
true
true
true
true
false
true
false
true
true
false
false
false
A comfortable water temperature for an adult to swim laps in a pool is between 78°F and 82°F. What is an expression that tells me when to avoid swimming laps in a pool based on its water temperature? If tempF is the water temperature in °F, tempF < 78 || tempF > 82. If either of these relational expressions is true, we’ll go out for a run instead.
Negation ! is the simplest of the Java logical operators. This prefix unary operator turns a true value into false and a false value into true. Consider the following example and truth table.
Let’s consider a few more practical examples of logical expressions.
Test if age is greater than 17 AND less than or equal to 67.
Logical expression: age > 17 && age <= 67
jshell> int age = 40;
age ==> 40
| created variable age : int
jshell> age > 17 && age <= 67; // Logical test
$2 ==> true
| created scratch variable $2 : boolean
jshell> age = 12;
age ==> 12
| assigned to age : int
jshell> age > 17 && age <= 67; // Logical test
$4 ==> false
| created scratch variable $4 : boolean
Test if age is less-than or equal-to 17 OR greater than 67.
Logical expression: age <= 17 || x > 67
jshell> int age = 40;
age ==> 40
| created variable age : int
jshell> x <= 17 || x > 67; // Logical test
$2 ==> false
| created scratch variable $2 : boolean
jshell> age = 12;
age ==> 12
| assigned to age : int
jshell> age <= 17 || x > 67; // Logical test
$4 ==> true
| created scratch variable $4 : boolean
jshell> age = 70;
age ==> 70
| assigned to age : int
jshell> age <= 17 || x > 67; // Logical test
$6 ==> true
| created scratch variable $6 : boolean
jshell>
Are the last two examples complementary? Is the truth table of one expression equal to the negated truth table of the other? Let’s check.
Logical expression: !(age <= 17 || age > 67) == (age > 17 && age <= 67)
jshell> int age = 40;
age ==> 40
| created variable age : int
jshell> !(age <= 17 || age > 67) == (age > 17 && age <= 67);
$2 ==> true
| created scratch variable $2 : boolean
jshell> age = 12;
age ==> 12
| assigned to aeg : int
jshell> !(age <= 17 || age > 67) == (age > 17 && age <= 67);
$4 ==> true
| created scratch variable $4 : boolean
jshell> age = 70;
age ==> 70
| assigned to age : int
jshell> !(age <= 17 || age > 67) == (age > 17 && age <= 67);
$6 ==> true
| created scratch variable $6 : boolean
jshell>
Forming logical expressions to test complex conditions takes practice. The more you practice the better you’ll be able to form the expressions you need to solve your computing problems.
Subsection2.11.1Practice
You’ve invented a dart game that awards a different number of points to a player when a dart lands within each of four concentric circles. The game board is 600mm by 600mm, so the center is at (300mm, 300mm). The outer (yellow) band is bounded by circles with radii 180mm and 240mm and has a point value of 1. This next (blue) band is bounded by circles with radii 120mm and 180mm and has a point value of 2. This next (green) band is bounded by circles with radii 60mm and 120mm and has a point value of 3. And the center (red) circle has radius of 60mm with a point value of 4. The game board is instrumented with sensors; it reports the (x, y) coordinates in mm when any dart lands.
On the right is an interactive illustration of your game board. Click anywhere on the illustration to see the coordinates that the board senses.
Write a logical expression that evaluates to true if a dart lands within the centermost circle, centered at (300, 300) with radius 60, and false otherwise. Write a second logical expression that evaluates to true if a dart lands within the next circular band (green) between circles with radii 60mm and 120mm.
Figure2.11.5.Dart Game Board
We can test if a point is within a circle by checking if the distance between a point and the center of the circle is less than the radius of the circle. The distance formula \((distance = \sqrt{(x_1 - x_2)^2 + (y_1 - y_2)^2})\) gives us the way to compute the distance between \((x_1, y_1)\) and \((x_2, y_2)\text{,}\) where one of the two points will be chosen as the circle center and the other is the point we are testing. Because our game board is a set of nested circles, the distance formula will be an integral part of our logical expressions.
A simple relational expression will tell us if a dart’s landing coordinate falls within the centermost circle. It is convenient to compute the distance to the circle center first and save it to a temporary variable, and then test the value of that distance variable. The following code snippet initializes variables, computes distance, and tests that distance to see if the dart has landed in the centermost circle. Note how we have organized these computations as a sequence of steps.
jshell> // Circle parameters
...> double xc = 300.0, yc = 300.0;
xc ==> 300.0
yc ==> 300.0
jshell> // Dart coordinates
...> double x1 = 210.0, y1 = 319.0;
x1 ==> 210.0
y1 ==> 319.0
jshell> // Compute distance
...> double dist = Math.sqrt( Math.pow(x1-xc, 2) + Math.pow(y1-yc, 2) );
dist ==> 91.98369420718001
jshell> // Test if in center red circle
...> dist < 60;
false
The final false tells us that the dart did not land in the centermost circle.
How about the next outer circular band with radius greater than or equal to 60 but less than 120. This test is more complex because two conditions must be true simultaneously in order for the dart to have landed in the green band. Here is the logical expression to be evaluated.
// Test if in green circle
dist >= 60 && dist < 120;
Enter this into our previous JShell session.
jshell> dist >= 60 && dist < 120;
true
The dart has landed in the green band. The player is awarded 3 points.
Activity2.11.1.
Demonstrate the following two De Morgan’s laws using Java logical expressions.
jshell> boolean A, B;
A ==> false
| created variable A : boolean
B ==> false
| created variable B : boolean
jshell> !(A || B) == (!A && !B)
$3 ==> true
| created scratch variable $3 : boolean
jshell> A = true
A ==> true
| assigned to A : boolean
jshell> !(A || B) == (!A && !B)
$5 ==> true
| created scratch variable $5 : boolean
jshell> B = true
B ==> true
| assigned to B : boolean
jshell> !(A || B) == (!A && !B)
$7 ==> true
| created scratch variable $7 : boolean
jshell> A = false
A ==> false
| assigned to A : boolean
jshell> !(A || B) == (!A && !B)
$9 ==> true
| created scratch variable $9 : boolean
jshell> A = B = false;
A ==> false
| assigned to A : boolean
jshell> !(A && B) == !A || !B;
$11 ==> true
| created scratch variable $11 : boolean
jshell> A = true;
A ==> true
| assigned to A : boolean
jshell> !(A && B) == !A || !B;
$13 ==> true
| created scratch variable $13 : boolean
jshell> B = true;
B ==> true
| assigned to B : boolean
jshell> !(A && B) == !A || !B;
$15 ==> true
| created scratch variable $15 : boolean
jshell> A = false;
A ==> false
| assigned to A : boolean
jshell> !(A && B) == !A || !B;
$17 ==> true
| created scratch variable $17 : boolean
jshell>