【C】 16_位运算符分析

逻辑浮尘
• 阅读 2403

位运算符分析

  • C 语言中的位运算符

位运算符直接对 bit 位进行操作,其效率最高!

【C】 16_位运算符分析

  • 左移和右移注意点

    • 左操作数必须为整数类型

      • char 和 short 被隐式转换为 int 后进行移位操作
    • 右操作数的范围为:[0, 31] (标准C语言规范,否则将得到不确定的结果,具体由编译器厂商决定)
    • 左移运算符 << 将运算数的二进制左移

      • 规则: 高位丢弃,低位补 0
    • 右移运算符 >> 将运算数的二进制右移

      • 规则: 高位补符号位,低位丢弃

0x1 >> 2 + 3 的值会是什么? 原作者的本意究竟想表达什么??

实例分析: 位运算符初探

#include <stdio.h>

int main()
{
    printf("%d\n", 3 << 2);
    printf("%d\n", 3 >> 1);
    printf("%d\n", -1 >> 1);
    printf("%d\n", 0x01 << 2 + 3);
    
    return 0;
}
输出:
12
1
-1
32
分析:
3 << 2 ==> 11 << 2 ==> 1100 ==> 12
3 >> 1 ==> 11 >> 1 ==> 01   ==> 1
-1 >> 1 ==> 1111 1111 >> 1==> 1111 1111 >> -1
0x01 << 2 + 3 ==> 0x01 << (2 + 3) << 0x01 << 5 ==> 32

小贴士

  • 防错准则:

    • 避免位运算符,逻辑运算符和数学运算符同时出现在一个表达式中
    • 当位运算符,逻辑运算符和数学运算符需要同时参与运算时,尽量使用括号 () 来表达计算次序

小技巧:

  • 左移 n 位相当于乘以 2 的 n 次方,但效率比数学运算符高
  • 右移 n 位相当于处以 2 的 n 次方,但效率比数学运算符高

编程实验: 交换两个整形变量的值

#include <stdio.h>

#define SWAP1(a, b)   \
{                     \
    int t = a;        \
    a = b;            \
    b = t;            \
}

#define SWAP2(a, b)   \
{                     \
    a = a + b;        \
    b = a - b;        \
    a = a - b;        \
}

#define SWAP3(a, b)   \
{                     \
    a = a ^ b;        \
    b = a ^ b;        \
    a = a ^ b;        \
}

int main()
{
    int a = 1;
    int b = 2;
    
    printf("a = %d\n", a);
    printf("b = %d\n", b);
    
    SWAP3(a, b);
    
    printf("a = %d\n", a);
    printf("b = %d\n", b);

    return 0;
}
输出:
a = 1
b = 2
a = 2
b = 1  

分析:
SWAP1 基础用法
SWAP2 基础用法,弊端,a + b 可能发生数据溢出,不安全
SWAP3 a = a ^ b; b = a ^ b;  ==> b = a ^ b ^ b ==> a ^ (b ^ b) ==> a ^ 0 ==> a 
      a = a ^ b; ==> a ^ b ^ b ==> a ^ b' ^ a == > b'
      

位运算与逻辑运算

  • 位运算与逻辑运算不同:

    • 位运算没有短路规则,每个操作数都参与运算
    • 位运算的结果为整数,而不是 0 或 1
    • 位运算优先级高于逻辑运算优先级

实例分析:混淆的判断条件

#include <stdio.h>

void func_1()
{
    int i = 0;
    int j = 0;
    int k = 0;
    
    if( ++i | ++j & ++k)
    {
        printf("func_1() : Runc here...\n");
    }
}

void func_2()
{
    int i = 0;
    int j = 0;
    int k = 0;
    
    if( ++i || ++j && ++k)
    {
        printf("func_2() : Runc here...\n");
    }
}

int main()
{
    func_1();
    func_2();
}
输出:
func_1() : Runc here...
func_2() : Runc here...

分析:
func_1(); func_2(); 运行表现一致,但本质发生了不同操作,留下隐患。
func_1() 运行后, i = 1, j = 0, k = 0
func_2() 运行后, i = 1, j = 1, k = 1

小结

  • 位运算符只能用于整数类型
  • 左移和右移运算符的右操作数范围必须为[0, 31]
  • 位运算符没有短路规则,所有操作数均会求值
  • 位运算的效率高于四则运算和逻辑运算
  • 运算优先级: 四则运算 > 位运算 > 逻辑运算

以上内容参考狄泰软件学院系列课程,请大家保护原创!

点赞
收藏
评论区
推荐文章
Wesley13 Wesley13
3年前
java中的7个位运算运算符
位运算指的是针对整数的二进制进行的位移操作。位运算提供比算术运算更高的效率,但是位运算的代码可读性较差,建议所有使用位运算的地方写上注释。Java中提供7个位运算符用于位运算。左移(<<)左移运算是将操作数二进制值逐位左移若干位,左移过程中符号位不变,高位溢出则舍弃,低位则补0。范例结果范例结果00000001<<
Wesley13 Wesley13
3年前
java位运算大全
位运算因为是CPU直接支持的操作指令,也是基于二进制的操作,所以具有相当高的效率,在一些场合,合理应用位运算将具有很高的性能。通常在一些加密算法,图型算法中都会使用到位运算。Java位运算符位 运 算 符 用 来 对 二 进制 位 进 行 操 作 ,Java中提 供 了 如 下所 示 的 位 运 算符:位 运 算 符 (>>,<<
Stella981 Stella981
3年前
ASMSupport教程4.8 生成逻辑运算操作
<p在java中有以下逻辑运算符:</p<ul<li&amp;&amp;:条件与</li<li||:条件或</li<li&amp;:布尔型的逻辑与</li<li|:布尔型的逻辑或</li<li^:布尔型的逻辑异或</li<li!:非操作</li</ul<p那么接下来我们将些段例子
Stella981 Stella981
3年前
Python运算符大全
  一、Python的算术运算  Python的算术运算符与C语言类似,略有不同。包括加()、减()、乘(\)、除(/)、取余(%)、按位或(|)、按位与(&)、按位求补(~)、左移位(<<)、右移位()、单目求反()、幂运算(\\)、整除运算(//)、增强运算、增强矩阵乘法(@)。  增强运算是将算术运算符或逻辑运算符放到等号的左
Wesley13 Wesley13
3年前
C语言位运算
位运算应用口诀清零取反要用与,某位置一可用或若要取反和交换,轻轻松松用异或移位运算要点1它们都是双目运算符,两个运算分量都是整形,结果也是整形。        2"<<"左移:右边空出的位上补0,左边的位将从字头挤掉,其值相当于乘2。       3""右移:右边的位被挤掉。对于左边移出的空位,如果是正数则空
Stella981 Stella981
3年前
20180109Java位运算
一,Java位运算1.表示方法:  在Java语言中,二进制数使用补码表示,最高位为符号位,正数的符号位为0,负数为1。补码的表示需要满足如下要求。 (1)正数的最高位为0,其余各位代表数值本身(二进制数)。 (2)对于负数,通过对该数绝对值的补码按位取反,再对整个数加1。 2.位运算符位运算表达式由
可莉 可莉
3年前
20180109Java位运算
一,Java位运算1.表示方法:  在Java语言中,二进制数使用补码表示,最高位为符号位,正数的符号位为0,负数为1。补码的表示需要满足如下要求。 (1)正数的最高位为0,其余各位代表数值本身(二进制数)。 (2)对于负数,通过对该数绝对值的补码按位取反,再对整个数加1。 2.位运算符位运算表达式由
Wesley13 Wesley13
3年前
C#位运算
在C中可以对整型运算对象按位进行逻辑运算,按位进行逻辑运算的意义是:依次取被运算对象的每个位,进行逻辑运算,每个位的逻辑运算结果是结果值的每个位,C支持的位逻辑运算符如下表。!(https://oscimg.oschina.net/oscnet/e3ff3ca0d8190d7cf6a5c8269feaab32004.jpg)1、位逻辑非运算
Wesley13 Wesley13
3年前
Java运算符
Java中的运算符分类:算术运算符二元运算符,,\,/,%一元运算符,赋值运算符\扩展运算符,,\,/关系运算符\,<,,<,,! instanceof逻辑运算符&&,||,!,^位运算符&,|,^,~,,<<,条件运算符
Wesley13 Wesley13
3年前
Java位运算原理及使用讲解
前言日常开发中位运算不是很常用,但是巧妙的使用位运算可以大量减少运行开销,优化算法。举个例子,翻转操作比较常见,比如初始值为1,操作一次变为0,再操作一次变为1。可能的做法是使用三木运算符,判断原始值为1还是0,如果是1,设置为0,否则设置为0.但是使用位运算,不用判断原始值,直接改变值就可以:1^num//num为原始值当然,一条语句可能
小万哥 小万哥
1年前
Python 运算符
运算符用于对变量和值执行操作。在下面的示例中,我们使用运算符将两个值相加:pythonprint(105)Python将运算符分为以下几组:算术运算符赋值运算符比较运算符逻辑运算符身份运算符成员运算符位运算符算术运算符算术运算符用于对数字值执行常见的数