С++ где инициализировать статический const

У меня есть класс

class foo {
public:
   foo();
   foo( int );
private:
   static const string s;
};

Где лучше всего инициализировать строку s в исходном файле?

Ответ 1

В любом месте в блоке компиляции один (обычно файл .cpp):

foo.h

class foo {
    static const string s; // Can never be initialized here.
    static const char* cs; // Same with C strings.

    static const int i = 3; // Integral types can be initialized here (*)...
    static const int j; //     ... OR in cpp.
};

foo.cpp

#include "foo.h"
const string foo::s = "foo string";
const char* foo::cs = "foo C string";
// No definition for i. (*)
const int foo::j = 4;

(*) В соответствии со стандартами вы должны определить i вне определения класса (например, j is), если он используется в коде, отличном от целочисленных постоянных выражений. Более подробную информацию см. В комментарии Дэвида ниже.

Ответ 2

Статические члены должны быть инициализированы в блоке трансляции .cpp в области файлов или в соответствующем пространстве имен:

const string foo::s( "my foo");

Ответ 3

В блоке трансляции в том же пространстве имен, обычно вверху:

// foo.h
struct foo
{
    static const std::string s;
};

// foo.cpp
const std::string foo::s = "thingadongdong"; // this is where it lives

// bar.h
namespace baz
{
    struct bar
    {
        static const float f;
    };
}

// bar.cpp
namespace baz
{
    const float bar::f = 3.1415926535;
}

Ответ 4

Начиная с С++ 17 встроенный спецификатор также применяется к переменным. Теперь вы можете определить статические переменные-члены в определении класса:

#include <string>

class foo {
public:
   foo();
   foo( int );
private:
   inline static const std::string s { "foo" };
};

Ответ 5

В заголовочном файле инициализируются только интегральные значения (например, static const int ARRAYSIZE), поскольку они обычно используются в заголовке класса, чтобы определить что-то такое, как размер массива. Нецелые значения инициализируются в файле реализации.

Ответ 6

const string foo::s( "my foo");

И он должен быть инициализирован в исходном файле, иначе он будет ошибочным, когда вы вызовете его в тестовом случае.