初阶C语言-操作符详解(下)

🌞 “等春风得意,等时间嘉许!” 接下来,我们把操作符没学完的继续学完!
操作符详解
- 6.2sizeof和数组
- 7.关系操作符
- 8.逻辑操作符
- 9.条件操作符
- 10.逗号表达式
- 11.下标引用、函数调用和结构成员
- 12.表达式求值
- 12.1隐式类型转换
- 12.2算术转换
- 12.3 操作符的属性
6.2sizeof和数组
📖我们来看一下下面这段代码:
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
void test1(int arr[])//相当于int *arr
//传的是数组首元素的地址
{printf("%d\n", sizeof(arr));//指针的大小在32位的编译器里是4个字节,在64位的编译器里是8个字节
}
void test2(char ch[])
{printf("%d\n", sizeof(ch));
}
int main()
{int arr[10] = { 0 };char ch[10] = { 0 };printf("%d\n", sizeof(arr));//数组有10个整型元素,所以结果是40printf("%d\n", sizeof(ch));//数组有10个字符型元素,结果是10test1(arr);test2(ch);return 0;
}


7.关系操作符
关系操作符:
> >= < <= !=(用于测试不相等) ==(用于测试相等)
注:不要把==和=混起来❗
✅==用于测试相等,=用于赋值。
8.逻辑操作符
📖逻辑操作符:
&&(逻辑与操作符)||(逻辑或操作符)
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
int main()
{int month = 0;scanf("%d",&month);if (month >= 3 && month <= 5){printf("春季\n");}if (month == 12 || month == 1 || month == 2){printf("冬季\n");}return 0;
}


🔎这里我们来看一道题:
#include <stdio.h>
int main()
{int i = 0, a = 0, b = 2, c = 3, d = 4;i = a++ && ++b && d++;//a++先得到0,对于&&操作符,前面为假,后面就不会计算了printf("a = %d\nb = %d\nc = %d\nd = %d\n", a, b, c, d);//1 2 3 4printf("i = %d\n", i);return 0;
}

📖这里,如果我们将a的值变为1,那么结果是什么呢?
#include <stdio.h>
int main()
{int i = 0, a = 1, b = 2, c = 3, d = 4;i = a++ && ++b && d++;//i = 1&&3&&4=1printf("a = %d\nb = %d\nc = %d\nd = %d\n", a, b, c, d);printf("i = %d\n", i);return 0;
}

💻同样的,如果我们这里将&&改为||,结果又是怎么样的呢?
#include <stdio.h>
int main()
{int i = 0, a = 1, b = 2, c = 3, d = 4;i = a++ || ++b || d++;//a++先得到结果1,表达式已经为真,后面不计算,b和d的值不变printf("a = %d\nb = %d\nc = %d\nd = %d\n", a, b, c, d);printf("i = %d\n", i);return 0;
}

9.条件操作符
📖条件操作符:
表达式1 ?表达式2:表达式3
✅唯一 一个三目操作符。
🧩表达式1为真,表达式2的结果为整个表达式的结果,表达式3不算.
🧩表达式1为假,表达式3的结果为整个表达式的结果,表达式2不算。
📝如果我们这里需要计算出两个数的最大值,按照前面所学的,我们可能会这样写:
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
int main()
{int a = 0;int b = 0;int m = 0;scanf("%d%d", &a, &b);if (a > b)m = a;elsem = b;printf("%d\n", m);return 0;
}

📖但是,这里如果我们用上条件操作符就会省很多事❗
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
int main()
{int a = 0;int b = 0;scanf("%d%d", &a, &b);printf("%d\n", a > b ? a : b);return 0;
}

10.逗号表达式
📖逗号表达式:
表达式1,表达式2,表达式3......
✅逗号表达式会从左往右依次执行,整个表达式的结果是最后一个表达式的结果。
📝那我们一起看看下面这段代码的运行结果是怎么样的呢?
#include <stdio.h>
int main()
{int a = 1;int b = 2;int c = (a > b, a = b + 5, a, b = a + 10);// 0 7 7 17printf("%d\n", c);return 0;
}

🔎示例:
#include <stdio.h>
int main()
{int a = 0;while (a > 0){a = get_val();count_val(a);}//上述while循环可以改写为以下的代码:/*while (a = get_val(), count_val(a), a>0){}*/return 0;
}
11.下标引用、函数调用和结构成员
📖下标引用操作符
[]
💻操作数为:一个数组名+一个索引值
#include <stdio.h>
int main()
{int arr[] = { 1,2,3,4,5,6,7,8,9,10 };//诸如0 1 2 3 4 5 6 7 8 9 称为索引值printf("%d\n", arr[2]);//[]-下标引用操作符return 0;
}

📖函数调用操作符
()
💻接受一个或多个操作数:第一个操作数是函数名,剩余的操作数是传递给函数的参数。
#include <stdio.h>
int add(int x, int y)
{return x + y;
}
int main()
{printf("%d", add(3, 4));//()函数调用操作符,最少有一个操作数为函数名return 0;
}

📖访问一个结构体成员
💻结构体.成员名 结构体指针->成员名
#include <stdio.h>
struct Book
{char name[20];int price;
};
int main()
{struct Book b = { "明解C语言",50 };printf("%s %d\n", b.name, b.price);return 0;
}

📖以下三种写法,得到的结果相同:
#include <stdio.h>
struct Book
{char name[20];int price;
};
void Print(struct Book* pb)
{printf("%s %d\n", (*pb).name, (*pb).price);printf("%s %d\n", pb->name, pb->price);
}
int main()
{struct Book b = { "明解C语言",50 };printf("%s %d\n", b.name, b.price);Print(&b);return 0;
}

12.表达式求值
📖表达式求值的顺序
一部分是由操作符的优先级和结合性来决定的。同样,有些表达式的操作数在求值的过程中可能需要转换为其他类型。
12.1隐式类型转换
C的整型算术运算总是至少以缺省整形类型的精度来进行。为了获取这个精度,表达式中的字符和短整型操作数之间被转换为普通整型,这样的类型转换为整型提升。
📖整型提升是按照变量的数据类型的符号位来提升的。
#include <stdio.h>
int main()
{char a = 5;//00000101//整型提升:00000000000000000000000000000101char b = 126;//01111110//整型提升:00000000000000000000000001111110char c = a + b;//c: 00000000000000000000000010000011//10000011//c整型提升:11111111111111111111111110000011//反码: 11111111111111111111111110000010//原码: 10000000000000000000000001111101//-(1+4+8+16+32+64)=-125printf("%d\n", c);return 0;
}

🔎整型提升的意义: 表达式的整型运算要在CPU的相应运算器件内执行,CPU内整型运算器(ALU)的操作数的字节长度。
一般就是int的字节长度,同时也是CPU的通用寄存器的长度。
因此,即使两个char类型的相加,在CPU执行时实际上也要先转换为CPU内整型操作数的标准长度。
通用CPU(general-purpose CPU)是难以直接实现两个8比特字节直接相加运算(虽然机器指令
中可能有这种字节相加指令)。所以,表达式中各种长度可能小于int长度的整型值,都必须先转 换为int或unsigned int,然后才能送入CPU去执行运算。
#include <stdio.h>
int main()
{char c = 1;printf("%d\n", sizeof(c));printf("%d\n", sizeof(+c));printf("%d\n", sizeof(-c));
//c只要参加表达式运算,就会发生整型提升return 0;
}

12.2算术转换
如果某个操作符的各个操作数属于不同的类型,那么除非其中一个操作数的转换为另一个操作数的类型,否则操作就无法进行。下面的层次体系称为寻常算术转换。
long double、double、float、unsigned long int、long int、unsigned int、int
如果某个操作数的类型在上面这个列表中排名较低,那么首先要转换为另外一个操作数的类型后执行运 算。 ❗警告: 但是算术转换要合理,要不然会有一些潜在的问题。
12.3 操作符的属性
📖复杂表达式的求值有三个影响的因素:
- 操作符的优先级
- 操作符的结合性
- 是否控制求值顺序。
✅两个相邻的操作符先执行哪个?取决于他们的优先级。如果两者的优先级相同,取决于他们的结合性。
🔎操作符的优先级
| 操作符 | 描述 | 结合性 | 是否控制求值顺序 |
|---|---|---|---|
| () | 聚组 | / | 否 |
| () | 函数调用 | 左结合性 | 否 |
| [ ] | 下标引用 | 左结合性 | 否 |
| . | 访问结构成员 | 左结合性 | 否 |
| -> | 访问结构指针成员 | 左结合性 | 否 |
| ++ | 后缀自增 | 左结合性 | 否 |
| - - | 后缀自减 | 左结合性 | 否 |
| ! | 逻辑反 | 右结合性 | 否 |
| ~ | 按位取反 | 右结合性 | 否 |
| + | 单目,表示正值 | 左结合性 | 否 |
| - | 单目,表示负值 | 右结合性 | 否 |
| ++ | 前缀自增 | 右结合性 | 否 |
| - - | 前缀自减 | 右结合性 | 否 |
| * | 间接访问 | 右结合性 | 否 |
| & | 取地址 | 右结合性 | 否 |
| sizeof | 取其长度,以字节表示 | 右结合性 | 否 |
| (类型) | 类型转换 | 右结合性 | 否 |
| * | 乘法 | 左结合性 | 否 |
| / | 除法 | 左结合性 | 否 |
| % | 整数取模 | 左结合性 | 否 |
| + | 加法 | 左结合性 | 否 |
| - | 减法 | 左结合性 | 否 |
| << | 左移位 | 左结合性 | 否 |
| >> | 右移位 | 左结合性 | 否 |
| > | 大于 | 左结合性 | 否 |
| >= | 大于等于 | 左结合性 | 否 |
| < | 小于 | 左结合性 | 否 |
| <= | 小于等于 | 左结合性 | 否 |
| == | 等于 | 左结合性 | 否 |
| != | 不等于 | 左结合性 | 否 |
| & | 位与 | 左结合性 | 否 |
| ^ | 位异或 | 左结合性 | 否 |
| I | 位或 | 左结合性 | |
| && | 逻辑与 | 左结合性 | 是 |
| II | 逻辑或 | 左结合性 | 是 |
| ?: | 条件操作符 | / | 是 |
| = | 赋值 | 左结合性 | 否 |
| += | 以…加 | 右结合性 | 否 |
| -= | 以…减 | 右结合性 | 否 |
| *= | 以…乘 | 右结合性 | 否 |
| /= | 以…除 | 右结合性 | 否 |
| %= | 以…取模 | 右结合性 | 否 |
| <<= | 以…左移 | 右结合性 | 否 |
| >>= | 以…右移 | 右结合性 | 否 |
| &= | 以…与 | 右结合性 | 否 |
| ^= | 以…异或 | 右结合性 | 否 |
| I= | 以…或 | 右结合性 | 否 |
| , | 逗号 | 左结合性 | 是 |
📖对于下面的这个表达式,表达式的计算顺序就不一定了!
a* b + c * d + e * f;


❌同样的,对于下面的这个表达式:
c + --c;
操作符的优先级只能决定自减的运算在+的运算的前面,但是我们并没有办法得知+操作符的左操作数的获取在右操作数之前还是之后求值,所以结果是不可预测的,是有歧义的。
📖📖📖我们写出的表达式如果不能通过操作符的属性确定唯一的计算路径,那这个表达式就是存在问题的,我们在写程序的时候,要避免写出这样的代码💯
好啦,关于操作符的知识点到这里就结束啦,后期会继续更新C语言的相关知识,欢迎大家持续关注、点赞和评论!❤️❤️❤️
相关文章:
初阶C语言-操作符详解(下)
🌞 “等春风得意,等时间嘉许!” 接下来,我们把操作符没学完的继续学完! 操作符详解 6.2sizeof和数组 7.关系操作符8.逻辑操作符9.条件操作符10.逗号表达式11.下标引用、函数调用和结构成员12.表达式求值12.1隐式类型转…...
reposync命令——下载yum仓库中全部的包到本地
reposync命令可以将远端yum仓库里面的包全部都下载到本地。这样构建自己的yum仓库,就不会遇到网络经常更新包而头疼的事情了。 reposync命令在软件包 yum-utils 里面,需要保证yum-utils已安装。 yum install yum-utils -y 常用参数 -r :指定…...
LC-杨辉三角
LC-杨辉三角 链接:https://leetcode.cn/problems/pascals-triangle/submissions/ 上图就是一个杨辉三角,每个数等于他左上角的数与右上角的数之和。 第一行就是一个1;第二行是两个1;第三行的2就是它肩膀上两个1之和,其余的类似。…...
Golang空结构体struct{}的作用是什么?
文章目录 占位符:通道标识:键集合:内存占用优化:总结: 在Go语言中,空结构体 struct{}是一种特殊的数据类型,它不占用任何内存空间。空结构体没有任何字段,也没有任何方法。尽管它看起…...
自然语言处理从入门到应用——LangChain:提示(Prompts)-[示例选择器(Example Selectors)]
分类目录:《自然语言处理从入门到应用》总目录 如果我们拥有大量的示例,我们可能需要选择在提示中包含哪些示例。ExampleSelector是负责执行此操作的类。 其基本接口定义如下所示: class BaseExampleSelector(ABC):"""Interf…...
【实战项目】c++实现基于reactor的高并发服务器
基于Reactor的高并发服务器,分为反应堆模型,多线程,I/O模型,服务器,Http请求和响应五部分 全局 反应堆模型 Channel 描述了文件描述符以及读写事件,以及对应的读写销毁回调函数,对应存储ar…...
Docker部署ElasticSearch7
前言 帮助小伙伴快速部署研发或测试环境进行学习测试。springboot版本需要与ElasticSearch版本想对应,不同版本api不一致,会产生异常调用的情况。 一、拉取镜像 这里选择固定版本7.15.2 docker pull docker.elastic.co/elasticsearch/elasticsearch:…...
【算法|数组】滑动窗口
算法|数组——滑动窗口 引入 给定一个含有 n 个正整数的数组和一个正整数 target 。 找出该数组中满足其和 ≥ target 的长度最小的 连续子数组 [numsl, numsl1, ..., numsr-1, numsr] ,并返回其长度**。**如果不存在符合条件的子数组,返回 0 。 示例…...
笙默考试管理系统-MyExamTest----codemirror(2)
笙默考试管理系统-MyExamTest----codemirror(2) 目录 一、 笙默考试管理系统-MyExamTest----codemirror 二、 笙默考试管理系统-MyExamTest----codemirror 三、 笙默考试管理系统-MyExamTest----codemirror 四、 笙默考试管理系统-MyExamTest---…...
一次面试下来Android Framework 层的源码就问了4轮
说起字节跳动的这次面试经历,真的是现在都让我感觉背脊发凉,简直被面试官折磨的太难受了。虽然已经工作了七年,但是也只是纯粹的在写业务,对底层并没有一个很深的认识,这次面试经历直接的让我感受到我和那些一线大厂开…...
知网期刊《中阿科技论坛》简介及投稿须知
知网期刊《中阿科技论坛》简介及投稿须知 主管单位:宁夏回族自治区科学技术厅 主办单位:宁夏回族自治区对外科技交流中心(中国一阿拉伯国家技术转移中心) 刊 期:月刊 国际刊号:ISSN 2096-7268 国内刊号:CN 64-…...
kafka是有序的吗?如何保证有序?
首先,Kafka无法保证消息的全局有序性,这是因为Kafka的设计中允许多个生产者并行地向同一个主题写入消息。而且,一个主题可能会被划分为多个分区,每个分区都可以在独立的生产者和消费者之间进行并行处理。因此,生产者将…...
centos 定时脚本检测tomcat是否启动,未启动情况下重新启动
编写脚本 tomcatMonitor.sh #!/bin/sh. /etc/profile . ~/.bash_profile#首先用ps -ef | grep tomcat 获得了tomcat进程信息,这样出来的结果中会包含grep本身, #因此通过 | grep -v grep 来排除grep本身,然后通过 awk {print $2}来打印出要…...
【Unity3D】消融特效
1 前言 选中物体消融特效中基于 Shader 实现了消融特效,本文将基于 Shader Graph 实现消融特效,两者原理一样,只是表达方式不同,另外,选中物体消融特效中通过 discard 丢弃片元,本文通过 alpha 测试丢弃片元…...
10.Eclipse配置Tomcat详细教程、如何使用Eclipse+tomcat创建并运行web项目
一、Tomcat的下载官网 -> 进入官网显示如图所示的界面,在下下载的是Tomcat9.0版本,你可以自己选一款 点击然后进入下面这个界面 最好是在你的D盘建立一个文件夹,把它解压在里面,文件夹名自己来吧,自己能知道里面装…...
MySQL索引1——索引基本概念与索引结构(B树、R树、Hash等)
目录 索引(INDEX)基本概念 索引结构分类 BTree树索引结构 Hash索引结构 Full-Text索引 R-Tree索引 索引(INDEX)基本概念 什么是索引 索引是帮助MySQL高效获取数据的有序数据结构 为数据库表中的某些列创建索引,就是对数据库表中某些列的值通过不同的数据结…...
2023-08-06力扣今日四题
链接: 剑指 Offer 59 - II. 队列的最大值 题意: 如题,要求O1给出数列的最大值 解: 类似滑动窗口 1 1 2 1 2用双端队列存储成2 2(每次从前面获取最大值,后面插入新数字)也就是第一个2覆盖了…...
Kubernetes入门 三、命令行工具 kubectl
目录 语法操作示例资源操作Pod 与集群资源类型与别名格式化输出 kubectl 是 Kubernetes 集群的命令行工具,通过它能够对集群本身进行管理,并能够在集群上进行容器化应用的安装和部署。 语法 使用以下语法从终端窗口运行 kubectl 命令: kub…...
18 | 基于DDD的微服务设计实例
为了更好地理解 DDD 的设计流程,这篇文章会用一个项目来带你了解 DDD 的战略设计和战术设计,走一遍从领域建模到微服务设计的全过程,一起掌握 DDD 的主要设计流程和关键点。 项目基本信息 项目的目标是实现在线请假和考勤管理。功能描述如下…...
router和route的区别
简单理解为,route是用来获取路由信息的,router是用来操作路由的。 一、router router是VueRouter的实例,通过Vue.use(VueRouter)和VueRouter构造函数得到一个router的实例对象,这个对象中是一个全局的对象,他包含了所…...
以下是对华为 HarmonyOS NETX 5属性动画(ArkTS)文档的结构化整理,通过层级标题、表格和代码块提升可读性:
一、属性动画概述NETX 作用:实现组件通用属性的渐变过渡效果,提升用户体验。支持属性:width、height、backgroundColor、opacity、scale、rotate、translate等。注意事项: 布局类属性(如宽高)变化时&#…...
spring:实例工厂方法获取bean
spring处理使用静态工厂方法获取bean实例,也可以通过实例工厂方法获取bean实例。 实例工厂方法步骤如下: 定义实例工厂类(Java代码),定义实例工厂(xml),定义调用实例工厂ÿ…...
python爬虫:Newspaper3k 的详细使用(好用的新闻网站文章抓取和解析的Python库)
更多内容请见: 爬虫和逆向教程-专栏介绍和目录 文章目录 一、Newspaper3k 概述1.1 Newspaper3k 介绍1.2 主要功能1.3 典型应用场景1.4 安装二、基本用法2.2 提取单篇文章的内容2.2 处理多篇文档三、高级选项3.1 自定义配置3.2 分析文章情感四、实战案例4.1 构建新闻摘要聚合器…...
linux 下常用变更-8
1、删除普通用户 查询用户初始UID和GIDls -l /home/ ###家目录中查看UID cat /etc/group ###此文件查看GID删除用户1.编辑文件 /etc/passwd 找到对应的行,YW343:x:0:0::/home/YW343:/bin/bash 2.将标红的位置修改为用户对应初始UID和GID: YW3…...
管理学院权限管理系统开发总结
文章目录 🎓 管理学院权限管理系统开发总结 - 现代化Web应用实践之路📝 项目概述🏗️ 技术架构设计后端技术栈前端技术栈 💡 核心功能特性1. 用户管理模块2. 权限管理系统3. 统计报表功能4. 用户体验优化 🗄️ 数据库设…...
音视频——I2S 协议详解
I2S 协议详解 I2S (Inter-IC Sound) 协议是一种串行总线协议,专门用于在数字音频设备之间传输数字音频数据。它由飞利浦(Philips)公司开发,以其简单、高效和广泛的兼容性而闻名。 1. 信号线 I2S 协议通常使用三根或四根信号线&a…...
解析两阶段提交与三阶段提交的核心差异及MySQL实现方案
引言 在分布式系统的事务处理中,如何保障跨节点数据操作的一致性始终是核心挑战。经典的两阶段提交协议(2PC)通过准备阶段与提交阶段的协调机制,以同步决策模式确保事务原子性。其改进版本三阶段提交协议(3PC…...
ui框架-文件列表展示
ui框架-文件列表展示 介绍 UI框架的文件列表展示组件,可以展示文件夹,支持列表展示和图标展示模式。组件提供了丰富的功能和可配置选项,适用于文件管理、文件上传等场景。 功能特性 支持列表模式和网格模式的切换展示支持文件和文件夹的层…...
书籍“之“字形打印矩阵(8)0609
题目 给定一个矩阵matrix,按照"之"字形的方式打印这个矩阵,例如: 1 2 3 4 5 6 7 8 9 10 11 12 ”之“字形打印的结果为:1,…...
归并排序:分治思想的高效排序
目录 基本原理 流程图解 实现方法 递归实现 非递归实现 演示过程 时间复杂度 基本原理 归并排序(Merge Sort)是一种基于分治思想的排序算法,由约翰冯诺伊曼在1945年提出。其核心思想包括: 分割(Divide):将待排序数组递归地分成两个子…...
