-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathCC.cpp
166 lines (146 loc) · 4.34 KB
/
CC.cpp
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
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
/**
* Validate full-fledge constructor and assignment operator.
* Interesting read: https://stackoverflow.com/a/3279550/571227
*
* Compile with and without -fno-elide-constructors with gcc to see different results to compare.
*
* With RVO copy move constructor will not be necessary as it creates a copy of object at the returning
* of the function, or from somewhere on rhs about to return and assign to lhs, it uses only a single
* copy then assign to the receiving variable at the call site. So don't be confused that RVO
* will need move constructor or move assignment operator to be implemented for both. It's feature
* provided by compiler. Move constructor/assignment operator will kick in and be used whenever
* RVO isn't possible (https://stackoverflow.com/questions/5031778/what-optimization-does-move-semantics-provide-if-we-already-have-rvo#comment5628760_5031859).
*
*
* Move semantics should not be thought as an optimization device - https://stackoverflow.com/a/5031878/571227
*
*
* If you implemented both non-const assignment operator, and move assignment operator then it will
* be ambiguous, see below.
*
* CC.cpp: In function ‘int main()’:
* CC.cpp:88:16: error: ambiguous overload for ‘operator=’ (operand types are ‘Widget’ and ‘Widget’)
* 88 | a = Widget(); // here
* | ^
* CC.cpp:38:13: note: candidate: ‘Widget& Widget::operator=(Widget)’
* 38 | Widget& operator=(Widget other)
* | ^~~~~~~~
* CC.cpp:46:13: note: candidate: ‘Widget& Widget::operator=(Widget&&)’
* 46 | Widget& operator=(Widget&& other)
*
*
* Interesting read about RVO vs std::move: https://www.ibm.com/developerworks/community/blogs/5894415f-be62-4bc0-81c5-3956e82276f3/entry/RVO_V_S_std_move?lang=en
*/
#include <iostream>
#include <algorithm>
struct Widget
{
Widget()
{
std::cout << "ctor" << std::endl;
}
Widget(const Widget& other)
{
std::cout << "ctor - const ref" << std::endl;
}
Widget(Widget&& other)
{
std::cout << "ctor - move" << std::endl;
using std::swap;
swap(*this, other);
}
//Widget& operator=(Widget other)
//{
// std::cout << "assignment operator" << std::endl;
// using std::swap;
// swap(*this, other);
// return *this;
//}
Widget& operator=(const Widget& other)
{
std::cout << "(const) assignment operator" << std::endl;
return *this;
}
Widget& operator=(Widget&& other)
{
std::cout << "assignment move operator" << std::endl;
using std::swap;
swap(*this, other);
return *this;
}
~Widget()
{
std::cout << "dtor" << std::endl;
}
friend inline void swap(Widget& first, Widget& second)
{
std::cout << "swap" << std::endl;
}
};
Widget foo()
{
return Widget();
}
Widget&& foo2()
{
return Widget();
}
Widget&& foo3()
{
return std::move(Widget());
}
int main()
{
// the following 1: and 2: are cases which can happen
// in case of 1: RVO, and 2: non-RVO
// 1: not apply
// 2: ctor, dtor
{
std::cout << "1" << std::endl;
Widget a;
}
// 1: ctor, dtor
// 2: ctor, move ctor, swap, dtor, dtor
{
std::cout << "2" << std::endl;
Widget a = Widget();
}
// wrong intention, not construct object
// but declare function named 'a' returning Widget
{
std::cout << "3" << std::endl;
Widget a();
}
// same as 1
// another form of initialization called 'list initialization'
{
std::cout << "4" << std::endl;
Widget a { };
}
// 1: ctor, ctor, dtor, dtor
// 2: ctor, ctor, move assignment, swap, dtor, dtor
{
std::cout << "5" << std::endl;
Widget a;
a = Widget(); // here
}
// 1: ctor, dtor
// 2: ctor, move ctor, swap, dtor, move ctor, swap, dtor, dtor
{
std::cout << "6" << std::endl;
Widget c = foo();
}
// 1: ctor, dtor, ctor - move, swap, dtor
// 2: ctor, dtor, ctor - move, swap, dtor
{
std::cout << "7" << std::endl;
Widget c = foo2();
}
// 1: ctor, dtor, ctor - move, swap, dtor
// 2: ctor, dtor, ctor - move, swap, dtor
{
std::cout << "8" << std::endl;
Widget c = foo3();
}
return 0;
}