Static constexpr of class inside class link problems

前端 未结 4 753
花落未央
花落未央 2021-01-21 06:27

I\'m trying to create static constexprs defined inside the class. I know about this question: static constexpr member of same type as class being defined, and metho

相关标签:
4条回答
  • 2021-01-21 06:33

    As with any variable definition, you should place it in one translation unit only.

    Move that last line out of your header.

    0 讨论(0)
  • 2021-01-21 06:38

    Well, you define first three static const variables and then you defines them as constexpr. Since constexpr static member must be complete, it cannot be in the scope of the class itself. Your code should look like this:

    class foo {
    int _x;
    public:
        constexpr foo(int x) : _x(x) {}
        constexpr int x() const { return _x; }
    };
    
    constexpr foo a = foo{1}, b = foo{2}, c = foo{1};
    

    If you still desire to have foo as a static member you can do this little trick:

    class foo {
    int _x;
        constexpr foo(int x) : _x(x) {}
        struct constants;
    public:
        constexpr int x() const { return _x; }
    };
    
    struct foo::constants {
        constexpr static foo a = foo{1}, b = foo{2}, c = foo{1};
    };
    

    Follow this if you're using C++14 and before. In C++17, all constexpr static data members are implicitly inline.

    Now why the link error?

    There's this little rule called the One Definition Rule, which states that there can be any number of declaration but one definition across all compilation units. By your code snippet you included in your question, it look like you are defining your static member in your header. If there's two compilation unit including your header, you break the rule. With my code example above you should not have this error anymore, but another one instead: there is no definition. the constexpr value might be used at compile time, but won't be usable at runtime. To do this, you must declare this in a .cpp file:

    #include "foo.h"
    
    constexpr foo foo::constants::a;
    constexpr foo foo::constants::b;
    constexpr foo foo::constants::c;
    

    Now your three static variables are correctly declared and defined.

    0 讨论(0)
  • 2021-01-21 06:38

    Move the

    constexpr foo foo::a = foo(1), foo::b = foo(2), foo::c = foo(1);

    line to a .cc file. That line defines the variables. A variable can be declared as many times as you like, but only defined once across your entire binary.

    0 讨论(0)
  • 2021-01-21 06:57

    In addition to Guillaume Racicot's answer you can add constexpr function that return copy or const reference to constexpr values, that defines in some other place. Or call constructor and return new instane every time, without keep constexpr variables. Tested on VS2015:

    class foo {
        int _x;
        constexpr foo(int x) : _x(x) {}
        struct constants;
    public:
    
        constexpr int x() const { return _x; }
    
        static constexpr const foo &a();
        static constexpr const foo &b();
        static constexpr const foo &c();
        static constexpr foo d() { return foo(5); }
    };
    
    struct foo::constants {
        constexpr static foo a = foo{ 1 }, b = foo{ 2 }, c = foo{ 1 };
    };
    
    constexpr const foo &foo::a() { return constants::a; }
    constexpr const foo &foo::b() { return constants::b; }
    constexpr const foo &foo::c() { return constants::c; }
    

    Then you can get values from the foo's scope, for example: static_assert(foo::a().x()==1,"");

    0 讨论(0)
提交回复
热议问题