-
Notifications
You must be signed in to change notification settings - Fork 160
/
namespace.cpp
307 lines (238 loc) · 6 KB
/
namespace.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
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
/*
# namespace
*/
#include "common.hpp"
// ERROR: same naming rules as vars
//namespace 2D{}
// BAD: by convention, namespaces start with lower case
namespace D2{}
int i;
void f() {}
void prototype();
namespace namea {
int in_namea_only = 0;
class C {
public:
C() {}
};
namespace nameaa {
int i;
void f() {}
class C {
public:
C() {}
};
}
namespace nameab {
int i;
void f() {
::i = 0;
// namea::nameab::i
i = 0;
// namea::nameaa::i
nameaa::i = 0;
// Only affects this function. see C::C() at the f() call.
using namespace nameaa;
}
class C :
public namea::C
//public C
// ERROR: refers to current incomplete C already
//not existem namea::C
{
C() {
// No ambiguity because using inside f() only afects the function.
f();
}
};
}
int i;
void f() {
::i = 0;
i = 0; //namea::i
namea::i = 0;
nameaa::i = 0; //namea::nameaa::i
}
}
namespace namea {
// Can add new members
int j;
void newFunc(){}
class B{};
// ERROR: redefinition
//int i;
// Implementation of namea::prototype.
//void prototype(){}
}
// ERROR: must be declared/defined inside.
//namea::i = 0;
//void namea::prototype(){}
//int namea::j;
//void namea::g(){}
//class namea::B{};
// ERROR.
//template<class T> namespace t {}
// ADL
namespace adl0 {
struct s {};
int adl(struct s s){
return 0;
}
int i;
int adlNoType(int i){
return 0;
}
int adlMultiArg(int i, struct s s, int j){
return 0;
}
}
namespace adl1 {
struct s {};
int adl(struct s s){
return 1;
}
int i;
int adlNoType(int i){
return 1;
}
int adl0FromAdl1(struct adl0::s s) {
return 1;
}
int adl0And1FromAdl1(struct adl0::s s0, struct s s1) {
return 1;
}
float adl01(struct adl0::s s, struct s s1){
return 0.5;
}
}
namespace adl0 {
float adl01(struct s s, struct adl1::s s1){
return 0.5;
}
}
int main() {
// variables
{
int i;
i = 0; //inner i
::i = 0; //global i
namea::i = 0; //namea i
i++;
assert(i == 1);
assert(::i == 0);
assert(namea::i == 0);
f();
namea::f();
namea::nameaa::f();
}
/*
# using
Be very careful with `using`, because there is no way to unuse afterwards.
In particuar, *never* use `using namespace X` on the toplevel a header file,
or you shall confuse includers to tears.
*/
{
using namespace namea;
// ERROR ambiguous
//f();
//::f
//namea::f
::f();
namea::f();
namea::nameaa::f();
}
// Brackets limit the using namespace scope.
// It is obligatory to specify unused namespaces.
//in_namea_only = 1;
// ERROR: no namespace inside funcs
//namespace main{}
// Namespace chaining
{
using namespace namea;
using namespace nameaa;
// ERROR ambiguous
// ::f
// namea::f
// namea::nameaa:f
//f();
::f();
namea::f();
namea::nameaa::f();
}
// Namespace alias
namespace newNamea = namea;
{
using namespace newNamea;
// ERROR: ambiguous.
// ::f
// namea::f
//f();
}
// Subimport
{
// Imports only name::f.
using namea::f;
f();
// OK: overwrides global f().
//namea::f
// ERROR: only f was imported.
//C c;
};
/*
# ADL
Argument dependent name lookup.
<http://en.wikipedia.org/wiki/Argument-dependent_name_lookup>
Allows for functions without namespace qualification:
- to be found
- to haves ambiguities resolved
based on the namespace in which the types of their arguments are defined.
Explains why operator `<<` does not need the `std::` qualifier,
even though *must* be implemented as a non-member function!!
(see info on operator overload for why)
ADL for operators is a major use case, because specifying namespaces
for operators completely destroys their eyecandy appeal.
*/
{
// ADL allows both to be found and differentiated!
{
{
struct adl0::s s;
assert(adl(s) == 0);
}
{
struct adl1::s s;
assert(adl(s) == 1);
}
}
// Only works if the type is defined on the same namespace as the function.
{
struct adl0::s s;
// ERROR: not declared on this scope
//assert(adl0FromAdl1(s) == 1);
}
// Works if at least one of the argument types is in the namespace.
{
struct adl0::s s;
assert(adlMultiArg(0, s, 1) == 0);
}
// Lookup works even if types from both namespaces are used.
{
struct adl0::s s0;
struct adl1::s s1;
assert(adl0And1FromAdl1(s0, s1) == 1);
}
// Of course, calls can still be ambiguous.
{
struct adl0::s s0;
struct adl1::s s1;
// ERROR: ambiguous call
//assert(adl01(s0, s1) == 0.5);
}
// Only works for *types* defined in the namespaces, not values.
{
//assert(adlNoType(adl0::i) == 0);
// ERROR: adlNoType not found on this scope
//assert(adlNoType(adl1::i) == 0);
}
}
}