C/C++的位运算符操作

 C/C++支持比较低阶的位运算,在是众人皆知的了。每本C/C++的教科书都会说到这部分的内容,不过都很简略,我想会有很多人不知道位运算用在什么地方。这个帖子就简略说说位运算的用处,更进一步的用法要大家自己去体会。而主要说的是操作标志值方面。
    考虑一个事物、一个系统、或者一个程序可能会出现一种或者几种状态。为了在不同的状态下,作出不同的行为,你可以设立一些标志值,再根据标志值来做判断。比如C++的文件流,你就可以设定一些标志值,ios::app,  ios::ate,  ios::binary,  ios::in,  ios::out,  ios::trunc,并且可以将它用|组合起来创建一个恰当的文件流。你可能会将这些标志值定义为bool类型,不过这样要是设置的标志值一多,就会很浪费空间。

而假如定义一个整型数值,unsigned  int  flags;  在现在的系统,flags应该是32位,  用1,2,3....32将位进行编号,我们可以进行这样的判断,  当位1取1时,表示用读方式打开文件,当位2取1时,表示用写方式打开文件,当位3取1时,用二进制方式打开文件....因为flags有32位,就可以设置32个不同的状态值,也相当于32个bool类型。这样一方面省了空间,  另一方面也多了个好处,就是如前面所说的,可以将标志值组合起来。
//>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>

好啦,上面有点不清不楚的。下面看看到底怎么操作这些标志值。
设想C++的类ios这样定义,  其实没有这个类,只有ios_basic类,typedef  basic_ios<char>  ios;

class  ios
{
public:
        enum        app  0x0001,  ate  0x0002,  binary  0x0004,
                in  0x0008,    out  0x0010,  trunc  0x0020  };
        ....
private:
        unsigned  int  flags;
};

注意上面enum语句中,每一个数值只有1位是1,其余是0,这个很重要,你可以将它化成2进制看看。

现在将flags相应的位设置为1, 可以这样做 flags |= app。这个等于flags flags app, 为什么呢? app只有1位是1,其余是0,因为0 0, 0, 这样0对应的位是不变的。而1 1, 1, 1对应的位不论原来是什么状态,都一定为1。如果想要将几个位都设置为1,可以这样做 flags |= (app ate binary)。因为每个enum常数各有一位为1, 与运算之后就有3位为1,就如上面的分析,就可以将那3位都设置为1, 其余位不变。这个就是标志可以组合起来用的原因。也可以用+组合起来,原因在于(下面的数字是2进制)0001 0010 0100 0111 跟与运算结果一样。不过不提倡用+, 考虑(app ate binary)要是我不小心写多了个标志值,(app ate ate binary)结果还是正确的,如果用+的话,就会产生进位,结果就会错误。通常我们不知道原先已经组合了多少个标志值了,用或运算会安全。

现在将flags对应的位设置为0,  可以这样做  flags  &=  ~app。相当于  flags  flags  (~app).  app取反之后,只有1位是0,其余是1,做与运算之后,1对应的位并不会改变,0对应的为不管原来是1是0,都肯定为0,这样就将对应的位设置了0。同样同时设置几个标志位可以这样做,flags  &=  ~(app  ate  binary)。

现在将flags对应的位,如果是1就变成0,如果是0就变成1,可以这样做  flags  ^=  app。同时设置几个标志位可以写成  flags  ^=  (app  ate  binary)。不再做分析了,不然就太罗嗦了。不过也给大家一个例子,你查查Ascii表,会发现对应的大小写字母是相差倒数第6位,可以用这样的函数统一的将大写变成小写,小写变成大写。
void  xchgUppLow(string&  letters)
{
                const  unsigned  int  mask  (1<<5);

                for  (size_t  i=0;  i<letters.length();  i++)
                                letters[i]  ^=  mask;
}
前提是输入的string一定要全是字母,  而要想是操作字母,可以在原来基础上加个判断。

     好啦,上面已经可以设置flags的对应位值了,要是判断呢?可以这样写 if (flags app) 这样可以判断对应的位值是否为1, 因为C/C++语言中非0就真。app只有一位是1,其余是0,如果, flags的对应位也是0,在与操作下就得到结果0,反之非0,这样就可以判断标志位了。

//>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>
上面关于标志值的操作就介绍完毕。其实在C++中已经有了个bitset了,没有必要去自己进行低阶的位运算,上面的四个操作在bitset中分别叫做set,  reset,  flip,  test。不过在C中,这样的代码还很常见,  反正知道多点也没有坏处。

用  windows  API  编程,你也经常会碰到这样的标志值,要互相组合,可以用|,  也可以用+(只是建议用|,理由上面说了).  它的标志值也是这样定义的,不过用#define
#define  WS_BORDER        0x0001
#define  WS_CAPTION        0x0002
......
当初我就是想不明白为什么可以用|或者用+来组合,现在知道了。

(注:上面出现的数字是我自己作的,到底实际怎么定义其实没有关系,只要保证只有一位是1,其余是0就可以的了.  因为编程的时候用的是常量值,没有人这样笨去直接用数值的)

//>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>
其实,位运算还有很多用处。比如移位相当于乘除2的幂数(不过通常编译器也将乘除2的幂数优化成汇编的移位指令,所以没有必要不要这样卖弄了。汇编的移位指令有两组,分别针对有符号和无符号的,  我猜想在C/C++的同一移位运算针对有符号整数和无符号整数的不同,会根据情况编译成不同的汇编移位指令,不过没有去证实),  其实移位更用得多的地方是去构造一个掩码,  比如上面的mask  (1<<5);

还有&运算,有时候可以用来求余数。比如  value  (1<<4  1)  这相当于将value的高位全变成0了,效果等于  value  8. 

还有值得一提的是^运算,它有个很特殊的性质。比如  ^=  B,  变成另一个数,跟着再执行A  ^=  B,又变回原来的数了,不信你可以列真值表或者化简逻辑式看看。就因为这个性质,^有很多用途。比如加密,你将原文看成A,  用同一个B异或一次,就相当于加密,跟着在用B异或一次,相当于解密。不过这样是很容易破解就是了。要是一个B不够,还可以加个C,  比如A  ^=  B,  ^=  C,  ^=  C,  ^=  B,  恢复原状。

下面一个小程序,用异或交换两个数字。
int  3;
int  4;

^=  y;
^=  x;
^=  y;

其实和止交换数字,连交换对象也可以的
template  <typename  T>
void  swap(T&  obj1,  T&  obj2)
{
                const  int  sizeOfObj  sizeof(T);
                char*  pt1  (char*)&obj1;
                char*  pt2  (char*)&obj2;

                for  (size_t  i=0;  i<sizeOfObj;  i++)
                {
                                pt1[i]  ^=  pt2[i];
                                pt2[i]  ^=  pt1[i];
                                pt1[i]  ^=  pt2[i];
                }
}

还有异或操作还可以用在图象的光栅操作。我们知道,颜色也是用二进制来表示的,对颜色进行不同的位运算,就可以得到不同的光栅。因为异或的特殊性质,我们用异或操作的光栅画了副图,跟着再在原来的地方画一次,那副图就刷除了。这样可以用来显示动画而不用保存原来的画像信息。以前我写过个双人的贪食蛇,就用了异或光栅。因为背景色是白色的,也就是全1,作A  A,  所以用画刷画一次是画了设定的颜色,再画一次就恢复。最有趣的是两蛇相交的时候,颜色也会作异或叠加,产生一种新的颜色了,离开的时候也会自动恢复。
//>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>
好啦,够长了,就停止吧。在最后再给大家一段代码,是用来看看对象在内存中的位值的。可以看看。
string  bitsOfUChar(unsigned  char  c)
{
                const  int  numOfBitsInUChar  8;
                unsigned  int  mask  (1<<7);
                string  result(8,  '0');

                for  (size_t  i=0;  i<numOfBitsInUChar;  i++)
                {
                                if  mask  c)
                                                result[i]  '1';

                                mask  >>=  1;
                }

                return  result;
}

template  <typename  T>
string  bitsInMemory(const  T&  obj)
{
                int  sizeOfObj  sizeof(obj);
                unsigned  char*  pt  (unsigned  char*)&obj;
                string  result;

                for  (size_t  i=0;  i<sizeOfObj;  i++)
                {
                                result  +=  bitsOfUChar(pt[i]);
                                result  +=  ';
                }

                return  result;
}

比如bitsInMemory(12),会输出00001100  00000000  00000000  00000000,  我就知道我自己的机器是小尾顺序的了。
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值