避免使用vector<bool>



作为一个STL容器,vector<bool>只有两点不对。首先,它不是一个STL容器。其次,它并不存储bool。除此之外,一切正常。

一个对象要成为容器,就必须满足C++标准中列出的所有条件,其中一个条件是,如果c是包含对象T的容器,而且c支持operator[],那么下面的代码必须能够被编译:

T *p = &c[0];

换句话说,如果用operator[]取得了container<T>中的一个T对象,那么就可以通过取它的地址得到一个指向该对象的指针。所以,如果vector<bool>是一个容器,那么下面这段代码必须可以被编译:

vector<bool> v;

bool *pb = &v[0];

但是它不能编译。不能编译的原因是,vector<bool>是一个假的容器,它并不真的存储bool,相反,为了节省空间,它存储的是bool的紧凑表示。在一个典型的实现中,存储在“vector”中的每个“bool”仅占一个二进制位,一个8位的字节可容纳8个“bool”。在内部,vector<bool>使用了与位域一样的思想,来表示它所存储的那些bool;实际上它只是假装存储了这些bool。

位域与bool相似,它只能表示两个可能的值,但是在bool和看似bool的位域之间有一个很重要的区别:我们可以创建一个指向bool的指针,而指向单个位的指针则是不允许的。指向单个位的引用也是被禁止的,这使得在设计vector<bool>的接口时产生了一个问题,因为vector<T>::operator[]的返回值应该是T&.如果vector<bool>中所存储的确实是bool,那么这就不是问题。但由于实际上并非如此,所以vector<bool>::operator[]需要返回一个指向一个单个位的引用,而这样的引用并不存在。

当我们需要vector<bool>时,我们有两种选择可以做:

  1. 用deque<bool>。deque几乎提供了vector所提供的一切(可以看到的省略只有reserve和capacity),但deque<bool>是一个STL容器,而且它确实存储bool。当然,deque中元素的内存不是连续的,所以你不能把deque<bool>中的数据传递给一个期望bool数组的C
    API,但对于vector<bool>,我们也不能这么做,因为没有一种可移植的方法能够得到vector<bool>中的数据。
  2. 选择bitset。bitset不是STL容器,但它是标准C++库的一部分。与STL容器不同的是,它的大小(即元素的个数)在编译时就确定了,所以它不支持插入和删除元素。而且,因为它不是一个STL容器,所以它不支持迭代器。但是,与vector<bool>一样,它使用了一种紧凑表示,只为所包含的每个值提供一个空间。它提供了vector<bool>特有的flip成员函数,以及其它一些特有的、对位的集合有意义的成员函数。

避免使用vector<bool>

时间: 2024-10-07 05:33:11

避免使用vector<bool>的相关文章

effective stl 条款18:避免使用vector&lt;bool&gt;

做为一个STL容器,vector<bool>确实只有两个问题.第一,它不是一个STL容器.第二,它并不容纳bool.除此以外,就没有什么要反对的了 . 在这些要求中有这样一条:如果c是一个T类型对象的容器,且c支持operator[],那么以下代码必须能够编译:T *p = &c[0]; // 无论operator[]返回什么,// 都可以用这个地址初始化一个T*换句话说,如果你使用operator[]来得到Container<T>中的一个T对象,你可以通过取它的地址而获得

std::vector&lt;bool&gt;中的坑

http://www.cplusplus.com/reference/vector/vector/?kw=vector C++中,vector<bool>为了达到节省内存的目的,专门做了特化,大概方式就是用bit位来存储数组中的元素.代价就是,这个容器里面的内置类型乱掉了: member type definition notes value_type The first template parameter (bool) allocator_type The second template

C++ 练习记录2---Effective STL中的vector&lt;bool&gt;

//vs2013 32位win8 cout << sizeof(bool) << endl;//1    deque<bool> d(1);     cout << sizeof(d[0]) << endl;//1 vector<bool> v(1);     cout << sizeof(v[0]) << endl;//16            cout << sizeof(d) <<

1.Cocos2dx 3.2中vector,ValueMap,Touch触摸时间的使用.iconv字符编解码

 Cocos2dx3.2以后使用Vector<T>代替了CCArray.案例如下: 头文件:T02Vector.h #ifndef __T02Vector_H__ #define __T02Vector_H__ #include "T32.h" class T02Vector : public Layer { public: CREATE_FUNC(T02Vector); //Cocos2dx3.2以后使用Vector代替了CCArray Vector<Sprit

vector数据查找方法

用STL编写程序时经常使用vector容器来存储数据,当容器中的数据有序时我们可以采取两种方式: (1) 利用<algorithm>中的find函数进行查找: (2) 折半查找. 另外也可以将数据存入hash_map中进行查找,下面来测试比较这两种方法的时间效率. 1. 测试数据集 生成比99999小的所有素数作为查询数据集,查找2到99999之间的所有数. 令数组A存储2~99999之间的所有数,则生成素数的方式 (1) 找到当前最小的数字min: (2) 然后删除min的所有倍数. 重复这

【C++】朝花夕拾——STL vector

STL之vector篇 N久之前是拿C的数组实现过vector中的一些简单功能,什么深拷贝.增删查找之类的,以为vector的实现也就是这样了,现在想想真是...too young too naive...ORZ ====================我是分割线============================= vector属于顺序容器,它的底层实现就是基于array,所以它可以支持随机访问,但是它比array更有效率,因为它动态分配的内存空间. 动态分配的内存空间: 每当vector

【STL基础】vector

vector 构造函数: //default: vector<T> v; //空的vector //fill: vector<T> v(n); //n个元素的vector,元素默认初始化 vector<T> v(n, value); //n个元素值为value的vector //range: vector<T> v(first, last); //两个迭代器之间的元素构成的vector vector<T> v(arr, arr + sizeof(

C++:STL vector:sizeof(vector) (转载)

原文地址:http://blog.csdn.net/zcsylj/article/details/7857009 int的大小是4,定义vector<int> vec,vec中有一个元素,sizeof(vec)=20,如果有1000个元素,则sizeof(vec)是多少? 1 #include <iostream>   2 #include <vector>   3 using namespace std;   4 int main()   5 {   6      v

Effective STL -- vector和string

13.vector和string优先于动态分配的数组 使用vector和string和数组相比,减少了管理内存的工作量. 可以使用begin,end,size等函数 vector和string拥有iterator,value_type等类型定义. 注意事项: string可能使用引用计数,在多线程环境下同步可能会导致性能变差. 14.使用reserve来避免不必要的内存分配 vector超过容量后会导致删除原来容器的对象,进行析构操作,扩大内存后会对原来数据对象复制构造,降低性能. 15.注意s