Programming in D – Tutorial and Reference
Ali Çehreli

Other D Resources

Logical Expressions

The actual work that a program performs is accomplished by expressions. Any part of a program that produces a value or a side effect is called an expression. It has a very wide definition because even a constant value like 42 and a string like "hello" are expressions, since they produce the respective constant values 42 and "hello".

Note: Don't confuse producing a value with defining a variable. Values need not be associated with variables.

Function calls like writeln are expressions as well because they have side effects. In the case of writeln, the effect is on the output stream by the placement of characters on it. Another example from the programs that we have written so far would be the assignment operation, which affects the variable that is on its left-hand side.

Because of producing values, expressions can take part in other expressions. This allows us to form more complex expressions from simpler ones. For example, assuming that there is a function named currentTemperature that produces the value of the current air temperature, the value that it produces may directly be used in a writeln expression:

    writeln("It's ", currentTemperature(),
            " degrees at the moment.");

That line consists of four expressions:

  1. "It's "
  2. currentTemperature()
  3. " degrees at the moment."
  4. The writeln() expression that makes use of the other three

In this chapter we will cover the particular type of expression that is used in conditional statements.

Before going further though, I would like to repeat the assignment operator once more, this time emphasizing the two expressions that appear on its left and right sides: the assignment operator (=) assigns the value of the expression on its right-hand side to the expression on its left-hand side (e.g. to a variable).

    temperature = 23      // temperature's value becomes 23
Logical Expressions

Logical expressions are the expressions that are used in Boolean arithmetic. Logical expressions are what makes computer programs make decisions like "if the answer is yes, I will save the file".

Logical expressions can have one of only two values: false that indicates falsity, and true that indicates truth.

I will use writeln expressions in the following examples. If a line has true printed at the end, it will mean that what is printed on the line is true. Similarly, false will mean that what is on the line is false. For example, if the output of a program is the following,

There is coffee: true

then it will mean that "there is coffee". Similarly,

There is coffee: false

will mean that "there is no coffee". I use the "... is ...: false" construct to mean "is not" or "is false".

Logical expressions are used extensively in conditional statements, loops, function parameters, etc. It is essential to understand how they work. Luckily, logical expressions are easy to explain and use.

The logical operators that are used in logical expressions are the following:

Grouping expressions

The order in which the expressions are evaluated can be specified by using parentheses to group them. When parenthesized expressions appear in more complex expressions, the parenthesized expressions are evaluated before they can be used in the expressions that they appear in. For example, the expression "if there is coffee or tea, and also cookie or scone; then I am happy" can be coded as follows:

writeln("I am happy: ",
(existsCoffee || existsTea) && (existsCookie || existsScone));

If the sub expressions were not parenthesized, the expressions would be evaluated according to operator precedence rules of D (which have been inherited from the C language). Since in these rules && has a higher precedence than ||, writing the expression without parentheses would not be evaluated as intended:

writeln("I am happy: ",
existsCoffee || existsTea && existsCookie || existsScone);

The && operator would be evaluated first and the whole expression would be the semantic equivalent of the following expression:

writeln("I am happy: ",
existsCoffee || (existsTea && existsCookie) || existsScone);

That has a totally different meaning: "if there is coffee, or tea and cookie, or scone; then I am happy".

The operator precedence table will be presented later in the book.

Reading bool input

All of the bool values above are automatically printed as "false" or "true". It is the same in the opposite direction: readf() automatically converts strings "false" and "true" to bool values false and true, respectively. It accepts any combination of lower and uppercase letters as well. For example, "False" and "FALSE" are converted to false and "True" and "TRUE" are converted to true.

Note that this is the case only when reading into bool variables. Otherwise, the input would be read as-is without conversion when reading into a string variable. (As we will see later in the strings chapter, one must use readln() when reading strings.)

Exercises
  1. We've seen above that the < and the > operators are used to determine whether a value is less than or greater than another value; but there is no operator that answers the question "is between?" to determine whether a value is between two other values.

    Let's assume that a programmer has written the following code to determine whether value is between 10 and 20. Observe that the program cannot be compiled as written:

    import std.stdio;
    
    void main() {
        int value = 15;
    
        writeln("Is between: ",
                10 < value < 20);        // ← compilation ERROR
    }
    

    Try using parentheses around the whole expression:

        writeln("Is between: ",
                (10 < value < 20));      // ← compilation ERROR
    

    Observe that it still cannot be compiled.

  2. While searching for a solution to this problem, the same programmer discovers that the following use of parentheses now enables the code to be compiled:
        writeln("Is between: ",
                (10 < value) < 20);      // ← compiles but WRONG
    

    Observe that the program now works as expected and prints "true". Unfortunately, that output is misleading because the program has a bug. To see the effect of that bug, replace 15 with a value greater than 20:

        int value = 21;
    

    Observe that the program still prints "true" even though 21 is not less than 20.

    Hint: Remember that the type of a logical expression is bool. It shouldn't make sense whether a bool value is less than 20. The reason it compiles is due to the compiler converting the boolean expression to a 1 or 0, and then evaluating that against 20 to see if it is less.

  3. The logical expression that answers the question "is between?" must instead be coded like this: "is greater than the lower value and less than the upper value?".

    Change the expression in the program according to that logic and observe that it now prints "true" as expected. Additionally, test that the logical expression works correctly for other values as well: for example, when value is 50 or 1, the program should print "false"; and when it is 12, the program should print "true".

  4. Let's assume that we can go to the beach when one of the following conditions is true:
    • If the distance to the beach is less than 10 miles and there is a bicycle for everyone
    • If there is fewer than 6 of us, and we have a car, and at least one of us has a driver license

    As written, the following program always prints "true". Construct a logical expression that will print "true" when one of the conditions above is true. (When trying the program, enter "false" or "true" for questions that start with "Is there a".).

    import std.stdio;
    
    void main() {
        write("How many are we? ");
        int personCount;
        readf(" %s", &personCount);
    
        write("How many bicycles are there? ");
        int bicycleCount;
        readf(" %s", &bicycleCount);
    
        write("What is the distance to the beach? ");
        int distance;
        readf(" %s", &distance);
    
        write("Is there a car? ");
        bool existsCar;
        readf(" %s", &existsCar);
    
        write("Is there a driver license? ");
        bool existsLicense;
        readf(" %s", &existsLicense);
    
        /* Replace the 'true' below with a logical expression that
         * produces the value 'true' when one of the conditions
         * listed in the question is satisfied: */
        writeln("We are going to the beach: ", true);
    }
    

    Enter various values and test that the logical expression that you wrote works correctly.