variant_subset.cpp 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159
  1. // Copyright 2017 Peter Dimov.
  2. //
  3. // Distributed under the Boost Software License, Version 1.0.
  4. //
  5. // See accompanying file LICENSE_1_0.txt or copy at
  6. // http://www.boost.org/LICENSE_1_0.txt
  7. #if defined(_MSC_VER)
  8. # pragma warning( disable: 4702 ) // unreachable code
  9. #endif
  10. #include <boost/variant2/variant.hpp>
  11. #include <boost/core/lightweight_test.hpp>
  12. #include <boost/core/lightweight_test_trait.hpp>
  13. #include <type_traits>
  14. #include <utility>
  15. #include <string>
  16. using namespace boost::variant2;
  17. #define STATIC_ASSERT(...) static_assert(__VA_ARGS__, #__VA_ARGS__)
  18. struct X1
  19. {
  20. int v;
  21. X1(): v(0) {}
  22. explicit X1(int v): v(v) {}
  23. X1(X1 const& r): v(r.v) {}
  24. X1(X1&& r): v(r.v) {}
  25. X1& operator=( X1 const& r ) { v = r.v; return *this; }
  26. X1& operator=( X1&& r ) { v = r.v; return *this; }
  27. };
  28. inline bool operator==( X1 const& a, X1 const& b ) { return a.v == b.v; }
  29. STATIC_ASSERT( !std::is_nothrow_default_constructible<X1>::value );
  30. STATIC_ASSERT( !std::is_nothrow_copy_constructible<X1>::value );
  31. STATIC_ASSERT( !std::is_nothrow_move_constructible<X1>::value );
  32. STATIC_ASSERT( !std::is_nothrow_copy_assignable<X1>::value );
  33. STATIC_ASSERT( !std::is_nothrow_move_assignable<X1>::value );
  34. struct X2
  35. {
  36. int v;
  37. X2(): v(0) {}
  38. explicit X2(int v): v(v) {}
  39. X2(X2 const& r): v(r.v) {}
  40. X2(X2&& r): v(r.v) {}
  41. X2& operator=( X2 const& r ) { v = r.v; return *this; }
  42. X2& operator=( X2&& r ) { v = r.v; return *this; }
  43. };
  44. inline bool operator==( X2 const& a, X2 const& b ) { return a.v == b.v; }
  45. STATIC_ASSERT( !std::is_nothrow_default_constructible<X2>::value );
  46. STATIC_ASSERT( !std::is_nothrow_copy_constructible<X2>::value );
  47. STATIC_ASSERT( !std::is_nothrow_move_constructible<X2>::value );
  48. STATIC_ASSERT( !std::is_nothrow_copy_assignable<X2>::value );
  49. STATIC_ASSERT( !std::is_nothrow_move_assignable<X2>::value );
  50. int main()
  51. {
  52. {
  53. variant<int, float> v1( 1 );
  54. variant<int> v2 = v1.subset<int>();
  55. BOOST_TEST( holds_alternative<int>( v2 ) );
  56. BOOST_TEST_EQ( get<int>( v1 ), get<int>( v2 ) );
  57. BOOST_TEST_THROWS( v1.subset<float>(), bad_variant_access );
  58. variant<int> v3 = std::move(v1).subset<int>();
  59. BOOST_TEST( holds_alternative<int>( v3 ) );
  60. BOOST_TEST_EQ( get<int>( v2 ), get<int>( v3 ) );
  61. BOOST_TEST_THROWS( std::move(v1).subset<float>(), bad_variant_access );
  62. }
  63. {
  64. variant<int, float> const v1( 1 );
  65. variant<int> v2 = v1.subset<int>();
  66. BOOST_TEST( holds_alternative<int>( v2 ) );
  67. BOOST_TEST_EQ( get<int>( v1 ), get<int>( v2 ) );
  68. BOOST_TEST_THROWS( v1.subset<float>(), bad_variant_access );
  69. variant<int> v3 = std::move(v1).subset<int>();
  70. BOOST_TEST( holds_alternative<int>( v3 ) );
  71. BOOST_TEST_EQ( get<int>( v2 ), get<int>( v3 ) );
  72. BOOST_TEST_THROWS( std::move(v1).subset<float>(), bad_variant_access );
  73. }
  74. {
  75. variant<int, float> v1( 1 );
  76. variant<int, float> v2 = v1.subset<int, float>();
  77. BOOST_TEST( holds_alternative<int>( v2 ) );
  78. BOOST_TEST_EQ( get<int>( v1 ), get<int>( v2 ) );
  79. variant<int, float> v3 = std::move(v1).subset<int, float>();
  80. BOOST_TEST( holds_alternative<int>( v3 ) );
  81. BOOST_TEST_EQ( get<int>( v2 ), get<int>( v3 ) );
  82. }
  83. {
  84. variant<int, float> v1( 1 );
  85. variant<float, int> v2 = v1.subset<float, int>();
  86. BOOST_TEST( holds_alternative<int>( v2 ) );
  87. BOOST_TEST_EQ( get<int>( v1 ), get<int>( v2 ) );
  88. variant<float, int> v3 = std::move(v1).subset<float, int>();
  89. BOOST_TEST( holds_alternative<int>( v3 ) );
  90. BOOST_TEST_EQ( get<int>( v2 ), get<int>( v3 ) );
  91. }
  92. {
  93. variant<int, float, std::string> v1( "s1" );
  94. variant<int, std::string> v2 = v1.subset<int, std::string>();
  95. BOOST_TEST( holds_alternative<std::string>( v2 ) );
  96. BOOST_TEST_EQ( get<std::string>( v1 ), get<std::string>( v2 ) );
  97. variant<float, std::string> v3 = std::move(v1).subset<float, std::string>();
  98. BOOST_TEST( holds_alternative<std::string>( v3 ) );
  99. BOOST_TEST_EQ( get<std::string>( v2 ), get<std::string>( v3 ) );
  100. }
  101. {
  102. variant<int, int, float, float, X1, X2> v1{ X1{1} };
  103. variant<X1, X2> v2 = v1.subset<X1, X2>();
  104. BOOST_TEST( holds_alternative<X1>( v2 ) );
  105. BOOST_TEST_EQ( get<X1>( v1 ).v, get<X1>( v2 ).v );
  106. variant<X1, X2> v3 = std::move( v1 ).subset<X1, X2>();
  107. BOOST_TEST( holds_alternative<X1>( v3 ) );
  108. BOOST_TEST_EQ( get<X1>( v2 ).v, get<X1>( v3 ).v );
  109. }
  110. return boost::report_errors();
  111. }