爱上C语言:操作符详解(下)

🚀 作者:阿辉不一般
🚀 你说呢:生活本来沉闷,但跑起来就有风
🚀 专栏:爱上C语言
🚀作图工具:draw.io(免费开源的作图网站)
请添加图片描述

如果觉得文章对你有帮助的话,还请点赞,关注,收藏支持博主,如有不足还请指点,博主及时改正,感谢大家支持!!!

文章目录

  • 🚀前言
  • 🚀整型的存储以及原、反、补码
  • 🚀移位操作符
    • ✈️左移操作符(<<)
    • ✈️右移操作符(>>)
  • 🚀位操作符
    • ✈️一道变态面试题
  • 🚀单目操作符
    • ✈️ 逻辑反操作(!)
    • ✈️sizeof
  • 🚀剩下的操作符
  • 🚀复杂的表达式如何计算
    • ✈️优先级
    • ✈️结合性

🚀前言

大家好啊😉!承接之前的操作符上篇,今天阿辉将介绍剩下的操作符,包括移位操作符,位操作符以及单目操作符,持续输出干货中,关注阿辉不迷路哦 😘 ,内容干货满满😋,接下来就跟着阿辉一起学习吧👊

🚀整型的存储以及原、反、补码

介绍移位操作符和位操作符前得先了解整型在内存中的存储以及原反补码,阿辉上篇文章上篇文章整型的存储里面有详细的讲解(点击直接跳转哦 😘)

🚀移位操作符

移位操作符有何作用?
移位操作符移动的是数据在内存中存储的补码的二进制位,其中移位操作符的“位”字就是指二进制位

移位操作符有左移操作符右移操作符两种

✈️左移操作符(<<)

左移操作符怎么使用呢?

左操作数 << 右操作数
左操作数是要被左移位的数
右操作数是要左移移动的位数

左移操作符的规则就是左边丢弃,右边补0
我们来看一个🌰栗子

int main()
{
	int a = 10;
	0000 0000 0000 0000 0000 0000 0000 1010 ->a原码
	0000 0000 0000 0000 0000 0000 0000 1010 ->a反码 
	0000 0000 0000 0000 0000 0000 0000 1010 ->a补码 
	int b = a << 2;
	0000 0000 0000 0000 0000 0000 0010 1000 ->b补码 b也就是40
	return 0;
}

在这里插入图片描述
是不是有点懵😏,别急阿辉用图演示👇
请添加图片描述
现在是不是对左移操作符用法清晰多了😁 ,上述演示仅仅使用了正数,不过负数同样可以,负数仅仅是多了原反补码之间的转换

✈️右移操作符(>>)

右移操作符的使用

左操作数 >> 右操作数
左操作数是要被右移位的数
右操作数是要右移移动的位数

右移操作符的规则

  • 算术右移:左边补符号位,右边丢弃
  • 逻辑右移:左边补0,右边丢弃

对于使用逻辑右移还是算术右移从语言并未明确规定,但大部分编译器都使用算术右移,对于无符号数使用逻辑右移

算术右移:左移用正数,咱们这次用负数演示👊

int main()
{
	int a = -10;
	1000 0000 0000 0000 0000 0000 0000 1010->a原码
	1111 1111 1111 1111 1111 1111 1111 0101->a反码
	1111 1111 1111 1111 1111 1111 1111 0110->a补码
	int b = a >> 2;
	1111 1111 1111 1111 1111 1111 1111 1101->b补码
	1000 0000 0000 0000 0000 0000 0000 0010
	1000 0000 0000 0000 0000 0000 0000 0011->b原码 也就是-3
	return 0;
}

请添加图片描述

逻辑右移:逻辑右移也与上述算数右移的运算是类似的,不过逻辑右移不管你是正数还是负数,左边通通补0

注意

  • 移位操作符的操作数只能为整型
  • 对于移位运算符,不要移动负数位,这个是标准未定义的
  • a>>1a<<1这两个都不会改变a的值,就像a+1这样并不会改变a的值

🚀位操作符

位操作符分类

& 按位与
| 按位或
^ 按位异或
~ 按位取反
& | ^ 这三个有两个操作数
~ 只有一个操作数

这里的位同样指的是二进制位,它们操作的依然是内存中的补码
&按位与: 两操作数内存中的补码两相同二进制位均为1就为1,其中一个0就为0
🌰栗子

int main()
{
	int a = 3;
	0000 0000 0000 0000 0000 0000 0000 0011 -> 3的原反补码
	int b = -5;
	1000 0000 0000 0000 0000 0000 0000 0101 -> -5的原码
	1111 1111 1111 1111 1111 1111 1111 1010 -> -5的反码
	1111 1111 1111 1111 1111 1111 1111 1011 -> -5的补码
	int c = a & b;
	1111 1111 1111 1111 1111 1111 1111 1011 -> -5的补码
	0000 0000 0000 0000 0000 0000 0000 0011 -> 3的补码
	0000 0000 0000 0000 0000 0000 0000 0011 -> 按位与结果
	符号位为0,原反补码相同,值为3
	return 0;
}

在这里插入图片描述
| 按位或: 两操作数内存中的补码两相同二进制位均为0就为0,其中一个1就为1

|&的用法类似这里就不举例子了

^按位异或: 两操作数内存中的补码两相同二进制位相同为0,相异为1
🌰栗子

int main()
{
	int a = 3;
	0000 0000 0000 0000 0000 0000 0000 0011 ->3原反补码
	int b = 3;
	int c = a ^ b;
	0000 0000 0000 0000 0000 0000 0000 0011 ->3原反补码
	0000 0000 0000 0000 0000 0000 0000 0011 ->3原反补码
	0000 0000 0000 0000 0000 0000 0000 0000 异或结果
	int d = c ^ a;
	0000 0000 0000 0000 0000 0000 0000 0000 ->c
	0000 0000 0000 0000 0000 0000 0000 0011 ->a
	0000 0000 0000 0000 0000 0000 0000 0011 异或结果
	int e = a ^ a ^ b;
	int f = a ^ b ^ a;
	return 0;
}

在这里插入图片描述
由上面的例子,我们可以得到以下结论👇

相同数字异或结果为0即a ^ a = 0,0与任何数字异或仍为该数字即0 ^ a = a
并且^支持交换律与结合律即a ^ b ^ c = a ^ c ^ b

~按位取反: 操作数在内存中的补码按二进制位0改为11改为0
🌰栗子

int main()
{
	int a = 0;
	0000 0000 0000 0000 0000 0000 0000 0000 ->0的原反补码
	1111 1111 1111 1111 1111 1111 1111 1111 ->按位取反结果
	1000 0000 0000 0000 0000 0000 0000 0001 ->原码值为-1
	int c = ~a;
	return 0;
}

在这里插入图片描述
注意:他们的操作数必须为整数

✈️一道变态面试题

不创建新变量交换两个整数

int main()
{
	int a = 10;
	int b = 12;
	a = a ^ b;
	b = a ^ b;
	a = b ^ a;
	return 0;
}

上面代码是不是很懵 😆,第一次我也很懵逼,我们接着看👇

a = a ^ b这时a的值不在是10而是10 ^ 12,然后b = a ^ b实际上是b = 10 ^ 12 ^12,由上面我们知道的^的特点可知10 ^ 12 ^ 12 = 10,这时b = 10,在看这句a = a ^ b实际上是a = 10 ^ 12 ^ 10,而10 ^ 12 ^ 10 = 12这时a = 12所以ab的值完成了交换

🚀单目操作符

单目操作符的分类:

!           逻辑反操作
-           负值
+           正值
&           取地址
sizeof      操作数的类型长度(以字节为单位)
~           对一个数的二进制按位取反
--          前置、后置--
++          前置、后置++
*           间接访问操作符(解引用操作符)
(类型)       强制类型转换

✈️ 逻辑反操作(!)

🌰栗子

int main()
{
	int a = 0;
	int b = 4;
	if (!(b == 4))
		printf("真");
	printf("%d ", !a);
	printf("%d\n", !b);
	return 0;
}

在这里插入图片描述
逻辑反操作即把0 (假) 改为1 (真) ,非0 (真) 改为0 (假)

✈️sizeof

sizeof这个操作符,sizeof是由来计算变量(类型)所占空间的大小,不关注存储的内容,单位是字节

int main()
{
	int a = 2;
	int b[4] = { 0 };
	char c = '1';
	printf("%u\n", sizeof(b));
	printf("%u\n", sizeof(b[0]));
	printf("%u\n", sizeof(a));
	printf("%u\n", sizeof(c));
	printf("%u\n", sizeof(int));
	return 0;
}

在这里插入图片描述
sizeof的操作数为数组名时计算的整个数组的大小

🚀剩下的操作符

其中有一些过于简单相信大家都掌握了,至于**结构体成员访问操作符(.)和(->)**会在后续结构体章节详细讲到, **解引用操作符(*)和取地址操作符(&)**会在后续数组篇中讲解

🚀复杂的表达式如何计算

C语言的操作符有2个重要的属性:优先级、结合性,这两个属性决定了表达式求值的计算顺序

✈️优先级

相邻操作符的优先级高的先算

int main()
{
	int a = 3;
	int b = 4;
	a - b * a;
}

上述代码*乘的优先级更高先算b * a而不是先算a - b

✈️结合性

当相邻两操作符优先级相同就要看结合性了,左结合从左向右计算,右结合从右向左计算

int main()
{
	int a = 3;
	int b = 4;
	a / b * a;
}

上述代码中/*的优先级相同都是左结合,因此先算a / b
下面是一张操作符优先级与结核性的表
在这里插入图片描述
虽然我们知道了操作符的结合性与优先级,但这并不能保证表达式具有唯一计算路径,因此尽量不要写过于复杂的表达式很容易出bug


到这里,阿辉今天对于C语言中操作符的分享就结束了,希望这篇博客能让大家有所收获, 如果觉得阿辉写得不错的话,记得给个赞呗,你们的支持是我创作的最大动力🌹
请添加图片描述

  • 85
    点赞
  • 69
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 76
    评论
C语言中缀表达式求值一般采用栈来实现,具体步骤如下: 1. 定义一个操作数栈和一个操作符栈。 2. 从左到右依次扫描表达式的每个元素,如果是数字则将其压入操作数栈中,如果是操作符则比较其与操作符栈栈顶元素的优先级。 3. 如果该操作符优先级高于栈顶操作符,则将该操作符压入操作符栈中。 4. 否则,将操作符栈栈顶元素弹出,从操作数栈中弹出两个数字,进行计算并将结果压入操作数栈中,直到该操作符可以入栈。 5. 如果扫描完整个表达式后,操作符栈不为空,则将操作符栈中的操作符依次弹出,从操作数栈中弹出两个数字进行计算并将结果压入操作数栈中。 6. 最终操作数栈中剩下的数字即为表达式的值。 例如,对于中缀表达式 "2 + 3 * 4 - 5",按照上述步骤可以得到如下计算过程: 1. 将2压入操作数栈中。 2. 将"+"压入操作符栈中。 3. 将3压入操作数栈中。 4. 将"*"与操作符栈栈顶元素"+"比较,"*"的优先级高,将"*"压入操作符栈中。 5. 将4压入操作数栈中。 6. 将"-"与操作符栈栈顶元素"*"比较,"-"的优先级低,将"*"弹出,从操作数栈中弹出4和3进行计算得到12,将12压入操作数栈中。 7. 将"-"压入操作符栈中。 8. 将5压入操作数栈中。 9. 扫描完整个表达式后,操作符栈中剩余"-",将"*"弹出,从操作数栈中弹出12和2进行计算得到24,将24压入操作数栈中,将"-"弹出,从操作数栈中弹出24和5进行计算得到19,最终操作数栈中剩下的数字19即为表达式的值。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

阿辉不一般

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值