C语言基础之——操作符(上)

本篇文章,我们将展开讲解C语言中的各种常用操作符,帮助大家更容易的解决一些运算类问题。

这里提醒一下小伙伴们,本章知识会大量涉及到二进制序列,不清楚二进制序列的小伙伴,可以去阅读我的另一篇文章《数据在内存中的存储》学习了解。


目录

一.操作符分类

二.操作符讲解

1.算数操作符

2.移位操作符

(1)左移操作符 

(2)右移操作符

3.位操作符

(1)& 按位与

 (2)| 按位或

 (3)^ 按位异或

4.赋值操作符

5.单目操作符

总结


一.操作符分类

  • 算术操作符
  • 移位操作符
  • 位操作符
  • 赋值操作符
  • 单目操作符
  • 关系操作符
  • 逻辑操作符
  • 条件操作符
  • 逗号表达式
  • 下标引用,函数调用和结构成员

二.操作符讲解

由于操作符数量过多,所以本期文章我们将仅仅讲解前五种。

1.算数操作符

算数操作符包括 “+,-,*,/,%” 五个。较为简单,小伙伴们只需要注意以下几点:

  1. 除了“%”操作符之外,其他的几个操作符可以作用于整数和浮点数。
  2. 对于“/”操作符如果两个操作数都为整数,执行整数除法。只要有浮点数就执行浮点数除法
  3. “%”操作符的两个操作数必须为整数,返回的是整除后的余数

2.移位操作符

<<        左移操作符

>>        右移操作符

移位操作符的左边是要操作的数右边则是要移动的位数

要注意的是,移位操作符的操作数只能是整数,移动的对象则是整数的二进制序列

一个整型占四个字节,也就是32个bit位,要记住,整型在数据中存储的是二进制序列的补码,所以我们对整型的操作都是对其补码进行操作的。

(1)左移操作符 

左移操作符,顾名思义就是将整数的二进制序列向左边移动呗,那么它的规则是什么呢???

移位规则:

左边丢弃,右边补0

#include<stdio.h>
int main()
{int n = 6;//00000000 00000000 00000000 00000110-移动前//00000000 00000000 00000000 00001100-移动后int m = n << 1;printf("%d\n", n);printf("%d\n", m);return 0;
}

如上代码,将“6”的二进制序列向左移动一位,便得到了一个新的二进制序列。结果如下:

 由结果可以看出,移位操作符并不会改变操作数本身,而且细心的小伙伴们可以看出,向左移动一位不就相当于每一位的数字都“乘2”嘛,也就是将数字翻倍,移动n位,便翻2的n次方倍

负数的操作与之一样,就是小伙伴们千万不要忘记原码和补码之间的转换

(2)右移操作符

移位规则:

右移运算分为两种:

1.逻辑移位

左边用0填充,右边丢弃

2.算术移位

左边用原值的符号位填充,右边丢弃

不同的编译器会有不同的右移运算,但是我们平时所使用的绝大多数编译器都是算数右移

#include<stdio.h>
int main()
{int n = -15;//10000000 00000000 00000000 00001111-原码//11111111 11111111 11111111 11110000-反码//11111111 11111111 11111111 11110001-补码//11111111 11111111 11111111 11111000-移动后补码//11111111 11111111 11111111 11110111-移动后反码//10000000 00000000 00000000 00001000-移动后原码int m = n >> 1;printf("%d\n", n);printf("%d\n", m);return 0;
}

结果如下: 

 同左移类似,右移则是将数字折半,但是如果是奇数的话,结果则会是比小数小的最临近于小数的负数。比如-15的右移结果就是-8。

3.位操作符

&       按位与

|         按位或

^        按位异或

位操作符的操作数也必须是整数,也是对其二进制序列动手。

(1)& 按位与

假如我是一个企业高管,我现在需要程序员A程序员B一起来完成某个项目,这说明,A和B是必不可少的,他们两个少了谁这个项目都完不成。这便是按位与

口诀:同真则真,有假则假

#include<stdio.h>
int main()
{int a = 6;//00000000 00000000 00000000 00000110-补码int n = -15;//11111111 11111111 11111111 11110001-补码int m = a & n;//00000000 00000000 00000000 00000110-补码//11111111 11111111 11111111 11110001-补码//00000000 00000000 00000000 00000000-m的补码(重点)printf("%d\n", m);return 0;
}

我们习惯上将二进制序列的“1”视为真“0”视为假,a&n,便是两个二进制序列对应的每一位相与,从而得到一个新的二进制序列。

由上可知,m的二进制序列全为0,也就代表m的值为0,结果如下:

 (2)| 按位或

假如我又是一个企业高管,我现在需要程序员A程序员B来完成某个项目,这说明,A和B他们两个只要有一个能来做这个项目,就能成,如果一个都没有,就做不了。这便是按位或

口诀:同假则假,有真则真

#include<stdio.h>
int main()
{int a = 6;//00000000 00000000 00000000 00000110-补码int n = -15;//11111111 11111111 11111111 11110001-补码int m = a | n;//00000000 00000000 00000000 00000110-补码//11111111 11111111 11111111 11110001-补码//11111111 11111111 11111111 11110111-m的补码(要点)//11111111 11111111 11111111 11110110-m的反码//10000000 00000000 00000000 00001001-m的原码printf("%d\n", m);return 0;
}

对两个二进制序列的每一位相或,便得到m的补码,但是m的符号位为1,是负数,所以要转化为原码来读。

结果如下:

 (3)^ 按位异或

假如我还是一个企业高管……这个不好举例子了哈哈哈直接来看口诀:

相同为0,不同为1

#include<stdio.h>
int main()
{int a = 6;//00000000 00000000 00000000 00000110-补码int n = -15;//11111111 11111111 11111111 11110001-补码int m = a ^ n;//00000000 00000000 00000000 00000110-补码//11111111 11111111 11111111 11110001-补码//11111111 11111111 11111111 11110111-m的补码(重点)//11111111 11111111 11111111 11110110-m的反码//10000000 00000000 00000000 00001001-m的原码printf("%d\n", m);return 0;
}

将两个二进制序列的每一位相异或,结果如下:

4.赋值操作符

所谓赋值操作符,也就是我们经常使用的“ = ”将一个常量或者常量表达式赋给一个变量

int a = 1;//不是赋值,是创建之后的初始化

a = 5;//是赋值

int b = 2;

int c = 0;

c = a + b;//也是赋值

除了等号以外,还有一些常用的复合赋值操作符:

+=        -=        *=        /=        %=        >>=        <<=        &=        |=        ^=

这些符合赋值其实是两个运算式的合并,例如:

int a = 2;

a = a + 5 和 a += 5 是一样的效果,后者看起来会更加的简洁

5.单目操作符

所谓单目,也就是这种操作符的操作数只有一个

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

 这些操作符我们大多数都知道,下面我们仅仅讲解一下不是那么熟悉的:

sizeof        操作数的类型长度

sizeof 计算的结果是 size_t 类型

size_t 是无符号整型

对 size_t 类型的数据进行打印,可以使用%zd或%u

int a = 10;

printf("%zd",sizeof(a));

结果为4。 

~                  对一个数的二进制按位取反

int a = 0;

printf("%d",~a);

0的补码二进制序列为:

00000000 00000000 00000000 00000000

111111111 111111111 111111111 111111111//为负数,取原码

111111111 111111111 111111111 111111110//反码

10000000 00000000 00000000 00000001//原码

结果为-1。

*        间接访问操作符

int a = 10;

int* p = &a;

*p;//这时候我们的*就是对p进行解引用操作,*p是通过p中存放的地址,找到p指向的对象。

*p 其实就是a。

(类型)           强制类型转换

int a = (int )3.14;

3.14在编译器中会被默认为是double类型,如果直接将其初始化给int型的a,则会在后续操作中出现误差甚至错误,所以要将其强制类型转化为int型。

总结

本期关于操作符的知识讲解到这里就要结束啦,稍后博主将更新C语言基础之——操作符(下)来讲解剩余的操作符。

喜欢博主文章的小伙伴们不要忘记一键三连哦,我们下期再见!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/80693.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Spring Boot 整合 分布式搜索引擎 Elastic Search 实现 搜索、分页与结果过滤

文章目录 ⛄引言一、酒店搜索和分页⛅需求分析⚡源码编写 二、酒店结果过滤⌚需求分析⏰修改搜索业务 ✅效果图⛵小结 ⛄引言 本文参考黑马 分布式Elastic search Elasticsearch是一款非常强大的开源搜索引擎&#xff0c;具备非常多强大功能&#xff0c;可以帮助我们从海量数据…

Java进阶(7)——手动实现LinkedList 内部node类的实现 增删改查的实现 toString方法 源码的初步理解

目录 引出从ArrayList到Linkedlist手动实现ArrayList从ArrayList到LinkedList 总体设计Node类Node的方法&#xff1a;根据index找node 增删改查的实现增加元素删除元素修改元素查询元素 toString方法完整代码List接口类LinkedList的实现测试类 总结 引出 1.linkedList的节点&am…

实战项目ssm权限系统 3-总结篇,权限模块保护业务模块

一 工程模块介绍 1.1 工程模块关系 在业务微服务模块中引入安全认证模块&#xff0c;起到对业务模块的认证授权保护

爬虫:绕过5秒盾Cloudflare和DDoS-GUARD

本文章仅供技术研究参考&#xff0c;勿做它用&#xff01; 5秒盾的特点 <title>Just a moment...</title> 返回的页面中不是目标数据&#xff0c;而是包含上面的代码&#xff1a;Just a moment... 或者第一次打开网页的时候&#xff1a; 这几个特征就是被Cloud…

汽车电子笔记之:AUTOSAR方法论及基础概念

目录 1、AUTOSAR方法论 2、AUTOSAR的BSW 2.1、MCAL 2.2、ECU抽象层 2.3、服务层 2.4、复杂驱动 3、AUTOSAR的RTE 4、AUTOSAR的应用层 4.1、SWC 4.2、AUTOSAR的通信 4.3、AUTOSAR软件接口 1、AUTOSAR方法论 AUTOSAR为汽车电子软件系统开发过程定义了一套通用的技术方法…

强化历程5-Java并发系列(2023.8.23)

文章目录 强化历程5-Java并发系列(2023.8.23)1 Java多线程1.1 Java中多线程有几种实现方式&#xff1f;1.2 那么Runnable和Callable都可以实现多线程&#xff0c;他们有什么区别?1.3 采用实现Runnable和Callable接口方式和采用继承Thread类方式各有什么好处?1.4 Java如何停止…

无涯教程-PHP - 移除的扩展

以下扩展已从PHP 7开始删除- eregmssqlmysqlsybase_ct 以下SAPI已从PHP 7开始删除- aolserverapacheapache_hooksapache2filtercaudiumcontinuityisapimilternsapiphttpdpi3webroxenthttpdtuxwebjames PHP - 移除的扩展 - 无涯教程网无涯教程网提供以下扩展已从PHP 7开始删除…

docker实践作业

1.安装docker服务&#xff0c;配置镜像加速器 2.下载系统镜像&#xff08;Ubuntu、 centos&#xff09; 3.基于下载的镜像创建两个容器 &#xff08;容器名一个为自己名字全拼&#xff0c;一个为首名字字母&#xff09; 4.容器的启动、 停止及重启操作 5.怎么查看正在运行的容器…

Android GreenDao数据库升级(附Demo)

前言 大家好久不见&#xff0c;一转眼马上八月份下旬了&#xff0c;最近由于工作比较忙&#xff0c;没时间给大家更新博文。百忙之中抽出时间&#xff0c;给大家来更新一篇关于GreenDao3数据库的升级。 关于GreenDao的详细介绍以及一些逻辑性的增、删、改、查等&#xff0c;可以…

Python快速入门体验

Python快速入门体验 一、环境信息1.1 硬件信息1.2 软件信息 二、Conda安装2.1 Conda介绍2.1.1 Conda简介2.1.2 Conda、Anaconda及Miniconda及的关系 2.2 Conda安装包下载2.2.1 Miniconda下载2.2.2 Anconda下载 2.3 Conda安装2.3.1 Miniconda安装2.3.2 Anconda安装 2.4 Conda初始…

RabbitMQ---work消息模型

1、work消息模型 工作队列或者竞争消费者模式 在第一篇教程中&#xff0c;我们编写了一个程序&#xff0c;从一个命名队列中发送并接受消息。在这里&#xff0c;我们将创建一个工作队列&#xff0c;在多个工作者之间分配耗时任务。 工作队列&#xff0c;又称任务队列。主要思…

为什么网络互联地址设置为30位地址

对于点对点链路&#xff0c;为了节约IPv4地址&#xff0c;一般为其分配/30地址块&#xff0c;这样包含4个地址&#xff1a;最小地址作为网络地址&#xff0c;最大地址作为广播地址&#xff0c;剩余两个可分配地址&#xff0c;分配给链路两端的接口&#xff0c;这是最普遍的方法…