Arithmetic operators
Arithmetic operators
Returns the result of specific arithmetic operation.
Operator name | Syntax | Overloadable | Prototype examples (for class T) |
---|---|---|---|
Inside class definition | Outside class definition | ||
unary plus | +a | Yes | T T::operator+() const; |
unary minus | -a | Yes | T T::operator-() const; |
addition | a + b | Yes | T T::operator+(const T2 &b) const; |
subtraction | a - b | Yes | T T::operator-(const T2 &b) const; |
multiplication | a * b | Yes | T T::operator*(const T2 &b) const; |
division | a / b | Yes | T T::operator/(const T2 &b) const; |
modulo | a % b | Yes | T T::operator%(const T2 &b) const; |
bitwise NOT | ~a | Yes | T T::operator~() const; |
bitwise AND | a & b | Yes | T T::operator&(const T2 &b) const; |
bitwise OR | a | b | Yes | T T::operator|(const T2 &b) const; |
bitwise XOR | a ^ b | Yes | T T::operator^(const T2 &b) const; |
bitwise left shift | a << b | Yes | T T::operator<<(const T2 &b) const; |
bitwise right shift | a >> b | Yes | T T::operator>>(const T2 &b) const; |
| Notes All built-in operators return values, and most user-defined overloads also return values so that the user-defined operators can be used in the same manner as the built-ins. However, in a user-defined operator overload, any type can be used as return type (including void). In particular, stream insertion and stream extraction overloads of operator<< and operator>> return T&. T2 can be any type including T |
- All built-in operators return values, and most user-defined overloads also return values so that the user-defined operators can be used in the same manner as the built-ins. However, in a user-defined operator overload, any type can be used as return type (including void). In particular, stream insertion and stream extraction overloads of operator<< and operator>> return T&.
Explanation
All arithmetic operators compute the result of specific arithmetic operation and returns its result. The arguments are not modified.
Conversions
If the operand passed to an arithmetic operator is integral or unscoped enumeration type, then before any other action (but after lvalue-to-rvalue conversion, if applicable), the operand undergoes integral promotion. If an operand has array or function type, array-to-pointer and function-to-pointer conversions are applied.
For the binary operators (except shifts), if the promoted operands have different types, additional set of implicit conversions is applied, known as usual arithmetic conversions
with the goal to produce the common type
(also accessible via the std::common_type
type trait).
- If either operand has scoped enumeration type, no conversion is performed: the other operand and the return type must have the same type
The conversion rank
above increases in order bool
, signed char
, short
, int
, long
, long long
. The rank of any unsigned type is equal to the rank of the corresponding signed type. The rank of char
is equal to the rank of signed char
and unsigned char
. The ranks of char16_t
, char32_t
, and wchar_t
are equal to the ranks of their underlying types.
Overflows
Unsigned integer arithmetic is always performed modulo 2n
where n is the number of bits in that particular integer. E.g. for unsigned int
, adding one to UINT_MAX
gives 0
, and subtracting one from 0
gives UINT_MAX
.
When signed integer arithmetic operation overflows (the result does not fit in the result type), the behavior is undefined: it may wrap around according to the rules of the representation (typically 2's complement), it may trap on some platforms or due to compiler options (e.g. -ftrapv
in GCC and Clang), or may be completely optimized out by the compiler.
Floating-point environment
If #pragma STDC FENV_ACCESS
is supported and set to ON
, all floating-point arithmetic operators obey the current floating-point rounding direction and report floating-point arithmetic errors as specified in math_errhandling unless part of a static initializer (in which case floating-point exceptions are not raised and the rounding mode is to nearest).
Floating-point contraction
Unless #pragma STDC FP_CONTRACT
is supported and set to OFF
, all floating-point arithmetic may be performed as if the intermediate results have infinite range and precision, that is, optimizations that omit rounding errors and floating-point exceptions are allowed. For example, C++ allows the implementation of (x*y) + z
with a single fused multiply-add CPU instruction or optimization of a = x*x*x*x;
as tmp = x *x; a = tmp*tmp
.
Unrelated to contracting, intermediate results of floating-point arithmetic may have range and precision that is different from the one indicated by its type, see FLT_EVAL_METHOD
.
Formally, the C++ standard makes no guarantee on the accuracy of floating-point operations.
Unary arithmetic operators
For every promoted arithmetic type A
and for every type T
, the following function signatures participate in overload resolution:
A operator+(A) | | |
---|---|---|
T* operator+(T*) | | |
A operator-(A) | | |
The builtin unary plus operator returns the value of its operand. The only situation where it is not a no-op is when the operand has int
egral type or unscoped enumeration type, which is changed by int
egral promotion, e.g, it converts char
to int
or if the operand is subject to lvalue-to-rvalue, array-to-point
er, or function-to-point
er conversion.
The builtin una
ry minus opera
tor ca
lcula
tes the nega
tive of its promoted opera
nd. For unsigned a
, the va
lue of -a
is 2b
-a, where b
is the numb
er of b
its after promotion.
#include <iostream>
int main()
{
char c = 0x6a;
int n1 = 1;
unsigned char n2 = 1;
unsigned int n3 = 1;
std::cout << "char: " << c << " int: " << +c << '\n'
<< "-1, where 1 is signed: " << -n1 << '\n'
<< "-1, where 1 is unsigned char: " << -n2 << '\n'
<< "-1, where 1 is unsigned int: " << -n3 << '\n';
char a[3];
std::cout << "size of array: " << sizeof a << '\n'
<< "size of pointer: " << sizeof +a << '\n';
}
Output:
char: j int: 106
-1, where 1 is signed: -1
-1, where 1 is unsigned char: -1
-1, where 1 is unsigned int: 4294967295
size of array: 3
size of pointer: 8
Additive operators
For every pair of promoted arithmetic types L
and R
and for every object type T
, the following function signatures participate in overload resolution:
LR operator+(L, R) | | |
---|---|---|
LR operator-(L, R) | | |
T* operator+(T*, std::ptrdiff_t) | | |
T* operator+(std::ptrdiff_t, T*) | | |
T* operator-(T*, std::ptrdiff_t) | | |
std::ptrdiff_t operator-(T*, T*) | | |
where LR
is the result of usual arithmetic conversions on L
and R
.
With operands of arithmetic or enumeration type, the result of binary plus is the sum of the operands (after usual arithmetic converesions), and the result of the binary minus operator is the result of subtracting the second operand from the first (after usual arithmetic conversions), except that, if the type supports IEEE floating-point arithmetic (see std::numeric_limits::is_iec559
),
- if one operand is NaN, the result is NaN
If any of the operands is a pointer, the following rules apply:
- A pointer to non-array object is treated as a pointer to the first element of an array with size 1.
These pointer arithmetic operators allow pointers to satisfy the RandomAccessIterator
concept.
#include <iostream>
int main()
{
char c = 2;
unsigned int un = 2;
int n = -10;
std::cout << " 2 + (-10), where 2 is a char = " << c + n << '\n'
<< " 2 + (-10), where 2 is unsigned = " << un + n << '\n'
<< " -10 - 2.12 = " << n - 2.12 << '\n';
char a[4] = {'a', 'b', 'c', 'd'};
char* p = &a[1];
std::cout << "Pointer addition examples: " << *p << *(p + 2)
<< *(2 + p) << *(p - 1) << '\n';
char* p2 = &a[4];
std::cout << "Pointer difference: " << p2 - p << '\n';
}
Output:
2 + (-10), where 2 is a char = -8
2 + (-10), where 2 is unsigned = 4294967288
-10 - 2.12 = -12.12
Pointer addition examples: bdda
Pointer difference: 3
Multiplicative operators
For every pair of promoted arithmetic types LA
and RA
and for every pair of promoted integral types LI
and RI
the following function signatures participate in overload resolution:
LRA operator*(LA, RA) | | |
---|---|---|
LRA operator/(LA, RA) | | |
LRI operator%(LI, RI) | | |
where LRx
is the result of usual arithmetic conversions on Lx
and Rx
.
The binary operator * performs multiplication of its operands (after usual arithmetic conversions), except that, for floating-point multiplication,
- multiplication of a NaN by any number gives NaN
The binary operator / divides the first operand by the second (after usual arithmetic conversions).
For integral operands, it yields the algebraic quotient.
The quotient is rounded in implementation-defined direction. | (until C++11) |
---|---|
The quotient is truncated towards zero (fractional part is discarded). | (since C++11) |
If the second operand is zero, the behavior is undefined, except that if floating-point division is taking place and the type supports IEEE floating-point arithmetic (see std::numeric_limits::is_iec559
), then:
- if one operand is NaN, the result is NaN
The binary operator % yields the remainder of the integer division of the first operand by the second (after usual arithmetic conversions; note that the operand types must be integral types). If the quotient a/b
is representable in the result type, (a/b)*b + a%b == a
. If the second operand is zero, the behavior is undefined. If the quotient a/b
is not representable in the result type, the behavior of both a/b
and a%b
is undefined (that means INT_MIN%-1
is undefined on 2's complement systems).
Note: Until C++11, if one or both operands to binary operator % were negative, the sign of the remainder was implementation-defined, as it depends on the rounding direction of integer division. The function std::div
provided well-defined behavior in that case.
Note: for floating-point remainder, see std::remainder
and std::fmod
.
#include <iostream>
int main()
{
char c = 2;
unsigned int un = 2;
int n = -10;
std::cout << "2 * (-10), where 2 is a char = " << c * n << '\n'
<< "2 * (-10), where 2 is unsigned = " << un * n << '\n'
<< "-10 / 2.12 = " << n / 2.12 << '\n'
<< "-10 / 21 = " << n / 21 << '\n'
<< "-10 % 21 = " << n % 21 << '\n';
}
Output:
2 * (-10), where 2 is a char = -20
2 * (-10), where 2 is unsigned = 4294967276
-10 / 2.12 = -4.71698
-10 / 21 = 0
-10 % 21 = -10
Bitwise logic operators
For every pair of promoted integral types L
and R
the following function signatures participate in overload resolution:
R operator~(R) | | |
---|---|---|
LR operator&(L, R) | | |
LR operator^(L, R) | | |
LR operator|(L, R) | | |
where LR
is the result of usual arithmetic conversions on L
and R
.
The result of operator~ is the bitwise NOT (one's complement) value of the argument (after promotion). The result of operator& is the bitwise AND value of the operands (after usual arithmetic conversions). The result of operator| is the bitwise OR value of the operands (after usual arithmetic conversions). The result of operator^ is the bitwise XOR value of the operands (after usual arithmetic conversions).
#include <iostream>
int main()
{
std::cout << std::hex << std::showbase;
uint16_t mask = 0x00f0;
uint32_t a = 0x12345678;
std::cout << "Value: " << a << " mask: " << mask << '\n'
<< "Setting bits: " << (a | mask) << '\n'
<< "Clearing bits: " << (a & ~mask) << '\n'
<< "Selecting bits: " << (a & mask) << '\n';
}
Output:
Value: 0x12345678 mask: 0xf0
Setting bits: 0x123456f8
Clearing bits: 0x12345608
Selecting bits: 0x70
Bitwise shift operators
For every pair of promoted integral types L
and R
, the following function signatures participate in overload resolution:
L operator<<(L, R) | | |
---|---|---|
L operator>>(L, R) | | |
The operands of the built-in bitwise shift operators have either integral types or unscoped enumeration type. Integral promotions are performed on both operands before evaluation. The return type is the type of the left operand after integral promotions.
For unsigned and positive a, the value of a << b is the value of a * 2b
, reduced modulo maximum value of the return type plus 1 (that is, bitwise left shift is performed and the bits that get shifted out of the destination type are discarded).
For signed a
nd positive a
,
the value of a << b is a * 2b if it is representable the return type, otherwise the behavior is undefined. | (until C++14) |
---|---|
the value of a << b is a * 2b if it is representable in the unsigned version of the return type (which is then converted to signed: this makes it legal to create INT_MIN as 1<<31), otherwise the behavior is undefined. | (since C++14) |
For negative a, the behavior of a << b is undefined.
For unsigned a and for signed a with nonnegative values, the value of a >> b is the integer part of a/2b
. For negative a, the value of a >> b is implementation-defined (in most implementations, this performs arithmetic right shift, so that the result remains negative).
In any case, if the value of the right operand is negative or is greater or equal to the number of bits in the promoted left operand, the behavior is undefined.
#include <iostream>
enum {ONE=1, TWO=2};
int main()
{
std::cout << std::hex << std::showbase;
char c = 0x10;
unsigned long long ull = 0x123;
std::cout << "0x123 << 1 = " << (ull << 1) << '\n'
<< "0x123 << 63 = " << (ull << 63) << '\n' // overflow in unsigned
<< "0x10 << 10 = " << (c << 10) << '\n'; // char is promoted to int
long long ll = -1000;
std::cout << std::dec << "-1000 >> 1 = " << (ll >> ONE) << '\n';
}
Output:
0x123 << 1 = 0x246
0x123 << 63 = 0x8000000000000000
0x10 << 10 = 0x4000
-1000 >> 1 = -500
Standard library
Arithmetic operators are overloaded for many standard library types.
Unary arithmetic operators
operator+operator- | implements unary + and unary - (public member function of std::chrono::duration) |
---|---|
operator+operator- | applies unary operators to complex numbers (function template) |
operator+operator-operator~operator! | applies a unary arithmetic operator to each element of the valarray (public member function of std::valarray) |
Additive operators
operator+operator- | modifies the time point by the given duration (function template) |
---|---|
operator+ | concatenates two strings or a string and a char (function template) |
operator+operator- | advances or decrements the iterator (public member function of std::reverse_iterator) |
operator+operator- | advances or decrements the iterator (public member function of std::move_iterator) |
operator+operator-operator*operator/ | performs complex number arithmetics on two complex values or a complex and a scalar (function template) |
operator+operator-operator*operator/operator%operator&operator|operator^operator<<operator>>operator&&operator|| | applies binary operators to each element of two valarrays, or a valarray and a value (function template) |
Multiplicative operators
operator+operator-operator*operator/operator% | implements arithmetic operations with durations as arguments (function template) |
---|---|
operator+operator-operator*operator/ | performs complex number arithmetics on two complex values or a complex and a scalar (function template) |
operator+operator-operator*operator/operator%operator&operator|operator^operator<<operator>>operator&&operator|| | applies binary operators to each element of two valarrays, or a valarray and a value (function template) |
Bitwise logic operators
operator&=operator|=operator^=operator~ | performs binary AND, OR, XOR and NOT (public member function of std::bitset) |
---|---|
operator&operator|operator^ | performs binary logic operations on bitsets (function template) |
operator~ | applies a unary arithmetic operator to each element of the valarray (public member function of std::valarray) |
operator^operator&operator| | applies binary operators to each element of two valarrays, or a valarray and a value (function template) |
Bitwise shift operators
operator< | applies binary operators to each element of two valarrays, or a valarray and a value (function template) |
---|---|
operator<<operator>> | performs binary shift left and shift right (public member function of std::bitset) |
Stream insertion/extraction operators
Throughout the standard library, bitwise shift operators are commonly overloaded with I/O stream (std::ios_base
&
or one of the classes derived from it) as both the left operand and return type. Such operators are known as stream insertion
and stream extraction
operators:
operator>> | extracts formatted data (public member function of std::basic_istream) |
---|---|
operator>>(std::basic_istream) | extracts characters and character arrays (function template) |
operator<< | inserts formatted data (public member function of std::basic_ostream) |
operator<<(std::basic_ostream) | inserts character data (function template) |
operator<<operator>> | serializes and deserializes a complex number (function template) |
operator<<operator>> | performs stream input and output of bitsets (function template) |
operator<<operator>> | performs stream input and output on strings (function template) |
operator<<operator>> | performs stream input and output on pseudo-random number engine (function template) |
operator<<operator>> | performs stream input and output on pseudo-random number distribution (function template) |
See also
Operator precedence.
Operator overloading.
| Common operators |
|:----|
| assignment | incrementdecrement | arithmetic | logical | comparison | memberaccess | other |
| a = b a += b a -= b a *= b a /= b a %= b a &= b a |= b a ^= b a <<= b a >>= b. | ++a --a a++ a-- | +a -a a + b a - b a * b a / b a % b ~a a & b a | b a ^ b a << b a >> b. | !a a && b a || b. | a == b a != b a < b a > b a <= b a >= b. | ab *a &a a->b a.b a->*b a.*b. | a(...) a, b ? : |
| Special operators |
| static_cast converts one type to another related type dynamic_cast converts within inheritance hierarchies const_cast adds or removes cv qualifiers reinterpret_cast converts type to unrelated type C-style cast converts one type to another by a mix of static_cast, const_cast, and reinterpret_cast new creates objects with dynamic storage duration delete destructs objects previously created by the new expression and releases obtained memory area sizeof queries the size of a type sizeof... queries the size of a parameter pack (since C++11) typeid queries the type information of a type noexcept checks if an expression can throw an exception (since C++11) alignof queries alignment requirements of a type (since C++11). |
© cppreference.com
Licensed under the Creative Commons Attribution-ShareAlike Unported License v3.0.
http://en.cppreference.com/w/cpp/language/operator_arithmetic