目录

BitVector32结构体位于System.Collections.Specialized命名空间内,相对.NET中另外一个位容器BitArray,他的优点是速度快,占用空间小,并可以存储小数字。他内部用一个32位的整数来存储数据,因此只能存储32位的比特数据。

返回目录

温习位操作

在看BitVector32前,温习一下简单的位操作还是很有必要的,常见的位操作无非就是与(AND),或(OR),非(NOT)。一般情况下,将一位设置成1就是把这个为或1的操作,将一位设置成0就是把这个位与0的操作。同时或0,与1结果不变。

这样的话,比如一个8位的数据:

0000 1111

我们想把第二个0设置成1,那么把他和下面这个数进行或操作:

0100 0000

结果就是

0100 1111

第二位变成了1

还是这个数

0000 1111

如果想把最后一个1设置成0,那么把他和下面这个数进行与操作

1111 1110

结果就是

0000 1110

那么我们可以这样总结一下:

想要操作一个位,我们把其他位设置成0,把这个位设置成1,这个数就是所谓的位掩码(也称位屏蔽,MSDN里用的是位屏蔽)。

那么如果想打开一个位(就是把这个位设置成1):源数据 = 源数据 OR 位掩码。

如果想关掉一个位(就是把这个位设置成0):源数据 = 源数据 AND 位掩码取反。

(位掩码取反就是非(NOT)操作:0变成1,1变成0)。

返回目录

BitVector32的位操作

了解了基本位操作BitVector32的理解就会简单多了。

首先BitVector32本质上用一个32位的数来表示数据,那么初始化BitVector32结构时必须制定一个最初值,用户可以传入一个int或者另一个已经存在的BitVector32来构造一个新的BitVector32.

BitVector32的Data属性返回一个int用来表示内部数据,如果用来显示BitVector32的内容,这个Data是没有意义的,因为他是十进制化的结果,这时候用BitVector32的ToString方法就可以返回有用的文字说明。

BitVector32 bits =newBitVector32(0xF); //初始化BitVector32:设置低4位为1  0x 00 00 00 00 00 00 00 0F Console.WriteLine(bits.Data); //(十六进制)0xF 等于 (二进制)1111 等于 (十进制)15 所以输出15 Console.WriteLine(bits.ToString()); //输出:BitVector32{00000000000000000000000000001111} (看得出来:后四位是1)

接下来就是最重要的位操作了。

BitVector32结构体提供索引器(Indexer)可以直接通过bool对象操作BitVector32结构,索引器参数是int,这个int可不是指第几位的意思(BitArray中的索引器是第几位),而是需要一个位掩码(位屏蔽),通过这个BitVector32通过这个位掩码来操作内部比特位,来看看Reflector下BitVector32索引器的源码

structBitVector32 { //data 就是 BitVector32的Data属性的对应字段 publicboolthis[int bit]     { get         { return ((this.data & bit) == ((ulong)bit)); //通过数据和掩码的与操作,来将其他位清0,保留掩码的设置位 //如果操作后的结果等于掩码,很显然这是操作位就是1,返回true //否则的话,返回false         } set         { if (value)             { this.data |= (uint)bit; //数据 = 数据 OR 掩码,将指定位设置成1             } else             { this.data &= (uint)~bit; //数据 = 数据 AND 掩码取反,将指定位设置成0 //C# 取反位操作运算符:~             }         }     }
}

上面代码我加了注释,可以看到,BitVector32的位操作就是利用普通位掩码的操作。

好了那么用BitVector32索引器操作其实就是定义好位掩码,接着取回信息或者赋值就可以了。

//注意using System.Collections.Specialized;
int mask1 =; //掩码代表最后一位:二进制表示:0...0001 int mask2 =; //掩码代表倒数第三位:二进制表示:0...0000100
BitVector32 bits =newBitVector32(-); //-1补码:1...1111 //设置BitVector32全部为1 Console.WriteLine(bits);
Console.WriteLine("设置最后一位和倒数第三位位0"); bits[mask1] = bits[mask2] =false; Console.WriteLine(bits);
Console.WriteLine("设置倒数第三位为1"); bits[mask2] =true; Console.WriteLine(bits);

输出:

BitVector32{11111111111111111111111111111111} 设置最后一位和倒数第三位位0 BitVector32{11111111111111111111111111111010} 设置倒数第三位为1 BitVector32{11111111111111111111111111111110}

返回目录

CreateMask方法

BitVector还有一个CreateMask方法,他的作用就是方便用户定义位掩码。

CreateMask(无参数):返回第一个位(最低位)的位掩码,那么就是0…00001,十进制的话就是1

CreareMask(int):首先判断一直位掩码的合法性,并返回向左移1位的结果。

参考CreateMask的源代码:

publicstaticint CreateMask(int previous) { if (previous ==)     { return;     } if (previous ==-) //这个数等于Int32.MinValue 二进制是100...0,无法再向左移位,因此异常     { thrownewInvalidOperationException(SR.GetString("BitVectorFull")); //抛出异常     } return (previous <<); //向左移1位 }

当然如果你掌握了上述位操作和位掩码的概念,我个人感觉CreateMask有点多此一举,我们就直接用MSDN的例子(我改了改注释)

// 初始化BitVector32:全部0 BitVector32 myBV =newBitVector32();
// 创建最低位的掩码,然后陆续创建倒数第二位,倒数第三位……倒数第五位的掩码 int myBit1 =BitVector32.CreateMask(); int myBit2 =BitVector32.CreateMask(myBit1); int myBit3 =BitVector32.CreateMask(myBit2); int myBit4 =BitVector32.CreateMask(myBit3); int myBit5 =BitVector32.CreateMask(myBit4); Console.WriteLine("最初值: \t{0}", myBV.ToString());
// 设置倒数第三位为1 myBV[myBit3] =true; Console.WriteLine("myBit3 = TRUE \t{0}", myBV.ToString());
// 将掩码加起来,同时设置两个位 myBV[myBit4 + myBit5] =true; Console.WriteLine("myBit4 + myBit5 = TRUE \t{0}", myBV.ToString()); myBV[myBit1 | myBit2] =true; Console.WriteLine("myBit1 | myBit2 = TRUE \t{0}", myBV.ToString());

输出:

最初值:                 BitVector32{00000000000000000000000000000000} myBit3 = TRUE           BitVector32{00000000000000000000000000000100} myBit4 + myBit5 = TRUE  BitVector32{00000000000000000000000000011100} myBit1 | myBit2 = TRUE  BitVector32{00000000000000000000000000011111}

如果你手动输出上面例子中的myBit1-5的值,其实就是我们上面讲到的倒数一到五位的掩码值。

位操作:BitVector32结构 z的更多相关文章

  1. NESPER的大体结构 z

    NEsper从内容上分为两块,NEsper的核心NEsper.dll和NEsper.IO.dll. (1)NEsper的核心包包含了EPL语法解析引擎,事件监听机制,事件处理等核心模块. (2)NEs ...

  2. C#编程(五十七)----------位数组

    位数组 如果需要处理很多位,就可以使用BitArray类和BitVector32.BitArray位于命名空间System.Collections中. BitVector32位于命名空间System. ...

  3. C#高级编程五十七天----位数组

    位数组 假设须要处理非常多位,就能够使用BitArray类和BitVector32.BitArray位于命名空间System.Collections中. BitVector32位于命名空间System ...

  4. C#学习总结之集合

    一.集合接口和类型 命名空间:  集合类型  命名空间  一般集合 System.Collections   泛型集合 System.Collections.Generic   特定类型集合 Syst ...

  5. [总结] JDBC数据库操作

    1.加载驱动--告诉驱动管理将使用哪一个数据库的驱动包. class.forName("com.mysql.jdbc.Driver"); 2.操作JDBC ADI完成数据库动作 D ...

  6. C#高级编程(第8版)

    http://spu.jd.com/11328513.html 第1章 .NET体系结构1.1 C#与.NET的关系1.2 公共语言运行库1.2.1 平台无关性1.2.2 提高性能1.2.3 语言的互 ...

  7. C#集合基础与运用

    C#集合基础与运用   C#集合基础与运用 1. 集合接口与集合类型............................................... 1 (1) 集合的命名空间..... ...

  8. s14 第5天 时间模块 随机模块 String模块 shutil模块(文件操作) 文件压缩(zipfile和tarfile)shelve模块 XML模块 ConfigParser配置文件操作模块 hashlib散列模块 Subprocess模块(调用shell) logging模块 正则表达式模块 r字符串和转译

    时间模块 time datatime time.clock(2.7) time.process_time(3.3) 测量处理器运算时间,不包括sleep时间 time.altzone 返回与UTC时间 ...

  9. C#高级编程第9版 阅读笔记(一)

    一.前言 C# 简洁.类型安全的面向对象的语言. .NET是一种在windows平台上编程的架构——一种API. C#是一种从头开始设计的用于.NET的语言,他可以利用.NET Framework及其 ...

随机推荐

  1. EXTJS 4.2 资料 控件之radiogroup 的用法

    最近在EXTJS4.2开发项目,radiogroup的用法,主要是和grid之间的编辑功能:看了好多资料都不对,特此在这里备注记录 代码如下, 1.这是一段Win窗体上的两个单选按钮,设置单选按钮都是 ...

  2. 1045: [HAOI2008] 糖果传递 - BZOJ

    Description 有n个小朋友坐成一圈,每人有ai个糖果.每人只能给左右两人传递糖果.每人每次传递一个糖果代价为1.Input 小朋友个数n 下面n行 aiOutput 求使所有人获得均等糖果的 ...

  3. 同一台Windows机器中启动多个Memcached服务

    同一台Windows机器中启动多个Memcached服务 这就需要在一台机器上启动多个Memcached服务了. 假设Memcached在如下目录:C:\memcached\memcached.exe ...

  4. Codeforces Round #260 (Div. 1) C. Civilization 树的中心+并查集

    题目链接: 题目 C. Civilization time limit per test1 second memory limit per test256 megabytes inputstandar ...

  5. 基于局部敏感哈希的协同过滤算法之simHash算法

    搜集了快一个月的资料,虽然不完全懂,但还是先慢慢写着吧,说不定就有思路了呢. 开源的最大好处是会让作者对脏乱臭的代码有羞耻感. 当一个做推荐系统的部门开始重视[数据清理,数据标柱,效果评测,数据统计, ...

  6. django 的mysql数据配置

    原地址:http://blog.csdn.net/gamesofsailing/article/details/21465327 在成功安装python-mysql后,开始配置django的mysql ...

  7. 注入攻击-SQL注入和代码注入

    注入攻击 OWASP将注入攻击和跨站脚本攻击(XSS)列入网络应用程序十大常见安全风险.实际上,它们会一起出现,因为 XSS 攻击依赖于注入攻击的成功.虽然这是最明显的组合关系,但是注入攻击带来的不仅 ...

  8. hdu 2516 取石子游戏 博弈论

    很显然的nim游戏的变形,很好找规律 先手败:2,3,5,8,13…… 其他先手胜.即满足菲波拉数列. 代码如下: #include<iostream> #include<stdio ...

  9. live555源码研究(三)------UsageEnvironment类

    一.UsageEnvironment类作用 1,不使用的时候回收当前的使用环境. 2,对返回结果消息和错误消息的维护. 二.类UsageEnvironment继承关系图

  10. the service mysql56 was not found in the Windows services的解决办法

    mysql无法启动,无法改变状态-CSDN论坛-CSDN.NET-中国最大的IT技术社区 http://bbs.csdn.net/topics/390943788   具体描述: 关闭,重启mysql ...