Статические данные в библиотеках только для заголовков

Я занимаюсь разработкой библиотеки, которая будет состоять только из заголовочных файлов. Пока что он содержит только классы, что было хорошо. Тем не менее, я пришел к тому, что мне нужно иметь в библиотеке некоторые неизменяемые неизменяемые данные в библиотеке (то есть не данные экземпляра класса) для реализации некоторых функций. Очевидно, вы не можете просто поместить глобальные данные в заголовочные файлы или в каждую единицу компиляции, которая #includes заголовок будет иметь определение для символа, и вы получите несколько ошибок определения во время соединения.

Кажется, я нашел обходной путь, который позволяет мне иметь статические данные в классе, не добавляя модуль компиляции в библиотеку, просто делая данные static переменная в функции и возвращает указатель на эти данные:

class StaticData {
public:
    void doSomething() { /* this uses getData */ }
    void doSomethingElse() { /* this does too */ }

private:
    static int* getData() {
        static int array[] { 1, 2, 3, 4 };

        return array;
    }
};

Кажется, это работает нормально, но я должен признать, что не знаю, что происходит сstatic данные в inline функции в заголовочных файлах. Мне интересно, есть ли у этого "взлома" какие-либо непреднамеренные последствия, такие как каждый модуль компиляции, который #includes этот заголовок получает свою собственную версию array, Как и где компилятор решает поместить его?

Также следует отметить, что я не использую это для реализации одноэлементного антипаттерна или чего-либо еще. Я просто использую его для хранения данных, которые понадобятся нескольким функциям (вот почему это не может быть static только в функции, которая использует его, но даже если бы он это сделал, это вызвало бы тот же вопрос).

2 ответа

Решение

Все в порядке. Гарантируется, что будет только одна копия arrayДо тех пор, пока функция имеет внешнюю связь, что и делает. Стандарт C++ гласит:

7.1.2 / 4 Статическая локальная переменная во внешней встроенной функции всегда ссылается на один и тот же объект.

Другой подход...

template<typename> class ArrayData {
    friend class ClassWithArray;
    static int array[4];
};

class ClassWithArray :
    ArrayData<ClassWithArray>
{
public:
    void doSomething() { 
        /* this uses getData */ 
        array[0] = 1;
        array[1] = 2;
        array[2] = 3;
        array[3] = 4;
    }
    void doSomethingElse() { 
        /* this does too */ 
        array[0] = 4;
        array[1] = 3;
        array[2] = 2;
        array[3] = 1;
    }
};

int ArrayData<ClassWithArray>::array[4] = { 1, 2, 3, 4 };

общая реализация

template<typename T> class ArrayDataT 
{
    friend T;
    static int array[4];
};

template<typename T>
int ArrayDataT<T>::array[4] = { 1, 2, 3 ,4 };

class DerivedFromArrayDataT :
    ArrayDataT<DerivedFromArrayDataT>
{

};
Другие вопросы по тегам