Search

5.5 — Comma and conditional operators

The comma operator

Operator Symbol Form Operation
Comma , x, y Evaluate x then y, returns value of y

The comma operator (,) allows you to evaluate multiple expressions wherever a single expression is allowed. The comma operator evaluates the left operand, then the right operand, and then returns the result of the right operand.

For example:

First the left operand of the comma operator is evaluated, which increments x from 1 to 2. Next, the right operand is evaluated, which increments y from 2 to 3. The comma operator returns the result of the right operand (3), which is subsequently printed to the console.

Note that comma has the lowest precedence of all the operators, even lower than assignment. Because of this, the following two lines of code do different things:

This makes the comma operator somewhat dangerous to use.

In almost every case, a statement written using the comma operator would be better written as separate statements. For example, the above code could be written as:

Most programmers do not use the comma operator at all, with the single exception of inside for loops, where its use is fairly common. We discuss for loops in future lesson 5.7 -- For statements.

Best practice

Avoid using the comma operator, except within for loops.

Comma as a separator

In C++, the comma symbol is often used as a separator, and these uses do not invoke the comma operator. Some examples of separator commas:

There is no need to avoid separator commas (except when declaring multiple variables, which you should not do).

The conditional operator

Operator Symbol Form Operation
Conditional ?: c ? x : y If c is nonzero (true) then evaluate x, otherwise evaluate y

The conditional operator (?:) (also sometimes called the “arithmetic if” operator) is a ternary operator (it takes 3 operands). Because it has historically been C++’s only ternary operator, it’s also sometimes referred to as “the ternary operator”. However, do not call it this, as C++20 is adding a second ternary operator (the three-way comparison operator, also known as the “spaceship operator”).

The ?: operator provides a shorthand method for doing a particular type of if/else statement. Please review lesson 4.10 -- Introduction to if statements if you need a brush up on if/else before proceeding.

An if/else statement takes the following form:

if (condition)
    statement1;
else
    statement2;

If condition evaluates to true, then statement1 is executed, otherwise statement2 is executed.

The ?: operator takes the following form:

(condition) ? expression1 : expression2;

If condition evaluates to true, then expression1 is executed, otherwise expression2 is executed. Note that expression2 is not optional.

Consider an if/else statement that looks like this:

can be rewritten as:

In such uses, the conditional operator can help compact code without losing readability.

Parenthesization of the conditional operator

It is common convention to put the conditional part of the operation inside of parenthesis, both to make it easier to read, and also to make sure the precedence is correct. The other operands evaluate as if they were parenthesized, so explicit parenthesization is not required for those.

Note that the ?: operator has a very low precedence. If doing anything other than assigning the result to a variable, the whole ?: operator also needs to be wrapped in parenthesis.

For example, to print the larger of values x and y to the screen, we could do this:

Or we could use the conditional operator to do this:

Let’s examine what happens if we don’t parenthesize the whole conditional operator in the above case.

Because the << operator has higher precedence than the ?: operator, the statement:

would evaluate as:

That would print 1 (true) if x > y, or 0 (false) otherwise!

Best practice

Always parenthesize the conditional part of the conditional operator, and consider parenthesizing the whole thing as well.

The conditional operator evaluates as an expression

Because the conditional operator operands are expressions rather than statements, the conditional operator can be used in some places where if/else can not.

For example, when initializing a const variable:

There’s no satisfactory if/else statement for this. You might think to try something like this:

However, this won’t compile, and you’ll get an error message that classSize isn’t defined. Much like how variables defined inside functions die at the end of the function, variables defined inside an if or else statement die at the end of the if or else statement. Thus, classSize has already been destroyed by the time we try to print it.

If you want to use an if/else, you’d have to do something like this:

This one works because we’re not defining variables inside the if or else, we’re just returning a value back to the caller, which can then be used as the initializer.

That’s a lot of extra work!

The type of the expressions must match or be convertible

To properly comply with C++’s type checking, both expressions in a conditional statement must either match, or the second expression must be convertible to the type of the first expression.

So while you might expect to be able to do something like this:

The above example won’t compile. One of the expressions is an integer, and the other is a string literal. The compiler will try to find a way to convert the string literal to an integer, but since it doesn’t know how, it will give an error. In such cases, you’ll have to use an if/else.

So when should you use the conditional operator?

The conditional operator gives us a convenient way to compact some if/else statements. It’s most useful when we need a conditional initializer (or assignment) for a variable, or to pass a conditional value to a function.

It should not be used for complex if/else statements, as it quickly becomes both unreadable and error prone.

Best practice

Only use the conditional operator for simple conditionals where it enhances readability.


5.6 -- Relational operators and floating point comparisons
Index
5.4 -- Increment/decrement operators, and side effects

91 comments to 5.5 — Comma and conditional operators

  • Matt

    Wow, more info regarding the language that my Intro course 'conveniently' left out. I am beginning to see just how much in terms of concepts, tools, and best practices were completely omitted from the assigned C++ textbook. Unlearning the bad habits the book taught is going to take some work.

    (If your curious, the textbook I was assigned was "C++ Programming: From Problem Analysis to Program Design" by D.S. Malik, and the very first thing it does it teaches its students use namespace std instead scope reducing with std::, and the professor never offered any feedback for improving code in any way.)

    • Yep, a lot of teaching resources teach `using namespace std;`, because it's convenient and makes to code smaller. But even if they say that it's bad, when they use it, their students will too.

  • Benur21

    I think it's worth noting that the else part on these ?: statements is not optional like in the if's, because it always must return something. Sometimes we could use empty string "" if we want it to return nothing. For example, in the following code I found at lesson 5.5:

    I was thinking if I could do like this:

    but obviously it needs an else return.

  • Randle

    I remember learning the conditional operator in uni, but was never shown a proper example where it was necessary, so I ignored it. Thank you for providing a clear example of its usage!

  • Brandon

    How could I use the conditional operator in place of an if/else with multiple statements? I'm practicing by converting if/else statements to use the conditional operator but what about those with multiple else if's like a switch statement? Is that possible? or would it be bad practice since the point is to reduce the clutter of code? Thanks.

    EDIT: I just noticed " It’s worth noting that the conditional operator evaluates as an expression, whereas if/else evaluates as a set of statements."

    But still, can I only evaluate a single expression at once right?  Sorry if this sounds weird.

    • Hi Brandon!

      The conditional operator should only be used when you need the result of the expression.

      You could use the comma operator to get what you want

      But don't do it, it's ugly, use if-statements.

  • Jules

    So i wrote a simple assembly program to find the square of a number (2 in this case, haven't given the functionality of user input):
    #include "pch.h"
    #include <iostream>
    int power(int);
    int main()
    {
        std::cout << "Hello World!\n";
        std::cout<<power(2);
    }

    int power(int x)
    {
        _asm
        {
            mov eax,x
            imul eax,x
            mov x,eax

        }
        return x;
        
    }

    my question is :
    in the instruction mov x,eax does the compiler already know that the contents of eax should be moved into the MEMORY LOCATION of "x"
    because the contents of the eax register cannot be moved into a variable.

    consequently mov [x],eax works as well.
    just wanted to know if the c++ compiler is optimized in handling asm in a way that traditional assemblers like NASM or FASM aren't.

  • Jeremy

    I can understand the logic of variables inside of functions being destroyed after the function call is over, but is there a specific reason why if/else statements share the same scope?

    • Alex

      The C++ specification says that if the if-statement or else-statement are not written as compound statements, they should be treated as if they were. So:

      Is treated as if it were written:

      And although we haven't covered blocks/compound statements yet (that happens in chapter 4), they behave similarly to functions in that local variables defined inside a block die at the end of the block.

  • aleksandr.a

    So you can use the conditional operator?
    Thank.

  • Rai

    I tried using the conditional operator with one of the older lesson project.

    How would you use conditional operators for the if statement here? I tried doing it with this :

    But I was getting an error saying "error C4716: 'getInteger': must return a value"

    • Without the full @getInterger function I can't tell you what your problem is.
      Here's how you could do it

    • leonidus007

      Maybe this can help..

      #include <iostream>
      int getInteger()
      {
          std::cout << "Enter a single digit integer: " << std::endl;
          int input;
          std::cin >> input;
          return input > 0 ? (input < 10 ? input : -1) : -1;
      }
      int main()
      {
          int y{ getInteger() };
          std::cout << y;
          return 0;
      }

Leave a Comment

Put all code inside code tags: [code]your code here[/code]