一文详解C语言操作符

目录
  • 一、基础篇
    • 1.算数操作符
    • 2.单目操作符
    • 3.逻辑操作符
    • 4.条件操作符
    • 5.逗号表达式
    • 6.下标引用,函数调用和结构体成员访问
    • 7.关系操作符
  • 二、进阶篇
    • 1.二进制
    • 2.移位操作符
    • 3.位操作符
  • 三、同一操作符的不同类型操作数之间的转化
  • 四、操作符的优先级

一、基础篇

1.算数操作符

+   -   *   /   %   +=   -=  *=  /=  这些操作符是我们编程时候最常用的几个算数操作符

+ - * 等算数操作符就不用再多说了,与正常的数学用法相同,下面是一些常用但是初学者可能不太容易把握、易错的算数操作符

(1)/ 操作符

/:除法,除法在C语言中分为整数除法浮点数除法

整数除法:/d的两个操作数都为整数,计算方法是向下取整

得到的结果为

浮点数除法:储存值的变量必须是浮点数, 两个操作数中至少有一个为浮点数,但除数或者被数不一定要用浮点数存储,只需要在计算时进行转化即可

%.nlf也可以使结果自动四舍五入保留n位小数

(2)%取余操作符

取余运算符得到的是余数,但取余操作符的两个操作数都要为整数

c的值就等于8除以3的余数 2

(3) +=  -=  *=  /=操作符

直接举例子:

a+=2就等同于a=a+2,将a的值加2再赋值给a

a-=2就等同于a=a-2,将a的值减2再赋值给a

*=、/=也是相同的道理

2.单目操作符

单目操作符,顾名思义,单目,就是操作数只有一个的操作符

(1)!  操作符

! 逻辑反操作,将真变为假,将假变为真

在逻辑定义中,0表示假,非0则表示真

在这段代码中,先给i赋初值0,然后在while判断是否进入循环时,!0就是真,进入循环,但为了防止死循环,我们加上一个限制条件,保证只打印5次

同样,如果while判断条件只有一个0时,则说明为假,不进入循环,打印0次

(2)sizeof 操作符

sizeof(类型名);sizeof()返回无符号的所计算的对象或者类型所占空间的大小,单位是字节

sizeof经常也用来计算数组元素的个数

(3)~操作符

~:按位取反操作符,对二进制位进行按位取反,将二进制补码每一位的0变1,1变0。注意,是对补码进行操作,负数要先将原码变为补码,同时取反得到的也是补码,看结果还得再变为原码

从图中可以看出,0在按位取反后应该是  -1,同样,在编译器上验证

(4)++、--操作符

前置++,前置--:先将变量+1或者-1,再作为表达式的值使用(先++,再使用)

后置++,后置--:先将变量的值作为表达式使用,再将变量+1或者-1(先使用,再++)

用上面的口诀想起来就容易多了

(5)&  取地址操作符

&:取出某个变量、数组、函数等的地址,在比价复杂的代码中,我们经常用指针进行操作,&操作符就可以取出地址赋值给指针变量

(6)*  间接访问操作符 

间接访问操作符,也称解引用操作符,操作数地址或指针,可以通过对地址的解引用,找到地址中存储的变量,对变量进行间接操作

(7)()强制类型转化

():将一种数据类型的变量强制转化为另一种数据类型

如参数需要一个无符号的整形变量,而你目前所有的是一个整形变量啊,就可以通过强制类型转化转化为组符号的整形,这个经常用于函数参数的优化部分

如:(unsigned int)a   即可将a转化为无符号的整形,同样,强制类型转化尽量要遵守由低到高转化,否则,也会丢失精度

3.逻辑操作符

&& 逻辑与 操作符:“同时都要满足”

||    逻辑或 操作符:“满足其中一个即可”

1&&2---->1       0&&1----->0

1||2   ---->1       0||1----->1

1表示真,0表示假

例:找出1000---2000之间的闰年

闰年:如果year能够被4整除,并且不能被100整除,则year是闰年。或者如果year能够被400整除,则year是闰年,判断闰年就恰好用到了这两个操作符

int year = 0;
	for (year = 1000; year < 2000; year++)
	{
		if ((year % 4 == 0 && year % 100 != 0) || year % 400 == 0)
			//判断闰年
			printf("%d ", year);

	}

逻辑操作符这里有一个超级易错点

对于&&来说,如果表达式左边为0,则说明在逻辑上已经为了,那么右边全部的表达就不再进行计算,也不执行。

int i = 0,a=0,b=2,c =3,d=4;
    i = a++ && ++b && d++;
    printf("a = %d\n b = %d\n c = %d\nd = %d\n", a, b, c, d);

a++;---->是先将a的值使用,在对a加1。刚使用的表达式a的值为0,并且在&&的左边,所以后面++b,d++不再计算。

同样,对于||来说,如果表达式左边为非0则说明在逻辑上已经为了,那么右边全部的表达就不再进行计算,也不执行。

int i = 0, a = 0, b = 2, c = 3, d = 4;
	i = a++||++b||d++;
	printf("a = %d\n b = %d\n c = %d\nd = %d\n", a, b, c, d);

a++这个表达式的值为0,继续执行,但是++b的值为3,在||操作符的左边,并且在逻辑上已经为真了,所以后面的d++不再计算

4.条件操作符

条件操作符是C语言中唯一一个三目操作符

表达式1如果为真,即计算表达式2,同时表达式2的结果就是三目操作符的运算结果

表达式2如果为假,即计算表达式3,同时表达式3的结果就是三木操作符的运算结果

也可以在编译器里面运行来验证

int a = 0, b= 3, max1 = 0,max2;
	if (b > a)
		max1 = b;
	else
		max1 = a;
	//等同于
	max2 = (b > a) ? b : a;
	printf("%d %d", max1, max2);

运算结果当然是max1=3  max2=3

5.逗号表达式

逗号表达式,就是用逗号隔开的多个表达式。里面的表达式从左向右依次执行,整个表达式的结果是最后一个表达式的结果。

int a = 1;
int b = 2;
int c = (a>b, a=b+10, a, b=a+1)

从左向右一次算a=b+10=12 ; b=a+1=13;然后将这个逗号表达式整体的值(即b=a+1)赋给变量c

即 c=13;

需要注意的是逗号表达式的每一个表达式都必须从左向右依次计算

6.下标引用,函数调用和结构体成员访问

(1)下标引用操作符[]

arr[2]=2;//将2赋值给arr[2]

[ ]就是下标引用操作符,arr和2是[ ]的两个操作数,arr为数组名,2为索引值

(2)函数调用操作符()

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

(3)结构体访问操作符.->

用法:

结构体变量.结构体成员

结构体指针变量->结构体成员

#include <stdio.h>
struct Stu
{
 char name[10];
 int age;
 char sex[5];
 double score;
};
void set_age1(struct Stu stu)
{
 stu.age = 18;//结构体变量访问结构体成员
}
void set_age2(struct Stu* pStu)
{
 pStu->age = 18;//结构体指针变量访问结构体成员
}
int main()
{
 struct Stu stu;
 struct Stu* pStu = &stu; 

 stu.age = 20;//结构成员访问
 set_age1(stu);

 pStu->age = 20;//结构成员访问
 set_age2(pStu);
 return 0;
}

7.关系操作符

>

>=

<

<=

== 用于测试“相等”

!=  用于测试“不相等”

二、进阶篇

1.二进制

(1)二进制的计算

(2)二进制的存储

2.移位操作符

移位操作符移动的是二进制的补码,且操作数只能为整数

<< 左移操作符:将二进制位整体向左移动,左边丢弃,右边补零

图为将-15左移1位

>> 右移操作符:将二进制位整体向右移动,右边丢弃,左边补符号位

图为将-15右移一位

负数学会了,整数就更简单了,因为整数的原码,反码,补码都一样,不用互相转化

3.位操作符

位操作符的操作数必须为整数

& 按位与: a&b 对应的二进制补码 :有0则为0,同时为1才为1

| 按位或: a|b 对应的二进制补码 :有1则为1,同时为0才为0

^ 按位异或a^b 对应的二进制补码 :相同为0,不同为1

实例1:不创建第三个变量,实现两个整数的交换

方法1:

    int a = 2;
	int b = 3;
	printf("交换前:a = %d, b= %d\n", a, b);
	a = a + b;
	b = a - b;
	a = a - b;
	printf("交换后:a = %d, b= %d", a, b);

这种方法在一般情况下可以,但是当a,b非常大时,a+b可能会超过存储范围(长度大于整形)

方法2:用按位异或计算

int main()
{
	int a = 2;
	int b = 3;
	printf("交换前:a = %d, b= %d\n", a, b);
	a = a ^ b;
	b = a ^ b;
	a = a ^ b;
	printf("交换后:a = %d, b= %d", a, b);
	return 0;
}

在解释原因前,先演示一下按位异或的几个规则

a^a=0 详细看图

a^0=a 详细看图

a^b^a=a^a^b,即按位异或运算满足交换律 详细看图

解释一下原因:a=a^b,然后 b=  (a^b)  ^b =a^0=a,即实现了将a的值赋给了b

同样:a=a^b,然后a=(a^b)^a=a^a^b=0^b=b,即实现了将b的值赋给了a

实例2:求一个整数存储在内存中的二进制中1的个数

一个数&1可以得到它的最低位(详细参考按位与操作符图解),如果a&1==1,说明a的最低位是1如果a&1==0,说明a的最低位是0,那么思路就清晰了,只需要将a的二进制位多次右移,每次右移后&1得到它的最低位,判断是否为1,为1的话,计数器就++,将32个二进制位尽皆遍历,即可知道共有多少个1

int main()
{

	int a = 31;//5个1
	int i = 0;
	int count = 0;
	for (i = 0; i < 32; i++)
	{
		if (a & 1 == 1)	//说明a的最低位是1
			count++;
		a >>= 1;
	}
	printf("%d", count);
}

三、同一操作符的不同类型操作数之间的转化

例如:将整形3存到浮点型变量中,没问题。但是,将一个浮点型的数字存到一个整形变量中去,就可能会出现精度丢失问题。这些转化需要我们在写代码的时候自行完成

四、操作符的优先级

C语言操作符的优先级

优先级 运算符符号 名称或者含义            
        1 []、().-> 数组下标引用、圆括号、结构体访问
        2 -、(类型)、++、--、*、&、!、~、sizeof 负号、强制类型转化、自加、自减、解引用、取地址、逻辑非、按位取反、siaeof
3 /、*、% 除号、乘号、取模
4 +、- 加、减
5 <<、>> 左移、右移
6 >、>=、<、<= 大于、大于等于、小于、小于等于
7 ==、!= 等于、不等于
8 &
按位与

9 ^ 按位异或
10 | 按位或
11 && 逻辑与
12 || 逻辑或
13 ?: 条件(三目)运算符

在使用这张表时需仔细核对运算符的符号和名称,但在大多数情况下,我们选择直接将需要优先计算的用圆括号括起来,不过最基本的顺序还是得知道的,总不能一份代码到处都是圆括号吧。

以上就是一文详解C语言操作符的详细内容,更多关于C语言操作符的资料请关注我们其它相关文章!

(0)

相关推荐

  • C语言简明讲解操作符++和--的使用方法

    目录 一.++与--操作符的本质 二.++与-- 操作符使用分析 三.小结 一.++与--操作符的本质 ++ 和 -- 操作符对应两条汇编指令 前置 变量自增(减)1 取变量值 后置 取变量值 变量自增(减)1 下面看一段神奇的代码: #include <stdio.h> int main() { int i = 0; int r = 0; r = (i++) + (i++) + (i++); printf("i = %d\n", i); printf("r =

  • C语言超全面覆盖操作符知识点

    目录 算术操作符 赋值操作符 算术操作符 主要是 (+ - * / %) 五种算数操作符. 1.除了 % 操作符之外,其他的几个操作符可以作用于整数和浮点数. 2.对于 / 操作符如果两个操作数都为整数,执行整数除法.而只要有浮点数执行的就是浮点数除 法. 3.% 操作符的两个操作数必须为整数.返回的是整除之后的余数. 当除数为小数时,int类型只会截取整数部分. 移位操作符 << 左移操作符 >> 右移操作符 左移操作符 移位规则: 左边抛弃.右边补0 右移操作符 移位规则: 首

  • C语言 操作符#与##使用方法详解

    目录 一.# 运算符 二.## 运算符 三.小结 一.# 运算符 # 运算符用于在预处理期将宏参数转换为字符串 # 的转换作用是在预处理期完成的,因此只在宏定义中有效 编译器不知道 # 的转换作用 用法: #define STRING(x) #x printf("%s\n",STRING(Hello World!)); 下面通过一个示例感受一下: test.c: #include <stdio.h> #define STRING(x) #x int main() { pri

  • C语言 操作符分类解析与使用

    目录 操作符的分类 算术操作符 移位操作符 位操作符 逻辑操作符 逗号表达式 表达式求值 隐式类型转换 算术转换 操作符的属性 xwg今天就带各位大佬来了解一波C语言的操作符. 操作符的分类 常见的操作符分别如下: 算术操作符 移位操作符 位操作符 逻辑操作符 逗号表达式 算术操作符 算术操作符是我们最常用的操作符:+ - * / % 注: 除了 % 操作符之外,其他的几个操作符可以作用于整数和浮点数. 对于 / 操作符如果两个操作数都为整数,执行整数除法,而只要有浮点数执行的就是浮点数除法.

  • C语言冷知识之预处理字符串操作符详解

    目录 在C语言中什么是标记 预处理字符串操作符 #字符串化操作符 ##标记(Token)连接操作符 当年学习C语言的第一门课就提到过标记(Token)的概念,不过,相信在多年之后你再次听到这个术语时会一脸懵逼,比如我. 因此特地翻了翻资料,整理下来这些笔记. 在C语言中什么是标记 标记是编程语言处理的基本单元,也叫最小划分元素,比如关键字.操作符.变量名.函数名.字符串.数值等等. 下面举例说明一下: printf("hello world!"); 对上面的语句进行标记划分,可分为5个

  • 一文详解C语言中文件相关函数的使用

    目录 一.文件和流 1.程序文件 2.数据文件 3.流 二.文件组成 三.文件的打开和关闭 1.文件的打开fopen 2.文件关闭fclose 四.文件的顺序读写 1.使用fputc和fgetc写入/读取单个字符 2.使用fputs和fgets写入/读取一串字符 3.使用fprintf和fscanf按照指定的格式写入/读取 4.使用fwrite和fread按照二进制的方式写入/读取 5.使用sprintf和sscanf将格式化数据和字符串互相转换(文件无关) 五.文件的随机读写 1.fseek(

  • 一文详解Go语言单元测试的原理与使用

    目录 前言 为什么要引用单元测试类 单元测试基本介绍 优点 Testing规范 基本使用 Golang运行 命令行 案例 前言 为什么要引用单元测试类 传统方法的缺点分析 不方便,我们需要在main函数中去调用,这样就需要去修改main函数,如果现在项目正在运行,就可能去停止项目 不利于管理,因为当我们测试多个函数或者多个模块时,都需要写在main函数,不利于我们管理和清晰我们的思路 单元测试基本介绍 Go语言中自带有一个轻量级的测试框架testing和自带的go test命令来实现单元测试和性

  • 一文详解Go语言fmt标准库的常用占位符使用

    目录 占位符 占位符使用示例 占位符 通过占位符,可以指定格式进行输入或输出,以下为 fmt 标准库里的占位符: 普通占位符 占位符 描述 举例 结果 %v 默认格式的值 fmt.Printf("%v", User{Name: "小明", Age: 18}) {小明 18} %+v 如果打印的是结构体,额外打印字段名 fmt.Printf("%+v", User{Name: "小明", Age: 18}) {Name:小明 A

  • 一文详解C++中动态内存管理

    目录 前言 1.C/C++程序的内存开辟 2.C语言中动态内存管理方式:malloc/calloc/realloc/free 2.1malloc.calloc.realloc区别? 3.C++内存管理方式 3.1 new/delete操作内置类型 3.2 new和delete操作自定义类型 3.3new和malloc处理失败 4.operator new与operator delete函数 4.1 operator new与operator delete函数 4.1.1 我们看看operator

  • 详解C 语言项目中.h文件和.c文件的关系

    详解C 语言项目中.h文件和.c文件的关系 在编译器只认识.c(.cpp))文件,而不知道.h是何物的年代,那时的人们写了很多的.c(.cpp)文件,渐渐地,人们发现在很多.c(.cpp)文件中的声明语句就是相同的,但他们却不得不一个字一个字地重复地将这些内容敲入每个.c(.cpp)文件.但更为恐怖的是,当其中一个声明有变更时,就需要检查所有的.c(.cpp)文件. 于是人们将重复的部分提取出来,放在一个新文件里,然后在需要的.c(.cpp)文件中敲入#include XXXX这样的语句.这样即

  • 详解C语言数组越界及其避免方法

    所谓的数组越界,简单地讲就是指数组下标变量的取值超过了初始定义时的大小,导致对数组元素的访问出现在数组的范围之外,这类错误也是 C 语言程序中最常见的错误之一. 在 C 语言中,数组必须是静态的.换而言之,数组的大小必须在程序运行前就确定下来.由于 C 语言并不具有类似 Java 等语言中现有的静态分析工具的功能,可以对程序中数组下标取值范围进行严格检查,一旦发现数组上溢或下溢,都会因抛出异常而终止程序.也就是说,C 语言并不检验数组边界,数组的两端都有可能越界,从而使其他变量的数据甚至程序代码

  • 一文详解JS私有属性的6种实现方式

    目录 _prop Proxy Symbol WeakMap #prop ts private 总结 class 是创建对象的模版,由一系列属性和方法构成,用于表示对同一概念的数据和操作. 有的属性和方法是对外的,但也有的是只想内部用的,也就是私有的,那怎么实现私有属性和方法呢? 不知道大家会怎么实现,我梳理了下,我大概用过 6 种方式,我们分别来看一下: _prop 区分私有和公有最简单的方式就是加个下划线 _,从命名上来区分. 比如: class Dong { constructor() {

  • 一文详解Java中的类加载机制

    目录 一.前言 二.类加载的时机 2.1 类加载过程 2.2 什么时候类初始化 2.3 被动引用不会初始化 三.类加载的过程 3.1 加载 3.2 验证 3.3 准备 3.4 解析 3.5 初始化 四.父类和子类初始化过程中的执行顺序 五.类加载器 5.1 类与类加载器 5.2 双亲委派模型 5.3 破坏双亲委派模型 六.Java模块化系统 一.前言 Java虚拟机把描述类的数据从Class文件加载到内存,并对数据进行校验.转换解析和初始化,最 终形成可以被虚拟机直接使用的Java类型,这个过程

  • 详解C语言中结构体的使用

    目录 结构体的声明 结构体成员的类型 结构体成员的访问 结构体的声明 结构体的定义:结构体是一些值的集合,这些值称为成员变量,结构体的每个成员可以是不同类型的变量. 举例: //定义结构体类型 struct tag//struct结构体关键字 tag结构体标签 struct tag结构体类型 { //成员变量 char name[20]; short age; char telphone[12]; char sex[5]; }s1,s2,s3;//s1,s2,s3是三个全局结构体变量 int m

  • 一文详解Golang中net/http包的实现原理

    目录 前言 http包执行流程 http包源码分析 端口监听 请求解析 路由分配 响应处理 前言 Go语言自带的net/http包提供了HTTP客户端和服务端的实现,实现一个简单的http服务非常容易,其自带了一些列结构和方法来帮助开发者简化HTTP服务开发的相关流程,因此我们不需要依赖任何第三方组件就能构建并启动一个高并发的HTTP服务器,net/http包在编写web应用中有很重要的作用,这篇文章会学习如何用 net/http 自己编写实现一个 HTTP Server 并探究其实现原理,具体

随机推荐

其他