Search code examples
c++c++17std-variant

compiler error with std::variant - use of deleted function error


I have a class C that contains a struct S and a std::variant as members. The struct S has an int member a that is initialized to 0. Here is the code:

#include <variant>

class C
{
    struct S {
        int a = 0;
    };
    std::variant<S> test;
};


int main()
{
    C ctest;
    return 0;
}

When I try to compile this code with gcc 12.2.1 (also with different compilers), I get the following error:

main.cpp: In function 'int main(':
main.cpp:14:7: error: use of deleted function 'C::C()'
   14 |     C ctest;
      |       ^~~~~
main.cpp:3:7: note: 'C::C()' is implicitly deleted because the default definition would be ill-formed:
    3 | class C
      |       ^
main.cpp:3:7: error: use of deleted function 'std::variant<_Types>::variant() [with _Types = {C::S}]'
In file included from main.cpp:1:
/opt/compiler-explorer/gcc-12.2.0/include/c++/12.2.0/variant:1402:7: note: 'std::variant<_Types>::variant() [with _Types = {C::S}]' is implicitly deleted because the default definition would be ill-formed:
 1402 |       variant() = default;
      |       ^~~~~~~
/opt/compiler-explorer/gcc-12.2.0/include/c++/12.2.0/variant:1402:7: error: use of deleted function 'constexpr std::_Enable_default_constructor<false, _Tag>::_Enable_default_constructor() [with _Tag = std::variant<C::S>]'
In file included from /opt/compiler-explorer/gcc-12.2.0/include/c++/12.2.0/variant:38:
/opt/compiler-explorer/gcc-12.2.0/include/c++/12.2.0/bits/enable_special_members.h:113:15: note: declared here
  113 |     constexpr _Enable_default_constructor() noexcept = delete;
      |               ^~~~~~~~~~~~~~~~~~~~~~~~~~~

You can see the code and the error message here: https://onlinegdb.com/ZdfGp9avn

However, if I remove the default assignment =0 from the struct S, the code compiles without errors. Why is this happening? How can I fix this error without removing the default assignment? What is the difference between having a default assignment or not in this case?


Solution

  • S is not default constructible until a complete-class context because the the non-static data member initializers's definitions are not available yet (only the declarations of the members).

    This means when std::variant<S> tries to calculate whether S is default constructible, all the compiler has is struct S { int a = ???; }. It doesn't know whether the compiler-generated default constructor should throw because it doesn't know if a's initializer is throwing.

    The fix is to specify it with a manual noexcept specifier:

    class C
    {
        struct S {
            int a = 0;
            S() noexcept = default;
            // Or `S() noexcept(false) = default;` if it isn't noexcept
        };
        std::variant<S> test;
    };
    

    Or to move the type out so it is complete when used:

    struct S {
        int a = 0;
    };
    
    class C {
        using S = ::S;
        std::variant<S> test;
    };