QuickCppLib 0.10
Eliminate all the tedious hassle when making state-of-the-art C++ 14 - 23 libraries!
Loading...
Searching...
No Matches
test_optional.cpp File Reference
#include "optional.hpp"
#include <vector>
#include <iostream>
#include <functional>
#include <complex>
#include <unordered_set>
#include <string>

Classes

struct  caller
 
struct  OracleVal
 
struct  Oracle
 
struct  Guard
 
struct  ExplicitStr
 
struct  Date
 
struct  MoveAware< T >
 
struct  BadRelops
 
struct  generic< T >
 
struct  generic< U & >
 
struct  Combined
 
struct  Nasty
 
struct  CountedObject
 
struct  NothrowBoth
 
struct  NothrowCtor
 
struct  NothrowAssign
 
struct  NothrowNone
 
struct  VEC
 

Namespaces

namespace  constexpr_optional_ref_and_arrow
 

Macros

#define CAT2(X, Y)   X ## Y
 
#define CAT(X, Y)   CAT2(X, Y)
 
#define TEST(NAME)   caller CAT(__VAR, __LINE__) = []
 

Typedefs

template<class T >
using Generic = typename generic< T >::type
 

Enumerations

enum  State {
  sDefaultConstructed , sValueCopyConstructed , sValueMoveConstructed , sCopyConstructed ,
  sMoveConstructed , sMoveAssigned , sCopyAssigned , sValueCopyAssigned ,
  sValueMoveAssigned , sMovedFrom , sValueConstructed
}
 

Functions

bool operator== (Oracle const &a, Oracle const &b)
 
bool operator!= (Oracle const &a, Oracle const &b)
 
 TEST (disengaged_ctor)
 
 TEST (value_ctor)
 
 TEST (assignment)
 
 TEST (moved_from_state)
 
 TEST (copy_move_ctor_optional_int)
 
 TEST (optional_optional)
 
 TEST (example_guard)
 
void process ()
 
void process (int)
 
void processNil ()
 
 TEST (example1)
 
 TEST (example_ref)
 
template<typename T >
getValue (tr2::optional< T > newVal=tr2::nullopt, tr2::optional< T & > storeHere=tr2::nullopt)
 
 TEST (example_optional_arg)
 
std::tuple< Date, Date, DategetStartMidEnd ()
 
void run (Date const &, Date const &, Date const &)
 
 TEST (example_date)
 
std::experimental::optional< char > readNextChar ()
 
void run (std::experimental::optional< std::string >)
 
void run (std::complex< double >)
 
template<class T >
void assign_norebind (tr2::optional< T & > &optref, T &obj)
 
template<typename T >
void unused (T &&)
 
 TEST (example_conceptual_model)
 
 TEST (example_rationale)
 
bool fun (std::string, std::experimental::optional< int > oi=std::experimental::nullopt)
 
 TEST (example_converting_ctor)
 
 TEST (bad_comparison)
 
 TEST (value_or)
 
 TEST (reset)
 
 TEST (mixed_order)
 
constexpr bool operator< (BadRelops a, BadRelops b)
 
constexpr bool operator> (BadRelops a, BadRelops b)
 
 TEST (bad_relops)
 
 TEST (mixed_equality)
 
 TEST (const_propagation)
 
 TEST (safe_value)
 
 TEST (optional_ref)
 
 TEST (optional_ref_const_propagation)
 
 TEST (optional_ref_assign)
 
 TEST (optional_swap)
 
 TEST (optional_ref_swap)
 
 TEST (optional_initialization)
 
 TEST (optional_hashing)
 
template<class X >
bool generic_fun ()
 
 TEST (optional_ref_emulation)
 
 TEST (optional_ref_hashing)
 
 TEST (arrow_operator)
 
 TEST (arrow_wit_optional_ref)
 
 TEST (no_dangling_reference_in_value)
 
 TEST (exception_safety)
 
 TEST (nested_optional)
 
 TEST (three_ways_of_having_value)
 
void test_noexcept ()
 
void constexpr_test_disengaged ()
 
int main ()
 

Variables

constexpr tr2::optional< int > g0 {}
 
constexpr tr2::optional< int > g2 {2}
 
constexpr tr2::optional< Combinedgc0 {tr2::in_place}
 
int gi = 0
 
constexpr tr2::optional< int & > gori = gi
 
constexpr tr2::optional< int & > gorn {}
 
constexpr int & gri = *gori
 
constexpr int gci = 1
 
constexpr tr2::optional< int const & > gorci = gci
 
constexpr tr2::optional< int const & > gorcn {}
 
constexpr Combined constexpr_optional_ref_and_arrow::c {1, 2}
 
constexpr optional< Combined const & > constexpr_optional_ref_and_arrow::oc = c
 

Macro Definition Documentation

◆ CAT2

#define CAT2 (   X,
 
)    X ## Y

◆ CAT

#define CAT (   X,
 
)    CAT2(X, Y)

◆ TEST

#define TEST (   NAME)    caller CAT(__VAR, __LINE__) = []

Typedef Documentation

◆ Generic

template<class T >
using Generic = typename generic<T>::type

Enumeration Type Documentation

◆ State

enum State
Enumerator
sDefaultConstructed 
sValueCopyConstructed 
sValueMoveConstructed 
sCopyConstructed 
sMoveConstructed 
sMoveAssigned 
sCopyAssigned 
sValueCopyAssigned 
sValueMoveAssigned 
sMovedFrom 
sValueConstructed 
26{
38};
@ sDefaultConstructed
Definition test_optional.cpp:27
@ sValueMoveAssigned
Definition test_optional.cpp:35
@ sMoveConstructed
Definition test_optional.cpp:31
@ sCopyAssigned
Definition test_optional.cpp:33
@ sValueConstructed
Definition test_optional.cpp:37
@ sValueMoveConstructed
Definition test_optional.cpp:29
@ sMoveAssigned
Definition test_optional.cpp:32
@ sValueCopyAssigned
Definition test_optional.cpp:34
@ sCopyConstructed
Definition test_optional.cpp:30
@ sValueCopyConstructed
Definition test_optional.cpp:28
@ sMovedFrom
Definition test_optional.cpp:36

Function Documentation

◆ operator==()

bool operator== ( Oracle const &  a,
Oracle const &  b 
)
92{ return a.val.i == b.val.i; }

◆ operator!=()

bool operator!= ( Oracle const &  a,
Oracle const &  b 
)
93{ return a.val.i != b.val.i; }

◆ TEST() [1/37]

TEST ( disengaged_ctor  )
100{
102 assert (!o1);
103
105 assert (!o2);
106
107 tr2::optional<int> o3 = o2;
108 assert (!o3);
109
110 assert (o1 == tr2::nullopt);
111 assert (o1 == tr2::optional<int>{});
112 assert (!o1);
113 assert (bool(o1) == false);
114
115 assert (o2 == tr2::nullopt);
116 assert (o2 == tr2::optional<int>{});
117 assert (!o2);
118 assert (bool(o2) == false);
119
120 assert (o3 == tr2::nullopt);
121 assert (o3 == tr2::optional<int>{});
122 assert (!o3);
123 assert (bool(o3) == false);
124
125 assert (o1 == o2);
126 assert (o2 == o1);
127 assert (o1 == o3);
128 assert (o3 == o1);
129 assert (o2 == o3);
130 assert (o3 == o2);
131};
Definition optional.hpp:365
constexpr nullopt_t nullopt
Definition optional.hpp:268

◆ TEST() [2/37]

TEST ( value_ctor  )
135{
136 OracleVal v;
138 assert (oo1 != tr2::nullopt);
139 assert (oo1 != tr2::optional<Oracle>{});
140 assert (oo1 == tr2::optional<Oracle>{v});
141 assert (!!oo1);
142 assert (bool(oo1));
143 // NA: assert (oo1->s == sValueCopyConstructed);
144 assert (oo1->s == sMoveConstructed);
145 assert (v.s == sValueConstructed);
146
147 tr2::optional<Oracle> oo2(std::move(v));
148 assert (oo2 != tr2::nullopt);
149 assert (oo2 != tr2::optional<Oracle>{});
150 assert (oo2 == oo1);
151 assert (!!oo2);
152 assert (bool(oo2));
153 // NA: assert (oo2->s == sValueMoveConstructed);
154 assert (oo2->s == sMoveConstructed);
155 assert (v.s == sMovedFrom);
156
157 {
158 OracleVal v;
160 assert (oo1 != tr2::nullopt);
161 assert (oo1 != tr2::optional<Oracle>{});
162 assert (oo1 == tr2::optional<Oracle>{v});
163 assert (!!oo1);
164 assert (bool(oo1));
165 assert (oo1->s == sValueCopyConstructed);
166 assert (v.s == sValueConstructed);
167
168 tr2::optional<Oracle> oo2{tr2::in_place, std::move(v)};
169 assert (oo2 != tr2::nullopt);
170 assert (oo2 != tr2::optional<Oracle>{});
171 assert (oo2 == oo1);
172 assert (!!oo2);
173 assert (bool(oo2));
174 assert (oo2->s == sValueMoveConstructed);
175 assert (v.s == sMovedFrom);
176 }
177};
constexpr struct std::experimental::in_place_t in_place
Definition test_optional.cpp:41
State s
Definition test_optional.cpp:42

◆ TEST() [3/37]

TEST ( assignment  )
181{
183 oi = tr2::optional<int>{1};
184 assert (*oi == 1);
185
186 oi = tr2::nullopt;
187 assert (!oi);
188
189 oi = 2;
190 assert (*oi == 2);
191
192 oi = {};
193 assert (!oi);
194};

◆ TEST() [4/37]

TEST ( moved_from_state  )
217{
218 // first, test mock:
219 MoveAware<int> i{1}, j{2};
220 assert (i.val == 1);
221 assert (!i.moved);
222 assert (j.val == 2);
223 assert (!j.moved);
224
225 MoveAware<int> k = std::move(i);
226 assert (k.val == 1);
227 assert (!k.moved);
228 assert (i.val == 1);
229 assert (i.moved);
230
231 k = std::move(j);
232 assert (k.val == 2);
233 assert (!k.moved);
234 assert (j.val == 2);
235 assert (j.moved);
236
237 // now, test optional
238 tr2::optional<MoveAware<int>> oi{1}, oj{2};
239 assert (oi);
240 assert (!oi->moved);
241 assert (oj);
242 assert (!oj->moved);
243
244 tr2::optional<MoveAware<int>> ok = std::move(oi);
245 assert (ok);
246 assert (!ok->moved);
247 assert (oi);
248 assert (oi->moved);
249
250 ok = std::move(oj);
251 assert (ok);
252 assert (!ok->moved);
253 assert (oj);
254 assert (oj->moved);
255};
Definition test_optional.cpp:199
bool moved
Definition test_optional.cpp:201
T val
Definition test_optional.cpp:200

◆ TEST() [5/37]

TEST ( copy_move_ctor_optional_int  )
259{
261 tr2::optional<int> oj = oi;
262
263 assert (!oj);
264 assert (oj == oi);
265 assert (oj == tr2::nullopt);
266 assert (!bool(oj));
267
268 oi = 1;
269 tr2::optional<int> ok = oi;
270 assert (!!ok);
271 assert (bool(ok));
272 assert (ok == oi);
273 assert (ok != oj);
274 assert (*ok == 1);
275
276 tr2::optional<int> ol = std::move(oi);
277 assert (!!ol);
278 assert (bool(ol));
279 assert (ol == oi);
280 assert (ol != oj);
281 assert (*ol == 1);
282};

◆ TEST() [6/37]

TEST ( optional_optional  )
286{
288 assert (oi1 == tr2::nullopt);
289 assert (!oi1);
290
291 {
293 assert (oi2 != tr2::nullopt);
294 assert (bool(oi2));
295 assert (*oi2 == tr2::nullopt);
296 //assert (!(*oi2));
297 //std::cout << typeid(**oi2).name() << std::endl;
298 }
299
300 {
302 assert (oi2 != tr2::nullopt);
303 assert (bool(oi2));
304 assert (*oi2 == tr2::nullopt);
305 assert (!*oi2);
306 }
307
308 {
310 assert (oi2 != tr2::nullopt);
311 assert (bool(oi2));
312 assert (*oi2 == tr2::nullopt);
313 assert (!*oi2);
314 }
315
317 auto ooi = tr2::make_optional(oi);
318 static_assert( std::is_same<tr2::optional<tr2::optional<int>>, decltype(ooi)>::value, "");
319
320};
constexpr optional< typename decay< T >::type > make_optional(T &&v)
Definition optional.hpp:1023

◆ TEST() [7/37]

TEST ( example_guard  )
323{
324 using namespace tr2;
325 //FAILS: optional<Guard> ogx(Guard("res1"));
326 //FAILS: optional<Guard> ogx = "res1";
327 //FAILS: optional<Guard> ogx("res1");
328 optional<Guard> oga; // Guard is non-copyable (and non-moveable)
329 optional<Guard> ogb(in_place, "res1"); // initialzes the contained value with "res1"
330 assert (bool(ogb));
331 assert (ogb->val == "res1");
332
333 optional<Guard> ogc(in_place); // default-constructs the contained value
334 assert (bool(ogc));
335 assert (ogc->val == "");
336
337 oga.emplace("res1"); // initialzes the contained value with "res1"
338 assert (bool(oga));
339 assert (oga->val == "res1");
340
341 oga.emplace(); // destroys the contained value and
342 // default-constructs the new one
343 assert (bool(oga));
344 assert (oga->val == "");
345
346 oga = nullopt; // OK: make disengaged the optional Guard
347 assert (!(oga));
348 //FAILS: ogb = {}; // ERROR: Guard is not Moveable
349};
Definition optional.hpp:102

◆ process() [1/2]

void process ( )
352{}

◆ process() [2/2]

void process ( int  )
353{}

◆ processNil()

void processNil ( )
354{}

◆ TEST() [8/37]

TEST ( example1  )
358{
359 using namespace tr2;
360 optional<int> oi; // create disengaged object
361 optional<int> oj = nullopt; // alternative syntax
362 oi = oj; // assign disengaged object
363 optional<int> ok = oj; // ok is disengaged
364
365 if (oi) assert(false); // 'if oi is engaged...'
366 if (!oi) assert(true); // 'if oi is disengaged...'
367
368 if (oi != nullopt) assert(false); // 'if oi is engaged...'
369 if (oi == nullopt) assert(true); // 'if oi is disengaged...'
370
371 assert(oi == ok); // two disengaged optionals compare equal
372
373 ///////////////////////////////////////////////////////////////////////////
374 optional<int> ol{1}; // ol is engaged; its contained value is 1
375 ok = 2; // ok becomes engaged; its contained value is 2
376 oj = ol; // oj becomes engaged; its contained value is 1
377
378 assert(oi != ol); // disengaged != engaged
379 assert(ok != ol); // different contained values
380 assert(oj == ol); // same contained value
381 assert(oi < ol); // disengaged < engaged
382 assert(ol < ok); // less by contained value
383
384 /////////////////////////////////////////////////////////////////////////////
385 optional<int> om{1}; // om is engaged; its contained value is 1
386 optional<int> on = om; // on is engaged; its contained value is 1
387 om = 2; // om is engaged; its contained value is 2
388 assert (on != om); // on still contains 3. They are not pointers
389
390 /////////////////////////////////////////////////////////////////////////////
391 int i = *ol; // i obtains the value contained in ol
392 assert (i == 1);
393 *ol = 9; // the object contained in ol becomes 9
394 assert(*ol == 9);
395 assert(ol == make_optional(9));
396
397 ///////////////////////////////////
398 int p = 1;
399 optional<int> op = p;
400 assert(*op == 1);
401 p = 2;
402 assert(*op == 1); // value contained in op is separated from p
403
404 ////////////////////////////////
405 if (ol)
406 process(*ol); // use contained value if present
407 else
408 process(); // proceed without contained value
409
410 if (!om)
411 processNil();
412 else
413 process(*om);
414
415 /////////////////////////////////////////
416 process(ol.value_or(0)); // use 0 if ol is disengaged
417
418 ////////////////////////////////////////////
419 ok = nullopt; // if ok was engaged calls T's dtor
420 oj = {}; // assigns a temporary disengaged optional
421};
void processNil()
Definition test_optional.cpp:354
void process()
Definition test_optional.cpp:352

◆ TEST() [9/37]

TEST ( example_ref  )
435{
436 using namespace std::experimental;
437 int i = 1;
438 int j = 2;
439 optional<int&> ora; // disengaged optional reference to int
440 optional<int&> orb = i; // contained reference refers to object i
441
442 *orb = 3; // i becomes 3
443 // FAILS: ora = j; // ERROR: optional refs do not have assignment from T
444 // FAILS: ora = {j}; // ERROR: optional refs do not have copy/move assignment
445 // FAILS: ora = orb; // ERROR: no copy/move assignment
446 ora.emplace(j); // OK: contained reference refers to object j
447 ora.emplace(i); // OK: contained reference now refers to object i
448
449 ora = nullopt; // OK: ora becomes disengaged
450};
void emplace(Args &&... args)
Definition optional.hpp:488

◆ getValue()

template<typename T >
T getValue ( tr2::optional< T >  newVal = tr2::nullopt,
tr2::optional< T & >  storeHere = tr2::nullopt 
)
455{
456 T cached{};
457
458 if (newVal) {
459 cached = *newVal;
460
461 if (storeHere) {
462 *storeHere = *newVal; // LEGAL: assigning T to T
463 }
464 }
465 return cached;
466}

◆ TEST() [10/37]

TEST ( example_optional_arg  )
469{
470 int iii = 0;
471 iii = getValue<int>(iii, iii);
472 iii = getValue<int>(iii);
473 iii = getValue<int>();
474
475 {
476 using namespace std::experimental;
477 optional<Guard> grd1{in_place, "res1", 1}; // guard 1 initialized
479
480 grd2.emplace("res2", 2); // guard 2 initialized
481 grd1 = nullopt; // guard 1 released
482
483 } // guard 2 released (in dtor)
484};

◆ getStartMidEnd()

std::tuple< Date, Date, Date > getStartMidEnd ( )
487{ return std::tuple<Date, Date, Date>{Date{1}, Date{2}, Date{3}}; }
Definition test_optional.cpp:82

◆ run() [1/3]

void run ( Date const &  ,
Date const &  ,
Date const &   
)
488{}

◆ TEST() [11/37]

TEST ( example_date  )
491{
492 using namespace std::experimental;
493 optional<Date> start, mid, end; // Date doesn't have default ctor (no good default date)
494
495 std::tie(start, mid, end) = getStartMidEnd();
496 run(*start, *mid, *end);
497};
std::tuple< Date, Date, Date > getStartMidEnd()
Definition test_optional.cpp:487
void run(Date const &, Date const &, Date const &)
Definition test_optional.cpp:488

◆ readNextChar()

std::experimental::optional< char > readNextChar ( )
500{ return{}; }

◆ run() [2/3]

void run ( std::experimental::optional< std::string >  )
502{}

◆ run() [3/3]

void run ( std::complex< double >  )
503{}

◆ assign_norebind()

template<class T >
void assign_norebind ( tr2::optional< T & > &  optref,
T &  obj 
)
508{
509 if (optref) *optref = obj;
510 else optref.emplace(obj);
511}

◆ unused()

template<typename T >
void unused ( T &&  )
513{}

◆ TEST() [12/37]

TEST ( example_conceptual_model  )
516{
517 using namespace std::experimental;
518
519 optional<int> oi = 0;
520 optional<int> oj = 1;
522
523 oi = 1;
524 oj = nullopt;
525 ok = 0;
526
527 unused(oi == nullopt);
528 unused(oj == 0);
529 unused(ok == 1);
530};
void unused(T &&)
Definition test_optional.cpp:513

◆ TEST() [13/37]

TEST ( example_rationale  )
533{
534 using namespace std::experimental;
536 unused(ch);
537 // ...
538 }
539
540 //////////////////////////////////
542 optional<int> opt2 = {};
543
544 opt1 = nullopt;
545 opt2 = {};
546
547 if (opt1 == nullopt) {}
548 if (!opt2) {}
549 if (opt2 == optional<int>{}) {}
550
551
552
553 ////////////////////////////////
554
555 run(nullopt); // pick the second overload
556 // FAILS: run({}); // ambiguous
557
558 if (opt1 == nullopt) {} // fine
559 // FAILS: if (opt2 == {}) {} // ilegal
560
561 ////////////////////////////////
566
571
572 /////////////////////////////////
574 o = make_optional(1); // copy/move assignment
575 o = 1; // assignment from T
576 o.emplace(1); // emplacement
577
578 ////////////////////////////////////
579 int isas = 0, i = 9;
582
583 /////////////////////////////////////
584 ////tr2::optional<std::vector<int>> ov2 = {2, 3};
585 ////assert (bool(ov2));
586 ////assert ((*ov2)[1] == 3);
587 ////
588 ////////////////////////////////
589 ////std::vector<int> v = {1, 2, 4, 8};
590 ////optional<std::vector<int>> ov = {1, 2, 4, 8};
591
592 ////assert (v == *ov);
593 ////
594 ////ov = {1, 2, 4, 8};
595
596 ////std::allocator<int> a;
597 ////optional<std::vector<int>> ou { in_place, {1, 2, 4, 8}, a };
598
599 ////assert (ou == ov);
600
601 //////////////////////////////
602 // inconvenient syntax:
603 {
604
606
607 assert (bool(ov2));
608 assert ((*ov2)[1] == 3);
609
610 ////////////////////////////
611
612 std::vector<int> v = {1, 2, 4, 8};
614
615 assert (v == *ov);
616
617 ov.emplace({1, 2, 4, 8});
618/*
619 std::allocator<int> a;
620 optional<std::vector<int>> ou { in_place, {1, 2, 4, 8}, a };
621
622 assert (ou == ov);
623*/
624 }
625
626 /////////////////////////////////
627 {
628 typedef int T;
632
634 auto ooi = make_optional(oi);
635 static_assert( std::is_same<optional<optional<int>>, decltype(ooi)>::value, "");
636 }
637};
std::experimental::optional< char > readNextChar()
Definition test_optional.cpp:500
void assign_norebind(tr2::optional< T & > &optref, T &obj)
Definition test_optional.cpp:507

◆ fun()

bool fun ( std::string  ,
std::experimental::optional< int >  oi = std::experimental::nullopt 
)
641{
642 return bool(oi);
643}

◆ TEST() [14/37]

TEST ( example_converting_ctor  )
646{
647 using namespace std::experimental;
648
649 assert (true == fun("dog", 2));
650 assert (false == fun("dog"));
651 assert (false == fun("dog", nullopt)); // just to be explicit
652};
bool fun(std::string, std::experimental::optional< int > oi=std::experimental::nullopt)
Definition test_optional.cpp:640

◆ TEST() [15/37]

TEST ( bad_comparison  )
656{
657 tr2::optional<int> oi, oj;
658 int i;
659 bool b = (oi == oj);
660 b = (oi >= i);
661 b = (oi == i);
662 unused(b);
663};

◆ TEST() [16/37]

TEST ( value_or  )
678{
679 tr2::optional<int> oi = 1;
680 int i = oi.value_or(0);
681 assert (i == 1);
682
683 oi = tr2::nullopt;
684 assert (oi.value_or(3) == 3);
685
687 assert (os.value_or("BBB") == "AAA");
688 os = {};
689 assert (os.value_or("BBB") == "BBB");
690};
constexpr T value_or(V &&v) const
Definition optional.hpp:608

◆ TEST() [17/37]

TEST ( reset  )
693{
694 using namespace std::experimental;
695 optional<int> oi {1};
696 oi.reset();
697 assert (!oi);
698
699 int i = 1;
701 oir.reset();
702 assert (!oir);
703};
void reset() noexcept
Definition optional.hpp:616

◆ TEST() [18/37]

TEST ( mixed_order  )
706{
707 using namespace std::experimental;
708
710 optional<int> o0 {0};
711 optional<int> o1 {1};
712
713 assert ( (oN < 0));
714 assert ( (oN < 1));
715 assert (!(o0 < 0));
716 assert ( (o0 < 1));
717 assert (!(o1 < 0));
718 assert (!(o1 < 1));
719
720 assert (!(oN >= 0));
721 assert (!(oN >= 1));
722 assert ( (o0 >= 0));
723 assert (!(o0 >= 1));
724 assert ( (o1 >= 0));
725 assert ( (o1 >= 1));
726
727 assert (!(oN > 0));
728 assert (!(oN > 1));
729 assert (!(o0 > 0));
730 assert (!(o0 > 1));
731 assert ( (o1 > 0));
732 assert (!(o1 > 1));
733
734 assert ( (oN <= 0));
735 assert ( (oN <= 1));
736 assert ( (o0 <= 0));
737 assert ( (o0 <= 1));
738 assert (!(o1 <= 0));
739 assert ( (o1 <= 1));
740
741 assert ( (0 > oN));
742 assert ( (1 > oN));
743 assert (!(0 > o0));
744 assert ( (1 > o0));
745 assert (!(0 > o1));
746 assert (!(1 > o1));
747
748 assert (!(0 <= oN));
749 assert (!(1 <= oN));
750 assert ( (0 <= o0));
751 assert (!(1 <= o0));
752 assert ( (0 <= o1));
753 assert ( (1 <= o1));
754
755 assert (!(0 < oN));
756 assert (!(1 < oN));
757 assert (!(0 < o0));
758 assert (!(1 < o0));
759 assert ( (0 < o1));
760 assert (!(1 < o1));
761
762 assert ( (0 >= oN));
763 assert ( (1 >= oN));
764 assert ( (0 >= o0));
765 assert ( (1 >= o0));
766 assert (!(0 >= o1));
767 assert ( (1 >= o1));
768};

◆ operator<()

constexpr bool operator< ( BadRelops  a,
BadRelops  b 
)
constexpr
775{ return a.i < b.i; }
int i
Definition test_optional.cpp:772

◆ operator>()

constexpr bool operator> ( BadRelops  a,
BadRelops  b 
)
constexpr
776{ return a.i < b.i; } // intentional error!

◆ TEST() [19/37]

TEST ( bad_relops  )
779{
780 using namespace std::experimental;
781 BadRelops a{1}, b{2};
782 assert (a < b);
783 assert (a > b);
784
785 optional<BadRelops> oa = a, ob = b;
786 assert (oa < ob);
787 assert (!(oa > ob));
788
789 assert (oa < b);
790 assert (oa > b);
791
792 optional<BadRelops&> ra = a, rb = b;
793 assert (ra < rb);
794 assert (!(ra > rb));
795
796 assert (ra < b);
797 assert (ra > b);
798};
Definition test_optional.cpp:771

◆ TEST() [20/37]

TEST ( mixed_equality  )
802{
803 using namespace std::experimental;
804
805 assert (make_optional(0) == 0);
806 assert (make_optional(1) == 1);
807 assert (make_optional(0) != 1);
808 assert (make_optional(1) != 0);
809
811 optional<int> o0 {0};
812 optional<int> o1 {1};
813
814 assert (o0 == 0);
815 assert ( 0 == o0);
816 assert (o1 == 1);
817 assert ( 1 == o1);
818 assert (o1 != 0);
819 assert ( 0 != o1);
820 assert (o0 != 1);
821 assert ( 1 != o0);
822
823 assert ( 1 != oN);
824 assert ( 0 != oN);
825 assert (oN != 1);
826 assert (oN != 0);
827 assert (!( 1 == oN));
828 assert (!( 0 == oN));
829 assert (!(oN == 1));
830 assert (!(oN == 0));
831
832 std::string cat{"cat"}, dog{"dog"};
833 optional<std::string> oNil{}, oDog{"dog"}, oCat{"cat"};
834
835 assert (oCat == cat);
836 assert ( cat == oCat);
837 assert (oDog == dog);
838 assert ( dog == oDog);
839 assert (oDog != cat);
840 assert ( cat != oDog);
841 assert (oCat != dog);
842 assert ( dog != oCat);
843
844 assert ( dog != oNil);
845 assert ( cat != oNil);
846 assert (oNil != dog);
847 assert (oNil != cat);
848 assert (!( dog == oNil));
849 assert (!( cat == oNil));
850 assert (!(oNil == dog));
851 assert (!(oNil == cat));
852};

◆ TEST() [21/37]

TEST ( const_propagation  )
855{
856 using namespace std::experimental;
857
859 static_assert(std::is_same<decltype(*mmi), int&>::value, "WTF");
860
861 const optional<int> cmi{0};
862 static_assert(std::is_same<decltype(*cmi), const int&>::value, "WTF");
863
865 static_assert(std::is_same<decltype(*mci), const int&>::value, "WTF");
866
868 static_assert(std::is_same<decltype(*cci), const int&>::value, "WTF");
869};

◆ TEST() [22/37]

TEST ( safe_value  )
875{
876 using namespace std::experimental;
877
878 try {
879 optional<int> ovN{}, ov1{1};
880
881 int& r1 = ov1.value();
882 assert (r1 == 1);
883
884 try {
885 ovN.value();
886 assert (false);
887 }
888 catch (bad_optional_access const&) {
889 }
890
891 { // ref variant
892 int i1 = 1;
894
895 int& r2 = or1.value();
896 assert (r2 == 1);
897
898 try {
899 orN.value();
900 assert (false);
901 }
902 catch (bad_optional_access const&) {
903 }
904 }
905 }
906 catch(...) {
907 assert (false);
908 }
909};
Definition optional.hpp:272
constexpr T const & value() const
Definition optional.hpp:569

◆ TEST() [23/37]

TEST ( optional_ref  )
912{
913 using namespace tr2;
914 // FAILS: optional<int&&> orr;
915 // FAILS: optional<nullopt_t&> on;
916 int i = 8;
917 optional<int&> ori;
918 assert (!ori);
919 ori.emplace(i);
920 assert (bool(ori));
921 assert (*ori == 8);
922 assert (&*ori == &i);
923 *ori = 9;
924 assert (i == 9);
925
926 // FAILS: int& ir = ori.value_or(i);
927 int ii = ori.value_or(i);
928 assert (ii == 9);
929 ii = 7;
930 assert (*ori == 9);
931
932 int j = 22;
933 auto&& oj = make_optional(std::ref(j));
934 *oj = 23;
935 assert (&*oj == &j);
936 assert (j == 23);
937};

◆ TEST() [24/37]

TEST ( optional_ref_const_propagation  )
940{
941 using namespace std::experimental;
942
943 int i = 9;
944 const optional<int&> mi = i;
945 int& r = *mi;
947 static_assert(std::is_same<decltype(*mi), int&>::value, "WTF");
948 static_assert(std::is_same<decltype(*ci), const int&>::value, "WTF");
949
950 unused(r);
951};

◆ TEST() [25/37]

TEST ( optional_ref_assign  )
954{
955 using namespace std::experimental;
956
957 int i = 9;
959
960 int j = 1;
962 ori = {j};
963 // FAILS: ori = j;
964
966 ori = orx;
967
969
970 assert (ori);
971 assert (*ori == 1);
972 assert (ori == orj);
973 assert (i == 9);
974
975 *ori = 2;
976 assert (*ori == 2);
977 assert (ori == 2);
978 assert (2 == ori);
979 assert (ori != 3);
980
981 assert (ori == orj);
982 assert (j == 2);
983 assert (i == 9);
984
985 ori = {};
986 assert (!ori);
987 assert (ori != orj);
988 assert (j == 2);
989 assert (i == 9);
990};

◆ TEST() [26/37]

TEST ( optional_swap  )
993{
994 namespace tr2 = std::experimental;
995 tr2::optional<int> oi {1}, oj {};
996 swap(oi, oj);
997 assert (oj);
998 assert (*oj == 1);
999 assert (!oi);
1000 static_assert(noexcept(swap(oi, oj)), "swap() is not noexcept");
1001};
void swap(optional< T > &x, optional< T > &y) noexcept(noexcept(x.swap(y)))
Definition optional.hpp:1016

◆ TEST() [27/37]

TEST ( optional_ref_swap  )
1005{
1006 using namespace std::experimental;
1007 int i = 0;
1008 int j = 1;
1009 optional<int&> oi = i;
1011
1012 assert (&*oi == &i);
1013 assert (&*oj == &j);
1014
1015 swap(oi, oj);
1016 assert (&*oi == &j);
1017 assert (&*oj == &i);
1018};

◆ TEST() [28/37]

TEST ( optional_initialization  )
1021{
1022 using namespace tr2;
1023 using std::string;
1024 string s = "STR";
1025
1026 optional<string> os{s};
1027 optional<string> ot = s;
1028 optional<string> ou{"STR"};
1029 optional<string> ov = string{"STR"};
1030
1031};

◆ TEST() [29/37]

TEST ( optional_hashing  )
1036{
1037 using namespace tr2;
1038 using std::string;
1039
1040 std::hash<int> hi;
1041 std::hash<optional<int>> hoi;
1042 std::hash<string> hs;
1043 std::hash<optional<string>> hos;
1044
1045 assert (hi(0) == hoi(optional<int>{0}));
1046 assert (hi(1) == hoi(optional<int>{1}));
1047 assert (hi(3198) == hoi(optional<int>{3198}));
1048
1049 assert (hs("") == hos(optional<string>{""}));
1050 assert (hs("0") == hos(optional<string>{"0"}));
1051 assert (hs("Qa1#") == hos(optional<string>{"Qa1#"}));
1052
1053 std::unordered_set<optional<string>> set;
1054 assert(set.find({"Qa1#"}) == set.end());
1055
1056 set.insert({"0"});
1057 assert(set.find({"Qa1#"}) == set.end());
1058
1059 set.insert({"Qa1#"});
1060 assert(set.find({"Qa1#"}) != set.end());
1061};

◆ generic_fun()

template<class X >
bool generic_fun ( )
1082{
1084 return bool(op);
1085}

◆ TEST() [30/37]

TEST ( optional_ref_emulation  )
1088{
1089 using namespace std::experimental;
1091 assert (*oi == 1);
1092
1093 int i = 8;
1094 int j = 4;
1096 assert (*ori == 8);
1097 assert ((void*)&*ori != (void*)&i); // !DIFFERENT THAN optional<T&>
1098
1099 *ori = j;
1100 assert (*ori == 4);
1101};

◆ TEST() [31/37]

TEST ( optional_ref_hashing  )
1140{
1141 using namespace tr2;
1142 using std::string;
1143
1144 std::hash<int> hi;
1145 std::hash<optional<int&>> hoi;
1146 std::hash<string> hs;
1147 std::hash<optional<string&>> hos;
1148
1149 int i0 = 0;
1150 int i1 = 1;
1151 assert (hi(0) == hoi(optional<int&>{i0}));
1152 assert (hi(1) == hoi(optional<int&>{i1}));
1153
1154 string s{""};
1155 string s0{"0"};
1156 string sCAT{"CAT"};
1157 assert (hs("") == hos(optional<string&>{s}));
1158 assert (hs("0") == hos(optional<string&>{s0}));
1159 assert (hs("CAT") == hos(optional<string&>{sCAT}));
1160
1161 std::unordered_set<optional<string&>> set;
1162 assert(set.find({sCAT}) == set.end());
1163
1164 set.insert({s0});
1165 assert(set.find({sCAT}) == set.end());
1166
1167 set.insert({sCAT});
1168 assert(set.find({sCAT}) != set.end());
1169};

◆ TEST() [32/37]

TEST ( arrow_operator  )
1193{
1194 using namespace std::experimental;
1195
1197 assert (oc1);
1198 assert (oc1->m == 1);
1199 assert (oc1->n == 2);
1200
1202 assert (on);
1203 assert (on->m == 1);
1204 assert (on->n == 2);
1205};

◆ TEST() [33/37]

TEST ( arrow_wit_optional_ref  )
1208{
1209 using namespace std::experimental;
1210
1211 Combined c{1, 2};
1213 assert (oc);
1214 assert (oc->m == 1);
1215 assert (oc->n == 2);
1216
1217 Nasty n{1, 2};
1218 Nasty m{3, 4};
1219 Nasty p{5, 6};
1220
1222 assert (on);
1223 assert (on->m == 1);
1224 assert (on->n == 2);
1225
1226 on = {m};
1227 assert (on);
1228 assert (on->m == 3);
1229 assert (on->n == 4);
1230
1231 on.emplace(p);
1232 assert (on);
1233 assert (on->m == 5);
1234 assert (on->n == 6);
1235
1237 assert (om);
1238 assert (om->m == 1);
1239 assert (om->n == 2);
1240};
constexpr Combined c
Definition test_optional.cpp:1460
constexpr optional< Combined const & > oc
Definition test_optional.cpp:1461
Definition test_optional.cpp:1172
Definition test_optional.cpp:1181

◆ TEST() [34/37]

TEST ( no_dangling_reference_in_value  )
1243{
1244 // this mostly tests compiler warnings
1245 using namespace std::experimental;
1246 optional<int> oi {2};
1247 unused (oi.value());
1248 const optional<int> coi {3};
1249 unused (coi.value());
1250};

◆ TEST() [35/37]

TEST ( exception_safety  )
1264{
1265 using namespace std::experimental;
1266 try {
1267 optional<CountedObject> oo(in_place, true); // throw
1269 }
1270 catch(...)
1271 {
1272 //
1273 }
1275
1276 try {
1277 optional<CountedObject> oo(in_place, true); // throw
1278 optional<CountedObject> o1(std::move(oo)); // now move
1279 }
1280 catch(...)
1281 {
1282 //
1283 }
1285};
static int _counter
Definition test_optional.cpp:1254

◆ TEST() [36/37]

TEST ( nested_optional  )
1288{
1289 using namespace std::experimental;
1290
1292 assert (!o1);
1293
1295 assert (o2);
1296 assert (!*o2);
1297
1298 optional<optional<optional<int>>> o3 (in_place, in_place, nullopt);
1299 assert (o3);
1300 assert (*o3);
1301 assert (!**o3);
1302};

◆ TEST() [37/37]

TEST ( three_ways_of_having_value  )
1305{
1306 using namespace std::experimental;
1307 optional<int> oN, o1 (1);
1308
1309 assert (!oN);
1310 assert (!oN.has_value());
1311 assert (oN == nullopt);
1312
1313 assert (o1);
1314 assert (o1.has_value());
1315 assert (o1 != nullopt);
1316
1317 assert (bool(oN) == oN.has_value());
1318 assert (bool(o1) == o1.has_value());
1319
1320 int i = 1;
1321 optional<int&> rN, r1 (i);
1322
1323 assert (!rN);
1324 assert (!rN.has_value());
1325 assert (rN == nullopt);
1326
1327 assert (r1);
1328 assert (r1.has_value());
1329 assert (r1 != nullopt);
1330
1331 assert (bool(rN) == rN.has_value());
1332 assert (bool(r1) == r1.has_value());
1333};
constexpr bool has_value() const noexcept
Definition optional.hpp:513

◆ test_noexcept()

void test_noexcept ( )
1356{
1357 {
1359 static_assert(noexcept(tr2::optional<NothrowBoth>{tr2::constexpr_move(b1)}), "bad noexcept!");
1360 static_assert(noexcept(b1 = tr2::constexpr_move(b2)), "bad noexcept!");
1361 }
1362 {
1364 static_assert(noexcept(tr2::optional<NothrowCtor>{tr2::constexpr_move(c1)}), "bad noexcept!");
1365 static_assert(!noexcept(c1 = tr2::constexpr_move(c2)), "bad noexcept!");
1366 }
1367 {
1369 static_assert(!noexcept(tr2::optional<NothrowAssign>{tr2::constexpr_move(a1)}), "bad noexcept!");
1370 static_assert(!noexcept(a1 = tr2::constexpr_move(a2)), "bad noexcept!");
1371 }
1372 {
1374 static_assert(!noexcept(tr2::optional<NothrowNone>{tr2::constexpr_move(n1)}), "bad noexcept!");
1375 static_assert(!noexcept(n1 = tr2::constexpr_move(n2)), "bad noexcept!");
1376 }
1377}
constexpr std::remove_reference< T >::type && constexpr_move(T &&t) noexcept
Definition optional.hpp:193

◆ constexpr_test_disengaged()

void constexpr_test_disengaged ( )
1381{
1382 constexpr tr2::optional<int> g0{};
1383 constexpr tr2::optional<int> g1{tr2::nullopt};
1384 static_assert( !g0, "initialized!" );
1385 static_assert( !g1, "initialized!" );
1386
1387 static_assert( bool(g1) == bool(g0), "ne!" );
1388
1389 static_assert( g1 == g0, "ne!" );
1390 static_assert( !(g1 != g0), "ne!" );
1391 static_assert( g1 >= g0, "ne!" );
1392 static_assert( !(g1 > g0), "ne!" );
1393 static_assert( g1 <= g0, "ne!" );
1394 static_assert( !(g1 <g0), "ne!" );
1395
1396 static_assert( g1 == tr2::nullopt, "!" );
1397 static_assert( !(g1 != tr2::nullopt), "!" );
1398 static_assert( g1 <= tr2::nullopt, "!" );
1399 static_assert( !(g1 < tr2::nullopt), "!" );
1400 static_assert( g1 >= tr2::nullopt, "!" );
1401 static_assert( !(g1 > tr2::nullopt), "!" );
1402
1403 static_assert( (tr2::nullopt == g0), "!" );
1404 static_assert( !(tr2::nullopt != g0), "!" );
1405 static_assert( (tr2::nullopt >= g0), "!" );
1406 static_assert( !(tr2::nullopt > g0), "!" );
1407 static_assert( (tr2::nullopt <= g0), "!" );
1408 static_assert( !(tr2::nullopt < g0), "!" );
1409
1410 static_assert( (g1 != tr2::optional<int>(1)), "!" );
1411 static_assert( !(g1 == tr2::optional<int>(1)), "!" );
1412 static_assert( (g1 < tr2::optional<int>(1)), "!" );
1413 static_assert( (g1 <= tr2::optional<int>(1)), "!" );
1414 static_assert( !(g1 > tr2::optional<int>(1)), "!" );
1415 static_assert( !(g1 > tr2::optional<int>(1)), "!" );
1416}
constexpr tr2::optional< int > g0
Definition test_optional.cpp:1419

◆ main()

int main ( )
1508 {
1509 tr2::optional<int> oi = 1;
1510 assert (bool(oi));
1511 oi.operator=({});
1512 assert (!oi);
1513
1514 VEC v = {5, 6};
1515
1517 std::cout << "Optional has rvalue references for *this" << std::endl;
1518 else
1519 std::cout << "Optional doesn't have rvalue references for *this" << std::endl;
1520
1522 std::cout << "Optional has constexpr initializer_list" << std::endl;
1523 else
1524 std::cout << "Optional doesn't have constexpr initializer_list" << std::endl;
1525
1527 std::cout << "Optional has constexpr move accessors" << std::endl;
1528 else
1529 std::cout << "Optional doesn't have constexpr move accessors" << std::endl;
1530}
#define OPTIONAL_HAS_THIS_RVALUE_REFS
Definition optional.hpp:75
#define OPTIONAL_HAS_MOVE_ACCESSORS
Definition optional.hpp:90
#define OPTIONAL_HAS_CONSTEXPR_INIT_LIST
Definition optional.hpp:83
Definition test_optional.cpp:1497

Variable Documentation

◆ g0

constexpr tr2::optional<int> g0 {}
constexpr
1419{};

◆ g2

constexpr tr2::optional<int> g2 {2}
constexpr
1420{2};

◆ gc0

constexpr tr2::optional<Combined> gc0 {tr2::in_place}
constexpr

◆ gi

int gi = 0

◆ gori

constexpr tr2::optional<int&> gori = gi
constexpr

◆ gorn

constexpr tr2::optional<int&> gorn {}
constexpr
1439{};

◆ gri

constexpr int& gri = *gori
constexpr

◆ gci

constexpr int gci = 1
constexpr

◆ gorci

constexpr tr2::optional<int const&> gorci = gci
constexpr

◆ gorcn

constexpr tr2::optional<int const&> gorcn {}
constexpr
1449{};