исправить (заблокировать) размер std :: vector - PullRequest
6 голосов
/ 08 июля 2011

Есть ли способ зафиксировать размер вектора и при этом изменить содержимое?

Я попытался создать константный вектор const std::vector<int> vec(10);, но это мешает мне изменить значения.

vec[3] = 3; выдает ошибку компилятора: назначение места только для чтения.

Я также пытался с константной ссылкой на неконстантный вектор

std::vector<int> vec(10);
const std::vector<int>& vecref(vec);

, которая выдает ту же ошибку компилятора.

Я хочу иметь возможность фиксировать размер вектора либо при объявлении, либо после этапа инициализации. Я мог бы использовать старомодный массив, но я хочу использовать векторные алгоритмы.

Я использую g ++, если это что-то меняет.

Ответы [ 4 ]

6 голосов
/ 08 июля 2011

С C ++ 0x вы можете использовать std :: array <>, который похож на старый добрый массив, с дополнительным преимуществом в качестве контейнера STL, что позволяет использовать много алгоритмов std ::

В качестве альтернативы, вы можете попробовать boost :: array.

Обратите внимание, что есть также std::tr1::array<>.


редактирование:

На самом деле, одним из случаев, в которые я не вошел, было увеличение вектора при чтении файлов конфигурации, а затем фиксированный размер - DanS

Тогда, почему не это (иллюстрация):

#include <vector>

int main () {
    std::vector<int> foo;

    /* ... crunch upon foo ... */

    // make a copy vector->vector:
    const std::vector<int> bar (foo); 

    // make a copy any->vector
    const std::vector<int> frob (foo.begin(), foo.end());
}

В качестве альтернативы, если вам нужна семантика reset (), но вы хотите запретить resize () и др., Вы можете написать контейнерный адаптер:

template <typename T, typename Allocator = allocator<T> >
class resettable_array {
public:
    // container ...
    typedef typename std::vector<T,Allocator>::iterator iterator;
    typedef typename std::vector<T,Allocator>::const_iterator const_iterator;
    ...

    iterator begin() { return vector_.begin() }
    const_iterator begin() const { return vector_.begin(); }
    ...

    void push_back (T const &v) { vector_.push_back (v); }
    ...

    // custom
    void reset () { ... }

private:
    std::vector<T,Allocator> vector_;
};

Смотри также:

3 голосов
/ 08 июля 2011

Вставьте его в объект, который предоставляет только те операции, которые вы хотите разрешить.

Приветствия и hth.,

1 голос
/ 08 июля 2011

Взгляните на boost.array, , он дает вам массив фиксированного размера с векторной семантикой (за исключением всего, что может изменить размер массива).

1 голос
/ 08 июля 2011

Вы можете сделать константный вектор указателей и изменить объекты, на которые они указывают.Не сказать, что это правильный ответ, просто возможно.

...