【C语言】 操作符(上): -- 算数 -- 移位 -- 位操作符 -- 赋值 -- 单目 -- 关系 -- 逻辑操作符2

简介: 【C语言】 操作符(上): -- 算数 -- 移位 -- 位操作符 -- 赋值 -- 单目 -- 关系 -- 逻辑操作符2

2.2 左移操作符

移位规则:左边抛弃,右边补0

我们来看一段代码:

#include <stdio.h>
int main()
{
    int a = 3;
    //3的二进制序列:00000000 00000000 00000000 00000011
    int b = a << 1;
    printf("%d\n", b);
    printf("%d\n", a);
    return 0;
}

效果展示:

Q:为什么 a << 1 变成了6呢?

A:我们使用图解:

3是正整数,原码就是补码。

这里应该注意的是 a ,对 a 进行了左移并将值赋给 b ,a 本身是不发生变化的。a << 1 这个表达式结果是 6 ,但是 a 是不变的。并且舍弃的确实是符号位,舍弃后紧挨着的那个数字充当符号位。

这次我们来看看 -3

#include <stdio.h>
int main()
{
    int a = -3;
    //原码二进制序列:10000000 00000000 00000000 00000011
    //反码二进制序列:11111111 11111111 11111111 11111100
    //补码二进制序列:11111111 11111111 11111111 11111101
    int b = a << 1;
    //补码二进制序列:11111111 11111111 11111111 11111010
    //反码二进制序列:10000000 00000000 00000000 00000101
    //原码二进制序列:10000000 00000000 00000000 00000110
    printf("%d\n", b);//打印是以原码来打印
    printf("%d\n", a);
    return 0;
}

效果展示:

我们继续画图分析:


注:负数的补码转换成反码有两种方式:-1,取反;取反+1。两种方式都是可以的。

2.3 右移操作符

移位规则:

右移运算分两种:
      1.逻辑移位

       左边用 0 补充,右边丢弃

       2.算术移位

       左边用原来值的符号位填充,右边丢弃
注:两种都是移几位相应补几位。

我们继续以代码来深入了解:

#include <stdio.h>
int main()
{
  int a = -5;
  int b = a >> 1;
  printf("%d\n", b);
  printf("%d\n", a);
  return 0;
}

效果展示:


Q:为什么 b 变成了-3 呢?

我们来分析一下,有了左移的经验,这里就会容易不少。

这里我们也可以了解到 vs 2019 编译器做的是算数右移。

如果这里是逻辑右移的话符号位就是 0 ,得出来的就是 3。

警告:对于移位运算符,不要移动负位数,这个是标准未定义的。

3、位操作符

位操作符有:

& //按位与

| //按位或

^ //按位异或

注:它们的操作数必须是整数。

这里的位:是二进制位。

3.1 按位与

我们来看一段代码:

#include <stdio.h>
int main()
{
  int a = 3;
  int b = -5;
  int c = a & b;//按位与:对应的二进制位上有0为0,双1为1
  //00000000 00000000 00000000 00000011 -- 3的补码
  //10000000 00000000 00000000 00000101 -- -5的原码
  //11111111 11111111 11111111 11111010 -- -5的反码
  //11111111 11111111 11111111 11111011 -- -5的补码
  //00000000 00000000 00000000 00000011 -- 3的补码
  //00000000 00000000 00000000 00000011 -- &的结果,这里的补码和3的补码是一样的,因此打印的就是3
  printf("%d\n", c);
  return 0;
}

按位与的规则是:对应的二进制位上有0为0,同是1为1。


按照这个规则我们来分析一下 c 的补码是:


11111111 11111111 11111111 11111011 -- -5的补码


00000000 00000000 00000000 00000011 -- 3的补码


00000000 00000000 00000000 00000011 -- &的结果,符号位是 0 (表示正数),原反补码是相同的。


这里的补码和3的补码是一样的,因此打印的就是3。


效果展示:



3.2 按位或

按位或的规则是:对应的二进制位上有1为1,同是0为0

我们来看一段代码:

#include <stdio.h>
int main()
{
  int a = 3;
  int b = -5;
  int c = a | b;//按位或:对应的二进制位上有1为1,双0为0
  //00000000 00000000 00000000 00000011 -- 3的补码
  //10000000 00000000 00000000 00000101 -- -5的原码
  //11111111 11111111 11111111 11111010 -- -5的反码
  //11111111 11111111 11111111 11111011 -- -5的补码
  //00000000 00000000 00000000 00000011 -- 3的补码
  //11111111 11111111 11111111 11111011 -- |的结果
  printf("%d\n", c);
  return 0;
}

按照这个规则我们来分析一下 c 的补码是:


11111111 11111111 11111111 11111011 -- -5的补码


00000000 00000000 00000000 00000011 -- 3的补码


11111111 11111111 11111111 11111011 -- |的结果(这里的补码和 -5 的补码是一样的)


这里的符号位为 1 (代表是负数),因此这需要将补码转为原码。


原码:10000000 00000000 00000000 00000101


c 的原码转为 10 进制就是 -5。


效果展示:

相关文章
|
3月前
|
存储 C语言 索引
【C语言篇】操作符详解(下篇)
如果某个操作数的类型在上⾯这个列表中排名靠后,那么⾸先要转换为另外⼀个操作数的类型后执⾏运算。
72 0
|
18天前
|
存储 缓存 C语言
【c语言】简单的算术操作符、输入输出函数
本文介绍了C语言中的算术操作符、赋值操作符、单目操作符以及输入输出函数 `printf` 和 `scanf` 的基本用法。算术操作符包括加、减、乘、除和求余,其中除法和求余运算有特殊规则。赋值操作符用于给变量赋值,并支持复合赋值。单目操作符包括自增自减、正负号和强制类型转换。输入输出函数 `printf` 和 `scanf` 用于格式化输入和输出,支持多种占位符和格式控制。通过示例代码详细解释了这些操作符和函数的使用方法。
32 10
|
1月前
|
存储 编译器 C语言
【C语言】简单介绍进制和操作符
【C语言】简单介绍进制和操作符
155 1
|
1月前
|
存储 编译器 C语言
初识C语言5——操作符详解
初识C语言5——操作符详解
164 0
|
27天前
|
C语言 C++
C语言 之 内存函数
C语言 之 内存函数
31 3
|
C语言
《啊哈C语言!逻辑的挑战(修订版)》| 每日读本书
“用编程轻松提升逻辑力!每个人都应该学习如何编程,因为它教会你如何思考。——史蒂夫.乔布斯” 。每日搜罗最具权威专业书籍,更多图书请关注“每日读本书”。
3250 0
|
12天前
|
存储 算法 程序员
C语言:库函数
C语言的库函数是预定义的函数,用于执行常见的编程任务,如输入输出、字符串处理、数学运算等。使用库函数可以简化编程工作,提高开发效率。C标准库提供了丰富的函数,满足各种需求。
|
17天前
|
机器学习/深度学习 C语言
【c语言】一篇文章搞懂函数递归
本文详细介绍了函数递归的概念、思想及其限制条件,并通过求阶乘、打印整数每一位和求斐波那契数等实例,展示了递归的应用。递归的核心在于将大问题分解为小问题,但需注意递归可能导致效率低下和栈溢出的问题。文章最后总结了递归的优缺点,提醒读者在实际编程中合理使用递归。
43 7
|
17天前
|
存储 编译器 程序员
【c语言】函数
本文介绍了C语言中函数的基本概念,包括库函数和自定义函数的定义、使用及示例。库函数如`printf`和`scanf`,通过包含相应的头文件即可使用。自定义函数需指定返回类型、函数名、形式参数等。文中还探讨了函数的调用、形参与实参的区别、return语句的用法、函数嵌套调用、链式访问以及static关键字对变量和函数的影响,强调了static如何改变变量的生命周期和作用域,以及函数的可见性。
25 4
|
22天前
|
存储 编译器 C语言
C语言函数的定义与函数的声明的区别
C语言中,函数的定义包含函数的实现,即具体执行的代码块;而函数的声明仅描述函数的名称、返回类型和参数列表,用于告知编译器函数的存在,但不包含实现细节。声明通常放在头文件中,定义则在源文件中。