What is short-circuit evaluation in C++?
Short-circuit evaluation is a behavior of logical operators in C++ where the second part of an expression is evaluated only if it is necessary to determine the final result.
For example, in an expression using the logical OR operator, if the left-hand side already evaluates to true, the right-hand side is never executed because the entire expression must already be true.
if (is_admin || hasPermission())
{
// ...
}If is_admin is true, the function hasPermission() will never be called. The same rule applies to the logical AND operator in the opposite direction: if the left-hand side is false, the right-hand side is skipped. Such behavior is often beneficial because it avoids unnecessary work.
How does this matter in practice?
Short-circuit evaluation becomes especially important in unit tests when mocking calls used as parts of if statements. Consider the following code:
class PermissionService
{
public:
virtual bool HasPermission() = 0;
};
bool CanAccess(bool is_admin, PermissionService& service)
{
if (is_admin && service.HasPermission())
{
return true;
}
return false;
}At first glance, one of the unit tests testing the above code could look like this:
class MockPermissionService : public PermissionService
{
public:
MOCK_METHOD(bool, HasPermission, (), (override));
};
TEST(CanAccessTest, ReturnsTrueForAdmin)
{
StrictMock<MockPermissionService> service;
EXPECT_CALL(service, HasPermission()).WillOnce(Return(true));
EXPECT_FALSE(CanAccess(false, service));
}However, when you run this test, you’ll see that it failed with the following error:
Actual function call count doesn't match EXPECT_CALL(service, HasPermission())...
Expected: to be called once
Actual: never called - unsatisfied and activeHow is this possible that HasPermission() was never called if we clearily see it in the tested code? The answer is: short-circuit evaluation.
Since is_admin is false, C++ short-circuit evaluation stops the expression immediately. The call to service.HasPermission() is never executed and as a result the mocked method is never called, so the EXPECT_CALL expectation is not satisfied.
The failure is not caused by incorrect production code. The issue is that the test assumes both sides of the condition are evaluated, while the language guarantees that only the necessary side is executed.
The correct test should reflect the actual execution path:
TEST(CanAccessTest, ReturnsTrueForAdmin)
{
StrictMock<MockPermissionService> service;
EXPECT_TRUE(CanAccess(false, service));
}Or if we specifically want to verify the permission check path, we should make the first condition evaluate to false:
bool CanAccess(bool is_admin, PermissionService& service)
{
if (service.HasPermission() && is_admin)
{
return true;
}
return false;
}Read also:
- GTest and short-circuit evaluation in C++
- AI is powerful. Snippets are instant.
- From AUTOSAR to S-Core: the first C++ pub/sub implementation
- How to write Arduino Uno code with Python?
- Combining Bazel with Docker
- Running commands with timeout on Linux
- Running Python unit tests with CMake
- Thirdparty dependencies with FetchContent
- Bug of the week #11
- Combining CMake with Docker
- How to search the internet from Linux terminal?
- Folding expressions in C++
- How to derive from an enum in Python?
- Bug of the week #10
- Trying ROS2: client/server within a single container
- Make C++ a better place #4: Go as an alternative
- How to convert hex to dec in Linux terminal?
- Setting up a Python project with CMake
- Separating builds for different configs with Bazel
- Trying ROS2: pub/sub within a single container
- Bug of the week #9
- UDP multicasting with Python
- Destruction order vs thread safety in C++
- Let’s review some code: C++ #2
- Make C++ a better place #3: D as an alternative
- Registering callback using std::function in C++
- Bug of the week #8
- TCP client/server with Python
- Simple menus in Bash scripts with select
- Calling member function on a nullptr in C++
- Bug of the week #7
- Python lru_cache explained
- How to dockerize a Python application?
- Make C++ a better place #2: CppFront as an alternative
- Parameters combinations in GoogleTest
- Data transfer with curl
- Python reduce explained
- Bug of the week #6
- Custom literals in C++
- Linux and hash command
- 5 Python good practices which make life easier
- Let’s review some code: Python #1
- Make C++ a better place #1: What does better mean
- Enums vs enum class in C++
- Bug of the week #5
- UDP client/server with Python
- Hard links in Linux
- Functions calling order in unit tests in C++
- Bug of the week #4
- Yield in Python – state machines, coroutines and more
- Copy files from another branch with Git
- Make C++ a better place #0: Introduction
- 5 misconceptions about std::move in C++
- How to use xargs on Linux?
- How to test method call order with unittest in Python?
- Bug of the week #3
- Build & run C++ unit tests with CMake
- Arrange text with sort on Linux
- Key derivation function with Python
- Let’s review some code #1: C++









