C语言操作符详解(上)

前言

在C语言中,C标准提供了丰富的操作符,被用来对数据进行操作和计算,常用数学计算、比较、逻辑运算等操作。操作符是编程语言中非常重要的一部分,它们可以使程序更加简洁、高效,同时也可以增强代码的可读性和可维护性。本文将详细系统介绍C语言操作符,希望能帮读者全面理解C语言操作符!

1. 算术操作符

算术操作符分为以下5种,虽然简单,但还是有一些细节需要注意。

+)、(-)、(*)、(/)、(%

  • 除了%(取余)操作符之外,其他几个操作符可以作用于整数和浮点数。
  • 对于/操作符,如果两个操作数都为整数,执行整数除法。而只要有一个浮点数就执行浮点数除法。
  • %操作符的两个操作数必须为整数。返回的是整除之后的余数。

2. 移位操作符

在C语言中,移位操作符分为以下两种

  • <<左移操作符
  • >>右移操作符

相关知识补充:

#include <stdio.h>
int main()
{
	int a = 15;
	//00000000000000000000000000001111 - 原码
	//00000000000000000000000000001111 - 反码
	//00000000000000000000000000001111 - 补码

	int b = -15;
	//10000000000000000000000000001111 - 原码
	//11111111111111111111111111110000 - 反码
	//11111111111111111111111111110001 - 补码
	return 0;
}

Tips:

  • 位移操作符的操作数只能是整数。
  • 计算机能处理的是二进制的信息。而整数的二进制表示形式分为三种:原码、反码、补码。
  • 正整数的原码、反码、补码是相同的。负整数的反码是原码符号位不变,其他未按位取反、反码在加1就得到补码。

2.1 左移操作符(<<)

移位规则:

左边抛弃、右边补0

例子:

#include <stdio.h>
int main()
{
	int a = 15;
	//00000000000000000000000000001111 - (a)补码

	int b = a << 1;
	//对a的补码右移一位得到b的补码,左边抛弃,右边补0
	//00000000000000000000000000011110 - (b)补码
	//00000000000000000000000000011110 - (b)反码
	//00000000000000000000000000011110 - (b)原码
	
	printf("b=%d\n", b); //b得值为:30
	return 0;
}

2.2 右移操作符(>>

移位规则:

C语言中,右移运算分为两种:

  1. 逻辑右移:左边用0填充,右边丢弃。
    2. 算术右移:左边用原该值的符号位填充,右边丢弃。

上述两种运算方式,C语言没有明确规定到底是算术右移还是逻辑右移。但一般编译器上采用的是算术右移。
例子:

int main()
{
	int a = -15;
	//10000000000000000000000000001111 - 原码
	//11111111111111111111111111110000 - 反码
	//11111111111111111111111111110001 - 补码

	int b = a >> 1;
	//左边用原该值的符号位填充,右边丢弃
	//11111111111111111111111111111000 - 补码
	//11111111111111111111111111110111 - 反码
	//10000000000000000000000000001000 - 原码
	
	printf("b=%d\n", b); //b的值为:-8
	return 0;
}

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

int num=10;
num>>-1; //error

3. 位操作符

在C语言中,位操作符有以下三种:&(按位与)、|(按位或)、^(按位异或)。
Tips:位操作符的操作数必须是整数,同时也是操作二进制位的。

3.1 按位与(&

& — 补码对应的二进制位有0则为0,两个同时为1才是1.

例子:

int main()
{
	int a = 3;
	//00000000000000000000000000000011 - 补码
	int b = -5;
	//10000000000000000000000000000101 - 原码
	//11111111111111111111111111111010 - 反码
	//11111111111111111111111111111011 - 补码

	int c = a & b;
	//&补码对应的二进制位有0则为0,两个同时为1才是1.
	//00000000000000000000000000000011 - 补码(a)
	//11111111111111111111111111111011 - 补码(b)
	//00000000000000000000000000000011 - 补码(c)

	printf("c=%d\n", c); //c的值为:3
	return 0;
}

3.2 按位或(|

| — 补码对应的二进制位有1则为1,两个同时为0才是0.

例子:

int main()
{
	int a = 3;
	//00000000000000000000000000000011 - 补码
	int b = -5;
	//10000000000000000000000000000101 - 原码
	//11111111111111111111111111111010 - 反码
	//11111111111111111111111111111011 - 补码

	int c = a | b;
	//&补码对应的二进制位有1则w为1,两个同时为0才是0.
	//00000000000000000000000000000011 - 补码(a)
	//11111111111111111111111111111011 - 补码(b)
	//11111111111111111111111111111011 - 补码(c)

	//11111111111111111111111111111010 - 反码(c)
	//10000000000000000000000000000101 - 原码(c)

	printf("c=%d\n", c);//c的值为:-5
	return 0;
}

3.4 按位异或(^

|^— 补码对应的二进制位不相同则为1,两个相同则为0.

int main()
{
	int a = 3;
	//00000000000000000000000000000011 - 补码
	int b = -5;
	//10000000000000000000000000000101 - 原码
	//11111111111111111111111111111010 - 反码
	//11111111111111111111111111111011 - 补码

	int c = a ^ b;
	//^ 补码对应的二进制位不相同则为1,两个同时为1或0时才是0.

	//00000000000000000000000000000011 - 补码(a)
	//11111111111111111111111111111011 - 补码(b)
	//11111111111111111111111111111000 - 补码(c)

	//11111111111111111111111111110111 - 反码(c)
	//10000000000000000000000000001000 - 原码(c)

	printf("c=%d\n", c);//c的值为:-8
	return 0;
}

3.5 一道变态的面试题

不能创建临时变量(第三个变量),实现两个数的交换。

int main()
{
	int a = 10;
	int b = 20;
	//交换,按位异或是支持交换律的
	a = a ^ b;
	b = a ^ b;
	a = a ^ b;
	printf("a=%d b=%d\n", a, b);
	return 0;
}
  • 按位异或是支持交换律的。

4. 关系操作符

在这里插入图片描述
这些关系运算符比较简单,没什么可介绍的。

5. 单目操作符

单目操作符介绍:

在这里插入图片描述

5.1 单目操作符(!

(!)逻辑反操作,顾名思义就是把真变成假,把假变成真。

int main()
{
	int flag = 0;//下面两种判断flag为0就执行语句的方法,效果一样
	//方法1:
	if (flag == 0)
	{
		//语句;
	}

	//方法2:flag为0,则为假。!flag则为真,执行语句
	if (!flag)
	{
		//语句;
	}
	return 0;
}

5.2 单目操作符(~

单目操作符(~)是按补码二进制位全部取反.(包括符号位)

例子:

int main()
{
	int a = 0;
	//00000000000000000000000000000000 - 补码

	int b = ~a;
	//~a--对a的补码全部取反
	//11111111111111111111111111111111 - 补码
	//11111111111111111111111111111110 - 补码
	//10000000000000000000000000000001 - 原码

	printf("%d\n", b); //b的值位:-1
	return 0;
}

5.3 单目操作符(&、*

单目操作符(&、*)都运用于指针。

int main()
{
	int a = 0;
	//pa是指针变量
	int* pa = &a; //&—取地址操作符—取出a的地址
	
	//解引用操作符(间接访问操作符)—通过pa中存放的地址,找到指向的空间(内容)
	*pa = 20;
	return 0;
}

5.4 单目操作符(++、–

单目单目操作符++- -),分为前置++- -)和后置++
前置++): 先++(- -),在使用。
后置++): 先使用,在++(- -)。

例子1:

int main()
{

	int a = 2;
	int b = --a; //前置--,先--,在使用
	//a=a-1,b=a
	printf("a=%d b=%d\n", a, b);//a,b的值都为:1	
	return 0;
}

例子2:

int main()
{
	int a = 2;
	int b = a--; //后置--,先使用,在--
	//b=a,a=a-1
	printf("a=%d b=%d\n", a, b);//a的值为1,b的值为2
	return 0;
}

前置++和后置++同理。

5.5 单目操作符(sizeof)和数组

单目操作符(sizrof)用于计算操作数的类型长度。(单位字节)

#include <stdio.h>
void test1(int arr[])
{
	printf("%d\n", sizeof(arr));
}

void test2(char ch[])
{
	printf("%d\n", sizeof(ch));
}
int main()
{
	int arr[10] = { 0 };
	char ch[10] = { 0 };
	printf("%d\n", sizeof(arr));//10X4
	printf("%d\n", sizeof(ch)); //10X1
	test1(arr);
	test2(ch);
	return 0;
}

运行结果(x64环境下):
在这里插入图片描述
上述前两个结果毋庸置疑,但为什么后两个都是8byte呢?
上面调用函数test1()和test2(),传过去的是指针。C语言中,在64位机器上(x64),指针大小为8byte;在32位机器上(x86)上,指针大小为4byte.

6. 逻辑操作符

C语言中,逻辑操作符分为以下两种:

&&—逻辑与
||—逻辑或

简单来说:逻辑与(&&)就是两个同时城里时为真,否则为假;逻辑或(||)则是只要有一个成立则为真,否则为假。
Tips:

  • &&操作符,左边为假,右边就不计算了。
  • ||操作符,左边为真,右边就不用计算了。

例子1:

#include <stdio.h>
int main()
{
	int i = 0, a = 0, b = 2, c = 3, d = 4;
	//后置++,先使用a,后再执行++
	//&&操作符,左边为假,右边就不计算了
	//a先使用,为0,则为假。所以&&后面所有运算都不执行
	i = a++ && ++b && d++;
	//结果:a=1 b=2 c=3 d=4
	printf("a=%d b=%d c=%d d=%d\n", a, b, c, d);
	return 0;
}

例子:

#include <stdio.h>
int main()
{
	int i = 0, a = 0, b = 2, c = 3, d = 4;
	//||操作符,左边为真,右边就不用计算了。
	//使用b时,b先++后使用,别的值为3,非0为真。所以后面所有运算都不进行
	i = a++ || ++b || d++;
	//结果:a=1 b=3 c=3 d=4
	printf("a=%d b=%d c=%d d=%d\n", a, b, c, d);
	return 0;
}

7. 条件操作符

C语言中,条件操作符如下:表达式exp1为真,则执行表达式exp2,否则执行表达式exp3.

exp1 ? exp2 : exp3

例子:

int main()
{
	int a = 0;
	int b = 0//a为0小于5,执行表达式xep3。所以b=-3
	b = (a > 5) ? 3 : -3;
	return 0;
}

8. 逗号表达式

C语言中,逗号表达式就是:用逗号隔开的多个表达式。

exp1, exp2, exp3, …expN

例子:

int main()
{
	int a = 1;
	int b = 2;
	int c = (a > b, a = b + 10, a, b = a + 1);//逗号表达式
	printf("c=%d\n", c);//结果为:c=13
	return 0;
}

那在上述代码中,为什么c的值最终为13呢?

  • 逗号表达式中,从左到右依次计算。整个表达式的结果为最后一个表达式的结果。

例子:

int d=1;
if(a=a+1,c=a/2,d>0)//d大于0,结果为真,执行if后的语句块
{}

9. 下标引用、函数调用和结构成员

9.1 [ ] 下标引用操作符

在C语言中,下标引用的操作数为:一个数组名 + 一个索引值两个操作数满足交换律)

例子:

int main()
{
	int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };//数组的起始是有下标的,下标从0开始
	//			    0 1 2 3 4 5 6 7 8 8 9
	printf("%d\n", arr[2]);// [] 下标引用操作符,arr和2是两个操作数
	printf("%d\n", 2[arr]);
	return 0;
}
  • 上述代码中2[arr]虽然看起来奇怪,但却是成立的,和arr[2]等价。[ ]和+一样,都是操作数,和两个操作数的前后顺序无关。

9.2 ( ) 函数调用操作符

( ) 函数调用操作符可以接受一个或多个操作数:第一个操作数是函数名,剩余操作数就是传递给函数的参数。

例子:

#include <stdio.h>
#include <string.h>
int Add(int x, int y)
{
	return x + y;
}

int main()
{
	int len = strlen("abc");//( )函数调用操作符
	//()的操作数是:strlrn 和 "abc
	printf("%d\n", len);

	int c = Add(3, 5);//( ) 函数调用操作符
	//()的操作数是:Add 和3 5
	//对于函数调用操作符来说,至少有一个操作数

	printf("%d\n", c);
	return 0;
}

9.3 访问一个结构的成员

在C语言中,访问一个结构的成员有两种方式:

.结构体 . 成员名
->结构体 -> 成员名

例子1:

#include <stdio.h>
//定义一个struct Book类型
struct Book
{
	char name[30];
	char author[20];
	float price;
};

int main()
{
	struct Book b= { "C primer Plus","Stephen Prate",47.8f };//创建一个变量b
	//结构体.成员名,答应相关信息
	printf("%s %s %.1f\n", b.name, b.author, b.price);
	return 0;
}

例子2:

struct Book
{
	char name[30];
	char author[20];
	float price;
};
//定义过程
void print(struct Book* b)
{
	//结构指针->成员名,访问成员
	printf("%s %s %.1f\n", b->name, b->author, b->price);
}

int main()
{
	struct Book b= { "C primer Plus","Stephen Prate",47.8f };//创建一个变量b
	//封装print()函数,打印信息
	print(&b);
	return 0;
}

10. 结尾

本篇文章到此就结束了!在C语言操作符详解(上)已经将C语言中所有操作全部详细系统的介绍完了,接下来将在C语言操作符详解(下)中详细介绍编译器执行这些代码背后所做的事情、操作符优先级、结合性以及求值顺序!
创作不易,如果对你有帮助,记得点赞加关注哦!感谢您的支持,同时也欢迎读者发表自己的见解!
C语言操作符详解(下)

评论 12
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

是小宇吖~~

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

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

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

打赏作者

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

抵扣说明:

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

余额充值