Type Conversions: Implicit Conversion

Download as docx, pdf, or txt
Download as docx, pdf, or txt
You are on page 1of 5

Type conversions

Implicit conversion
Implicit conversions are automatically performed when a value is copied to a compatible type.
For example:
1
2
3

short a=2000;
int b;
b=a;

Here, the value of a is promoted from short to int without the need of any explicit operator.
This is known as a standard conversion. Standard conversions affect fundamental data types, and
allow the conversions between numerical types (short to int, int to float, double to int...),
to or from bool, and some pointer conversions.
Converting to int from some smaller integer type, or to double from float is known as
promotion, and is guaranteed to produce the exact same value in the destination type. Other
conversions between arithmetic types may not always be able to represent the same value
exactly:

If a negative integer value is converted to an unsigned type, the resulting value


corresponds to its 2's complement bitwise representation (i.e., -1 becomes the largest
value representable by the type, -2 the second largest, ...).

The conversions from/to bool consider false equivalent to zero (for numeric types) and
to null pointer (for pointer types); true is equivalent to all other values and is converted
to the equivalent of 1.

If the conversion is from a floating-point type to an integer type, the value is truncated
(the decimal part is removed). If the result lies outside the range of representable values
by the type, the conversion causes undefined behavior.

Otherwise, if the conversion is between numeric types of the same kind (integer-tointeger or floating-to-floating), the conversion is valid, but the value is implementationspecific (and may not be portable).

Some of these conversions may imply a loss of precision, which the compiler can signal with a
warning. This warning can be avoided with an explicit conversion.
For non-fundamental types, arrays and functions implicitly convert to pointers, and pointers in
general allow the following conversions:

Null pointers can be converted to pointers of any type

Pointers to any type can be converted to void pointers.

Pointer upcast: pointers to a derived class can be converted to a pointer of an accessible


and unambiguous base class, without modifying its const or volatile qualification.

Implicit conversions with classes


In the world of classes, implicit conversions can be controlled by means of three member
functions:

Single-argument constructors: allow implicit conversion from a particular type to


initialize an object.

Assignment operator: allow implicit conversion from a particular type on assignments.

Type-cast operator: allow implicit conversion to a particular type.

For example:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24

// implicit conversion of classes:


#include <iostream>
using namespace std;
class A {};
class B {
public:
// conversion from A (constructor):
B (const A& x) {}
// conversion from A (assignment):
B& operator= (const A& x) {return *this;}
// conversion to A (type-cast operator)
operator A() {return A();}
};
int main ()
{
A foo;
B bar = foo;
bar = foo;
foo = bar;
return 0;
}

// calls constructor
// calls assignment
// calls type-cast operator

Edit & Run

The type-cast operator uses a particular syntax: it uses the operator keyword followed by the
destination type and an empty set of parentheses. Notice that the return type is the destination
type and thus is not specified before the operator keyword.

Keyword explicit
On a function call, C++ allows one implicit conversion to happen for each argument. This may
be somewhat problematic for classes, because it is not always what is intended. For example, if
we add the following function to the last example:
void fn (B arg) {}

This function takes an argument of type B, but it could as well be called with an object of type A
as argument:
fn (foo);

This may or may not be what was intended. But, in any case, it can be prevented by marking the
affected constructor with the explicit keyword:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25

// explicit:
#include <iostream>
using namespace std;
class A {};
class B {
public:
explicit B (const A& x) {}
B& operator= (const A& x) {return *this;}
operator A() {return A();}
};
void fn (B x) {}
int main ()
{
A foo;
B bar (foo);
bar = foo;
foo = bar;
//

fn (foo);
fn (bar);

// not allowed for explicit ctor.

Edit & Run

26
return 0;
27 }

Additionally, constructors marked with explicit cannot be called with the assignment-like
syntax; In the above example, bar could not have been constructed with:
B bar = foo;

Type-cast member functions (those described in the previous section) can also be specified as
explicit. This prevents implicit conversions in the same way as explicit-specified
constructors do for the destination type.

Type casting
C++ is a strong-typed language. Many conversions, specially those that imply a different
interpretation of the value, require an explicit conversion, known in C++ as type-casting. There
exist two main syntaxes for generic type-casting: functional and c-like:
1
2
3
4

double x = 10.3;
int y;
y = int (x);
// functional notation
y = (int) x;
// c-like cast notation

The functionality of these generic forms of type-casting is enough for most needs with
fundamental data types. However, these operators can be applied indiscriminately on classes and
pointers to classes, which can lead to code that -while being syntactically correct- can cause
runtime errors. For example, the following code compiles without errors:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17

// class type-casting
#include <iostream>
using namespace std;
class Dummy {
double i,j;
};
class Addition {
int x,y;
public:
Addition (int a, int b) { x=a; y=b; }
int result() { return x+y;}
};
int main () {
Dummy d;

Edit & Run

18
19
20
21
22

Addition * padd;
padd = (Addition*) &d;
cout << padd->result();
return 0;

The program declares a pointer to Addition, but then it assigns to it a reference to an object of
another unrelated type using explicit type-casting:
padd = (Addition*) &d;

Unrestricted explicit type-casting allows to convert any pointer into any other pointer type,
independently of the types they point to. The subsequent call to member result will produce
either a run-time error or some other unexpected results.
In order to control these types of conversions between classes, we have four specific casting
operators: dynamic_cast, reinterpret_cast, static_cast and const_cast. Their format is
to follow the new type enclosed between angle-brackets (<>) and immediately after, the
expression to be converted between parentheses.
dynamic_cast <new_type> (expression)
reinterpret_cast <new_type> (expression)
static_cast <new_type> (expression)
const_cast <new_type> (expression)

The traditional type-casting equivalents to these expressions would be:


(new_type) expression
new_type (expression)

but each one with its own special characteristics:

You might also like