作为一个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&lt;bool&gt;的更多相关文章

  1. 说一说vector<bool>

    vector<T>标准库模版类应该是绝大多数c++程序员使用频率比较高的一个类了.不过vector<bool>也许就不那么被程序员所了解.关于vector<bool> ...

  2. STL:vector<bool> 和bitset

    今天某个地方要用到很多位标记于是想着可以用下bitset,不过发现居然是编译时确定空间的,不能动态分配.那就只能用vector来代替一下了,不过发现居然有vector<bool>这个特化模 ...

  3. C++ std::vector<bool>

    std::vector template < class T, class Alloc = allocator<T> > class vector; // generic te ...

  4. vector 内部方法大全 学习(初学者的参考资料)

    1    vector构造函数:也就是如何对一个vector对象进行初始化 ////////////////////////////代码//////////////////////////////// ...

  5. QVector 和vector的比较

    QVector和vector的比较: Qvector默认使用隐式共享,可以用setSharable改变其隐式共享.使用non-const操作和函数将引起深拷贝.at()比operator[](),快, ...

  6. vector的成员函数解析

    vector是线性容器,它的元素严格的依照线性序列排序,和动态数组非常相似,和数组一样,它的元素存储在一块连续的存储空间中,这也意味着我们不仅能够使用迭代器(iterator)訪问元素,还能够使用指针 ...

  7. QVector 和vector的比较(QVector默认使用隐式共享,而且有更多的函数提供)

    QVector和vector的比较: Qvector默认使用隐式共享,可以用setSharable改变其隐式共享.使用non-const操作和函数将引起深拷贝.at()比operator[](),快, ...

  8. vector成员函数解析

    vector线性集装箱,其元素颜格排序根据线性序列,和动态数组很阶段似,像阵列,它的元素被存储在连续的存储空间,这也意味着,我们不仅能够使用迭代器(iterator)访问元素,也可以用一个指针访问偏移 ...

  9. vector函数用法

    vector是线性容器,它的元素严格的按照线性序列排序,和动态数组很相似,和数组一样,它的元素存储在一块连续的存储空间中,这也意味着我们不仅可以使用迭代器(iterator)访问元素,还可以使用指针的 ...

随机推荐

  1. bonding的系统初始化介绍

    bond0模块的加载 Bonding原理 为方便理解bonding的配置及实现,顺便阐述一下Linux的网络接口及其配置文件.在 Linux 中,所有的网络通讯都发生在软件接口与物理网络设备之间.与网 ...

  2. Java会话(session)管理

    会话概述 什么是会话 简单的理解:用户打开浏览器,点击多个超链接,访问Web服务器上多个资源,然后关闭浏览器,整个过程称之为一次会话. 需要解决的问题 每个用户在使用浏览器与服务器会话的过程中,会产生 ...

  3. swift详解之十-------------异常处理、类型转换 ( Any and AnyObject )

    异常处理.类型转换 ( Any and AnyObject ) 1.错误处理 (异常处理) swift 提供第一类错误支持 ,包括在运行时抛出 ,捕获 , 传送和控制可回收错误.在swift中 ,错误 ...

  4. react随笔-1(为什么在react使用jq无法正确渲染组件位置)

    今天心血来潮,打开了sublime想玩玩react,然后大家都知道的先引入一大串 就是在百度静态资源库里找到的. 然后贴html代码 对的,没错,就这么一行,毕竟只是测试嘛 然后js代码 大家一定要注 ...

  5. JavaScript设计模式基础之this、call、apply

    1.this的指向 除去不常用的with和eval,具体应用中this指向大概能分为4种情况分别是 1.作为对象的方法调用. 2.作为普通函数的方法调用. 3.Function.prototype.c ...

  6. 2018 CCPC 女生赛 hdoj6287 口算训练

    题目地址:http://acm.hdu.edu.cn/showproblem.php?pid=6287 Summarize: 1.分解质因数: 2.二分查找函数lower_bound与upper_bo ...

  7. rc-local.service服务启动失败,导致rc.local中的开机启动服务不能启动

    chmod  +x   /etc/rc.d/rc.local 打开/etc/rc.local文件,将启动非后台执行的指令的最后添加 &,以使相关指令后台运行,然后启动服务 systemctl  ...

  8. 用cpp写对拍程序

    #include <bits/stdc++.h> using namespace std; int main() { while(true) { puts(""); p ...

  9. HTML元素的基本特性

    1,Disabled 特性: //Disabled 设置元素不可用: $(this).attr("disabled","disabled") //移除push元 ...

  10. idea没有tomcatserver问题解决

    https://www.cnblogs.com/a8457013/p/7795987.html 在配置tomcate时有时候按照网上说的找不到tomcat Server,不知不觉花了很长时间这时我们在 ...