Держите boost :: allocators в качестве членов класса.Можно использовать их для инициализации контейнеров - PullRequest
0 голосов
/ 15 марта 2019

Я хочу поместить в разделяемую память struct1, содержащую вектор struct2, содержащий wstring. Итак:

  • Я сделал два распределителя, один для wchar, один для struct2.
  • Затем определили мои контейнеры wstring и vector.

Я хочу сохранить распределители wchar и struct2 как члены класса, чтобы мне не приходилось хранить ссылку на shm (поскольку мне не нужно создавать их с использованием shm.get_manager ( ) каждый раз, когда я хочу манипулировать данными). Но какую бы конфигурацию я ни использовал, она терпит неудачу.

Теперь я получаю следующую ошибку по адресу:

SharedData(InnerDataAllocator eAlloc):data(eAlloc) { 

Ошибка C2664:
'boost :: container :: vector, void> :: vector (const boost :: container :: vector, void> &)': невозможно преобразовать аргумент 1 из 'InnerDataAllocator' в 'const boost :: container :: new_allocator &'

header.h

typedef bip::allocator<wchar_t, bip::managed_shared_memory::segment_manager> CharAllocator;
typedef bip::basic_string<wchar_t, std::char_traits<wchar_t>, CharAllocator> MyShmString;

class InnerData {
public:
    uint64_t id;
    MyShmString name;
    uint8_t status;
    static const uint8_t defaultStatus = 3;

    InnerData(CharAllocator cAlloc):name(cAlloc) {

    }
};

typedef bip::allocator<InnerData, bip::managed_shared_memory::segment_manager> InnerDataAllocator;
class SharedData {
public:
    int a;
    bool newRequestUntreated = false;
    bool newReplyUntreated = false;

    bip::vector<InnerData> data; 
    SharedData(InnerDataAllocator eAlloc):data(eAlloc) {

    }
};

UsualInnerData{//like InnerData but with normal std::wstring
public:
    uint64_t id;
    std::wstring name;
    uint8_t status;
    static const uint8_t status Default = 3;
};

class Zone {
public:
    SharedData * sharedData;

    CharAllocator     charAllocator;
    InnerDataAllocator   innerDataAllocator;

    void putInShm(const std::vector<UsualInnerData> &in);
    void initializeZone(std::string &_name, std::string &_nameSeg);
    static bool alreadyCreated;
};

code.cpp

void Zone::createNewCheckingZone(std::string &_name, std::string &_nameSeg) {
    if (!alreadyCreated) {
        int size = getOverallSize();
        bip::managed_shared_memory shm(bip::create_only, _name.c_str(), 10000 + size);
        CharAllocator scharAllocator(shm.get_segment_manager());
        swap(charAllocator, scharAllocator); //use of swap because = operator says Not assignable from other allocator

        InnerDataAllocator sinnerDataAllocator(shm.get_segment_manager());
        swap(innerDataAllocator, sinnerDataAllocator);
        sharedData = shm.construct<SharedData>(_nameSeg)(innerDataAllocator);
        alreadyCreated = true;
    }
    return ;
}
void Zone::putInShm(const std::vector<UsualInnerData> &in){
    InnerData newInnerData(charAllocator);
    for (auto a : in) {
        newInnerData.id = a.id;
        newInnerData.name = a.xml.c_str();
        newInnerData.status = InnerData::defaultStatus;
        sharedData->data.push_back(newInnerData);
    }
    return;
}

Поскольку я могу без проблем инициализировать InnerData.name с charAllocator of Zone, почему я не могу сделать то же самое, то есть инициализировать SharedData.data с innerDataAllocator of Zone?

1 Ответ

0 голосов
/ 15 марта 2019

На самом деле, я допустил одну ошибку (вчера допил свой кофейный пакетик, начинаю чувствовать его сейчас), я определил:

typedef  bip::vector<InnerData, InnderDataAllocator> InnerDataVector;

, но не использовал его для определения SharedData.data.Вот как я это изменил:

class SharedData {
public:
    int a;
    bool newRequestUntreated = false;
    bool newReplyUntreated = false;
    InnerDataVector data; 
    SharedData(InnerDataAllocator eAlloc):data(eAlloc) {}
};

А теперь он компилируется.

...