在C语言中要对一个整数的某一个位进行操作需要用到很多的技巧。这种情况在C++里面通过标准库提供的一个抽象数据类型

bitset得到了改善。

一、标准库bitset类型

1、bitset的作用

  bitset可以看成bit的集合,可以单独的访问集合中的某一位,访问的结果可以作为逻辑的判断的条件。使用bitset的时候可以

不关心这些bit的存储方式,而通过bitset类型提供的一套接口进行操作。

  和string和vector一样,要使用bitset类型,需要报备,如下所示:

#include <bitset>     //包含相关的头文件
#using std::bitset; //声明使用相应的命名空间的名字

2、bitset的定义

  在定义bitset对象的时候,需要指出bitset对象要存储多少bit, 这通过在bitset后面的<>中设置,如下所示:

bitset<n>  bitTset;  //定义一个bitset对象bitTest, n为对象可以存储的bit的位数

Exp:

  bitset<16>  b16Test;   // 定义存储16bit 的 bitset的对象

  bitset<32>  b32Test;   //定义存储32bit的bitset对象

要点:

  bitset<n>  bitObj;  定义的时候n的值必须是一个整型字面值或者用整型字面值初始化的const对象,不能是变量,这一点

必须要注意。

Exp:

int main()
{
bitset<> bitObj(); cout<<bitObj<<endl; return ;
}

执行结果如下所示:

[root@localhost cpp_src]# g++ test.cpp
[root@localhost cpp_src]# ./a.out

3、bitset类的初始化

  bitset提供了多种初始化的方法,如下所示:

  bitset<n>  bitsetObj1;        //初始化bitset对象bitsetObj1的所有的bit为0;

  bitset<n>  bitsetObj2(u);    //利用无符号数整型数 u 来初始化bitset对象bitsetObj2;

bitset<n>  bitsetObj3(string);  //利用string对象初始化bitset对象bitsetObj3;

  bitset<n>  bitsetObj4(string, pos, m);  //利用string对象来初始化从pos位开始的m位

要点:

  利用无符号数u和string对象初始化的时候,可能会存在长度不匹配的情况。而且string对象是一个0、1组成的字符序列。

利用无符号数初始化的时候会先将无符号数转换成二进制格式然后存储到bitset对象中; 利用string对象初始化的时候也会先将string

对象转换为二进制格式然后存储到bitset对象中。

Exp:  初始化bitset对象

int main()
{
string str="";
bitset<> bitObj1; //16个bit全为0
bitset<> bitObj2(0xffff); //32个bit, 高16bit的全是0, 低16bit全是1
bitset<> bitObj3(str); //这个需要在下面细说一下 cout<<sizeof(0xffff)<<endl; return ;
}

  上面的初始化中需要注意string对象的初始化的时候,不能用字符串字面值来初始化。

  对于 :

    string  str="1111_1010_1101"; //这利用 _ 分隔是为了更好的查看

    bitset<8> bitObj3(str);

    bitset<32> bitObj4(str);

  这样初始化后 bitObj3的bit的存放的结果如下所示:  11111010;

bitObj4的结果如下:  00000000000000000000111110101101

  可以发现规律:  当string对象的长度大于bitset对象容量的时候,会将string的后半部分截断, 而保留string对象中下标相对较小

的字符序列。

  当string对象的长度小于bitset对象的容量的时候,就将string对象全部进行转换,并且将string对象进行"平行移动"到bitset对象的

低位部分, 剩下的bitset不足的高位部分用0补齐。

Exp:

int main()
{
string str="";
bitset<> bitObj1;
bitset<> bitObj2(0xffff);
bitset<> bitObj3(str);
bitset<> bitObj4(str); cout<<bitObj3<<endl;
cout<<bitObj4<<endl;
cout<<sizeof(0xffff)<<endl; return ;
}

执行结果为:

[root@localhost cpp_src]# g++ bitset_init.cpp
[root@localhost cpp_src]# ./a.out

Exp:  利用部分的string的字符序列初始化bitset对象

int main()
{
string str="";
bitset<> bitObj1(str,,);
bitset<> bitObj2(str,str.size()-); cout<<bitObj1<<endl;
cout<<bitObj2<<endl;
cout<<str.size()<<endl; return ;
}

执行结果如下所示:

[root@localhost cpp_src]# ./a.out 

下面解释一下这个结果:

  string str("11_1111_1000_0000_1100_1101") ; //

  bitset<32>  bitObj1(str, 5, 4); 这个初始化的意思就是从str对象中的str[5]开始,取4个字符来初始化bitObj1对象。因为取出来的

字符长度小于32bit,因此遵循前面说的string长度小于bitset对象长度的赋值方法。

bitset<32>  bitObj2(str, str.size() - 4); 这里只有两个参数,最后一个参数没有,就表示从str.size()-4开始取字符序列,一直取字

符到string对象的最后一个字符,然后进行转换, 而且转换过程遵循前面说过的关于string对象和bitset对象长度的赋值规则。

  因此得到了上面的运行结果。

  其实后面这两种的初始化主要需要注意的是如何取字符序列,字符序列取出来后,就把取出来的字符序列当成一个新的string对象进行初始

化就可以啦

  

4、bitset类提供的操作

  和其他的标准库类一样,bitset也提供了很多的操作,主要有如下一些:

   bitset<n>  bitObj;

  bitObj.any( );   返回bitObj对象中是否存在已经设为 1 的bit

  bitObj.none();  返回bitObj不存在为1的二进制bit吗, 就是测试是否所有的bit全是0, 全是0时返回true

  bitObj.count(); 返回bitObj中1的位数

  bitObj.size();  返回bitObj对象的bit个数

  bitObj[pos] ;  返回bitObj的pos位置处的二进制bit的值

  bitObj.set();   设置为1;

  bitObj.set(pos); 设置pos处的bit为1;

  bitObj.reset();   设置全为0;

  bitObj.reset(pos);  设置pos位置为0

  bitObj.flip(); 所有的bit取反;

  bitObj.flip(pos); 将pos位置处的bit取反。

  bitObj.to_ulong() 将bitObj转换为unsigned long int类型

os<<b;  将b中的bitset集输出到os流。

int main()
{
bitset<> bitObj(); cout<<bitObj<<endl;
cout<<"The size of bitObj is:"<<bitObj.size()<<endl;
cout<<"bitObj.to_ulong() is:"<<bitObj.to_ulong()<<endl; cout<<"bitObj[3] is:"<<bitObj[]<<endl; if(bitObj.any())
cout<<"bitObj has some bit == 1"<<endl;
else
cout<<"bitObj has no bit == 1"<<endl; if(bitObj.none())
cout<<"bitObj all bit is 0."<<endl;
else
cout<<"bitObj has some bit == 1"<<endl; bitObj.set();
cout<<"after bit.set() bitObj is:"<<bitObj<<endl;
bitObj.reset();
cout<<"after bit.reset(3),the bitObj is:"<<bitObj<<endl; bitObj.reset();
cout<<"after bitObj.reset() bitObj is:"<<bitObj<<endl;
bitObj.set();
cout<<"after bitObj.set(4),bitObj is:"<<bitObj<<endl; cout<<"bitObj.flip() is:"<<bitObj.flip()<<endl; cout<<"bitObj is"<<bitObj<<endl;
bitObj.flip();
cout<<"after bitObj.flip(1),bitObj is:"<<bitObj<<endl; return ;
}

执行结果如下所示:

[root@localhost cpp_src]# ./a.out 

The size of bitObj is:
bitObj.to_ulong() is:
bitObj[] is:
bitObj has some bit ==
bitObj has some bit ==
after bit.set() bitObj is:
after bit.reset(),the bitObj is:
after bitObj.reset() bitObj is:
after bitObj.set(),bitObj is:
bitObj.flip() is:
bitObj is01111
after bitObj.flip(),bitObj is:
[root@localhost cpp_src]#

  通过上面的实例可以知道两个特点需要特别的注意:

   b.set(n)、b.reset(n)、b.flip(n);  这个n是从0开始计算的, 而且是从最低位开始计算的。

 下面还有一个例子:

int main()
{
bitset<> bitObj(); cout<<bitObj<<endl;
cout<<bitObj[]<<endl;
cout<<bitObj[]<<endl;
cout<<bitObj[]<<endl;
cout<<bitObj[]<<endl;
cout<<bitObj[]<<endl; return ;
}

执行结果为:

[root@localhost cpp_src]# g++ test.cpp
[root@localhost cpp_src]# ./a.out

  从上面的结果看,bitset的下标操作为从0开始计数,而且是从最低为开始计数的, 就是说 bitObj[0]  表示的bitset的最低位。

  这次就说的这里,接下来将是与C语言相关的数组和指针的内容啦,待续.........

C++_系列自学课程_第_6_课_bitset集_《C++ Primer 第四版》的更多相关文章

  1. C++_系列自学课程_第_7_课_数组_《C++ Primer 第四版》

    说到数组,大家应该都很熟悉,在C.Pascal.Java等语言中,都有数组的概念.在C++中也提供了对数组的支持.数组简单来说就是一堆相同 数据类型对象的集合. 这里要把握住两个要点: 相同的数据类型 ...

  2. C++_系列自学课程_第_5_课_vector容器_《C++ Primer 第四版》

    再一次遇到 vector 这个单词; 每一次见到这个单词都感觉这个单词非常的 "高大上"; 数字遇到vector马上就可以360度旋转: 当 "电" 遇到vec ...

  3. C++_系列自学课程_第_3_课_变量和基本类型_《C++ Primer 第四版》

    最近复习C++相关内容,决定在这里记录自己复习的过程. 以前写过部分文字,但是没有坚持连续写,因此学完后 基本又忘光啦,主要是没有实践,这一次决定自学完后,在这里在复习一遍增强自己的记忆和理解程度. ...

  4. C++_系列自学课程_第_12_课_结构体

    #include <iostream> #include <string> using namespace std; struct CDAccount { double bal ...

  5. C++_系列自学课程_第_12_课_语句_《C++ Primer 第四版》

    前面的文章说完了表达式和类型转换的部分内容,在我参考的书里面,接下来讨论的是各种语句,包括:顺序语句.声明语句.复合语句(块语句).语句作用域 .if语句.while语句.for语句.do...whi ...

  6. C++_系列自学课程_第_11_课_类型转换_《C++ Primer 第四版》

    上次说了关于表达式的一些内容,说到还有一些关于数据类型转换的内容,今天我们接着八一八C++中的数据类型转换. 一.隐式类型转换 在表达式中,有些操作符可以对多种类型的操作数进行操作, 例如 + 操作符 ...

  7. C++_系列自学课程_第_10_课_表达式_《C++ Primer 第四版》

    程序设计语言中大部分程序都在进行表达式的求值操作, 例如求两个数的和,求一个表达式的逻辑结果,或者通过输入输出表达式语句进行输入和输出. 这里我们对表达式进行讨论. 一.表达式 1.表达式 表达式由一 ...

  8. C++_系列自学课程_第_9_课_C语言风格字符串_《C++ Primer 第四版》

    前面说了写关于数组和指针的内容,这次在这里讨论一下字符串,讨论一下C语言风格的字符串. 在C语言里面我们利用字符数组来对字符串进行处理, 在C++里面我们前面说过一种类类型string可以对字符串进行 ...

  9. C++_系列自学课程_第_8_课_指针和引用_《C++ Primer 第四版》

    C语言最富有迷幻色彩的部分当属指针部分,无论是指针的定义还是指针的意义都可算是C语言中最复杂的内容.指针不但提供给了程序员直接操作硬件部分的操作接口,还提供给了程序员更多灵活的用法.C++继承这一高效 ...

随机推荐

  1. 在C#代码中应用Log4Net(四)在Winform和Web中捕获全局异常

    毕竟人不是神,谁写的程序都会有bug,有了bug不可怕,可怕的是出错了,你却不知道错误在哪里.所以我们需要将应用程序中抛出的所有异常都记录起来,不然出了错,找问题就能要了你的命.下面我们主要讨论的是如 ...

  2. .NET陷阱之六:从枚举值持久化带来大量空间消耗谈起

    好长时间没有写博文了,今天继续. 这次跟大家分享的内容起因于对一个枚举值列表的序列化,下面简化后的代码即能重现.为了明确起见,我显式指定了枚举的基础类型. // 定义一个枚举类型. public en ...

  3. 【原】通过JS打开IE新tab(非Window)的解决方案

    近日项目里遇到限定在IE的tab窗口里打开新窗口的需求,结合网上的资源和亲自实践,总结以下比较可行的解决方案. 1.首先必须保证IE的设置正确.打开IE的Internet options ->G ...

  4. 【PRINCE2是什么】PRINCE2认证之七大原则(4)

    我们先来回顾一下,PRINCE2七大原则分别是持续的业务验证,经验学习,角色与责任,按阶段管理,例外管理,关注产品,剪裁. 第四个原则:按阶段管理. 阶段管理其实是给高层提供了项目生命周期中相对应的控 ...

  5. ASP.NET Core 1.0 静态文件、路由、自定义中间件、身份验证简介

    概述 ASP.NET Core 1.0是ASP.NET的一个重要的重新设计. 例如,在ASP.NET Core中,使用Middleware编写请求管道. ASP.NET Core中间件对HttpCon ...

  6. java.io中的System.in、 System.out和System.err

    System.in, System.out, System.err这3个流同样是常见的数据来源和数据流目的地.使用最多的可能是在控制台程序里利用System.out将输出打印到控制台上. JVM启动的 ...

  7. C#单例模式详解

    C#要实现单例模式必须要有以下三点: 声明私有静态成员.私有化构造函数.静态函数返回实例. private static GameManager s_GameManager=null; private ...

  8. css 权重

    <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...

  9. Web APi之手动实现JSONP或安装配置Cors跨域(七)

    前言 照理来说本节也应该讲Web API原理,目前已经探讨完了比较底层的Web API消息处理管道以及Web Host寄宿管道,接下来应该要触及控制器.Action方法,以及过滤器.模型绑定等等,想想 ...

  10. 如何用C#代码判断一个类的类型

     var s = "";  s.GetType().IsClass; 来自为知笔记(Wiz)