Почему vector <bool>:: reference не возвращает ссылку на bool? - PullRequest
55 голосов
/ 06 декабря 2011
#include <vector>

struct A
{
    void foo(){}
};

template< typename T >
void callIfToggled( bool v1, bool &v2, T & t )
{
    if ( v1 != v2 )
    {
        v2 = v1;
        t.foo();
    }
}

int main()
{
    std::vector< bool > v= { false, true, false };

    const bool f = false;
    A a;

    callIfToggled( f, v[0], a );
    callIfToggled( f, v[1], a );
    callIfToggled( f, v[2], a );
}

Компиляция примера выше выдает следующую ошибку:

dk2.cpp: In function 'int main()':
dk2.cpp:29:28: error: no matching function for call to 'callIfToggled(const bool&, std::vector<bool>::reference, A&)'
dk2.cpp:29:28: note: candidate is:
dk2.cpp:13:6: note: template<class T> void callIfToggled(bool, bool&, T&)

Я скомпилировал с использованием g ++ (версия 4.6.1) следующим образом:

g++ -O3 -std=c++0x -Wall -Wextra -pedantic dk2.cpp

Вопрос в том, почему это происходит? Разве vector<bool>::reference не bool&? Или это ошибка компилятора?
Или я пытаюсь сделать что-то глупое? :)

Ответы [ 6 ]

58 голосов
/ 06 декабря 2011

Vector специализируется на bool .

Это считается ошибкой стандарта.Вместо этого используйте vector<char>:

template<typename t>
struct foo {
  using type = t;
};
template<>
struct foo<bool> {
  using type = char;
};

template<typename t, typename... p>
using fixed_vector = std::vector<typename foo<t>::type, p...>;

Иногда вам могут понадобиться ссылки на bool, содержащийся внутри вектора.К сожалению, использование vector<char> может дать вам только ссылки на символы.Если вам действительно нужен bool&, посмотрите библиотеку Boost Containers .Имеет неспециализированную версию vector<bool>.

49 голосов
/ 06 декабря 2011

Ваши ожидания нормальны, но проблема в том, что std::vector<bool> был своего рода экспериментом комитета C ++.Это на самом деле шаблонная специализация, которая хранит значения bool, плотно упакованные в памяти: один бит на значение.

И, поскольку у вас не может быть ссылки на бит, есть ваша проблема.

15 голосов
/ 06 декабря 2011

std::vector< bool > упаковывает свое содержимое, чтобы каждое логическое значение сохранялось в одном бите, восемь бит в байте.Это экономит память, но требует больших вычислительных ресурсов, поскольку процессор должен выполнять арифметику для доступа к запрошенному биту.И он не работает с bool ссылкой или семантикой указателя, поскольку биты внутри байта не имеют адресов в объектной модели C ++.

Вы все равно можете объявить переменную типа std::vector<bool>::reference и использовать еекак будто это было bool&.Это позволяет совместить универсальные алгоритмы.

std::vector< bool > bitvec( 22 );
std::vector< bool >::reference third = bitvec[ 2 ];
third = true; // assign value to referenced bit

В C ++ 11 вы можете обойти это, используя auto и спецификатор &&, который автоматически выбирает ссылку lvalue, привязанную к элементу вектора, илиссылка на значение, привязанное к временному объекту.

std::vector< bool > bitvec( 22 );
auto &&third = bitvec[ 2 ]; // obtain a std::vector< bool >::reference
third = true; // assign value to referenced bit
15 голосов
/ 06 декабря 2011

std::vector<bool> - это несоответствующий контейнер. Чтобы оптимизировать пространство, он упаковывает bool с и не может предоставить ссылку.

Вместо этого используйте boost::dynamic_bitset.

5 голосов
/ 06 декабря 2011

Только мои 2 цента:

std::vector<bool>::reference - это typedef для struct _Bit_reference, который определяется как

typedef unsigned long _Bit_type;

struct _Bit_reference
  {
    _Bit_type * _M_p;
    _Bit_type _M_mask;

    // constructors, operators, etc...

    operator bool() const
    { return !!(*_M_p & _M_mask); }
  };

Если изменить функцию следующим образом, она работает (ну, по крайней мере, компилирует, еще не тестировала):

template< typename T >
void callIfToggled( bool v1, std::vector<bool>::reference v2, T & t )
{
    bool b = v2;  
    if ( v1 != b )
    {
        v2 = v1;
        t.foo();
    }
}

РЕДАКТИРОВАТЬ: я изменил условие с (v1! = V2), что не было хорошей идеей, на (v1! = B).

1 голос
/ 25 июня 2015

Создайте структуру с bool в ней и создайте vector<>, используя этот тип структуры.

Попробуйте:

vector<struct sb>, где sb равно struct {boolean b];

тогда вы можете сказать

push_back({true})

сделать

typedef struct sbool {bool b;} boolstruct;, а затем vector<boolstruct> bs;

...