C语言学习NO.-操作符(二)二进制相关的操作符,原码、反码、补码是什么,左移右移操作符、按位与,按位或,按位异或,按位取反

news2024/11/23 19:14:00

一、操作符的分类

操作符的分类

  • 算术操作符: + 、- 、* 、/ 、%
  • 移位操作符: << >>
  • 位操作符: & | ^ `
  • 赋值操作符: = 、+= 、 -= 、 *= 、 /= 、%= 、<<= 、>>= 、&= 、|= 、^=
  • 单⽬操作符: !、++、–、&、*、+、-、~ 、sizeof、(类型)
  • 关系操作符: > 、>= 、< 、<= 、 == 、 !=
  • 逻辑操作符: && 、||
  • 条件操作符: ? :
  • 逗号表达式: ,
  • 下标引用: [ ]
  • 函数调用: ( )
  • 结构成员访问: . 、->

在C语言学习NO.2中操作符中已经讲过算术操作符、赋值操作符、逻辑操作符、条件操作符和部分的单目操作符。
也在NO.2操作符中介绍了整型常量的不同进制关系,今天来更详细地介绍操作符中和⼆进制有关系的操作符。

二、⼆进制和进制转换

0,10-70-90-9,a-f
二进制八进制十进制十六进制

详细见C语言学习NO.2-操作符的整形常量的不同进制关系。

三、原码、反码、补码

整数的2进制表示方法有三种,即原码、反码和补码
2进制序列中,有符号整数的三种表示方法均有**符号位(最左侧数字)数值位(除符号位剩余位)**两部分。

  • 符号位都是用0表示“正”,用1表示“负”。
  • 正整数的原、反、补码都相同:
    • 直接将数值按照正负数的形式翻译成⼆进制得到的就是原码;
    • 正整数的原、反、补码相同。
  • 负整数的三种表达方式各不相同:
    • 原码:直接将数值按照正负数的形式翻译成⼆进制得到的就是原码。;
    • 反码:将原码的符号位不变,其他位依次按位取反就可以得到反码;
    • 补码:反码+1就得到补码。;
    • 反码得到原码也是可以使用:取反,+1的操作。

对于整形来说:数据存放内存中其实存放的是补码。
为什么呢? 在计算机系统中,数值一律用补码来表示和存储。原因在于使用补码可以将符号位和数值域统一处理;同时,加法和减法也可以统一处理(CPU只有加法器)此外,补码与原码相互转换,其运算过程是相同的,不需要额外的硬件电路。

四、移位操作符

<<左移操作符
>>右移操作符
注:移位操作符的操作数只能是整数

(一)左移操作符

移位规则:左边抛弃、右边补0.

当num为正数时:

#include <stdio.h>

int main()
{
	int num = 10;
  int n = num<<1;
  printf("n = %d\n",n);
  printf("num = %d\n",num);
  return 0;
}
  

image.png
现在让我们来看看一下n值是怎么求得的。
由于<<是移动二进制位的符号,我们首先把整型的num转化为二进制。
int拥有四个字节,每个字节有8个比特位,转换为二进制后为:
1010 →→→ 是二进制为的10。1*2^3 +0+1*2^1+0 = 10;
int有32个比特位,所以转换二进制后应为:
00000000 00000000 00000000 00001010;
<<是指二进制位向左移动一位,右侧补上一位。


同理,当num为负数时:

#include <stdio.h>

int main()
{
	int num = -10;
  int n = num<<1;
  printf("n = %d\n",n);
  printf("num = %d\n",num);
  return 0;
}
  

image.png
补码转换到原码,两种方法:

  1. 补码 减一 到反码, 反码 取反 到 原码(符号位不变)。
  2. 补码 取反 加一 得到 原码。(符号位不变)


(二)右移操作符

移位规则:右移运算分两种:

  1. 逻辑右移:左边用0填充,右边丢弃;
  2. 算术右移:左边用原值符号位填充,右边丢弃;
  3. 采用哪种右移取决于编译器设置。

当num为正整数时:

#include <stdio.h>
int main()
{
 int num = 10;
 int n = num>>1;
 printf("n= %d\n", n);
 printf("num= %d\n", num);
 return 0; 
}

image.png

当num为负整数:

#include <stdio.h>
int main()
{
 int num = -10;
 int n = num>>1;
 printf("n= %d\n", n);
 printf("num= %d\n", num);
 return 0; 
}

image.png

(三)总结

  • 数据存放内存存放的是补码
  • 左移操作符、右移操作符操作移动的是数据存放的补码
  • 打印出来的值是原码

五、位操作符:&、|、^、~

&: //按位与 双目操作符 a&b
//两个数的二进制位只有对应的二进制位都为1,才为1.否则为0
|: //按位或 双目操作符 a|b
//两个数的二进制位只有对应的二进制位都为0,才为0.否则为1
^: //按位异或 双目操作符 a^b
//两个数的二进制位对应的二进制位:相同为0,相异为1
~: //按位取反 单目操作符 ~a
//整数的二进制位全部取反
注:他们的操作数都必须是整数

#include <stdio.h>

int main()
{
 int num1 = -3;//内存中是补码,打印出的是原码
 int num2 = 5;
 //-3:10000000 00000000 00000000 00000011 原码
 //		11111111 11111111 11111111 11111101 补码
 // 5:00000000 00000000 00000000 00000101 原码 正整数相同
 //		00000000 00000000 00000000 00000101 补码

//先操作看补码,后打印出的值看原码
 
 printf("%d\n", num1 & num2);
 //只有对应的二进制位都为一,才为1,否则为0
 //00000000 00000000 00000000 00000101  → 补码 符号位为正
 //00000000 00000000 00000000 00000101  → 原码 5
 
 printf("%d\n", num1 | num2);
//两个数的二进制位只有对应的二进制位都为0,才为0.否则为1
//11111111 11111111 11111111 11111101  → 补码 符号位为负
//10000000 00000000 00000000 00000011  → 原码-3
 
 printf("%d\n", num1 ^ num2);
//两个数的二进制位对应的二进制位:相同为0,相异为1
//11111111 11111111 11111111 11111000  → 补码 符号位为负
//10000000 00000000 00000000 00001000  → 原码 +8
 
 printf("%d\n", ~0);
//整数的二进制位全部取反
//11111111 11111111 11111111 11111111  → 补码 符号位为负
//10000000 00000000 00000000 00000001  → 原码-1
 
 return 0; 
}

image.png

(一)^ 按位异或交换两个数的小技巧:

a^a = 0;
0^a = a;
**不能创建临时变量(第三个变量),实现两个数的交换。 **

#include <stdio.h>
//不能创建临时变量(第三个变量),实现两个数的交换。
int main()
{
 int a = 10;
 int b = 20;
 a = a^b;//
 
 b = a^b;
 //b = a^b = a^b^b = 0^a = a
 
 a = a^b;
 // a = a^b =  a^b^a = b^0 =b
 
 printf("a = %d b = %d\n", a, b);
 return 0; 
}

(二)练习

练习1:整数在内存中1的个数

编写代码实现:求一个整数存储在内存中的⼆进制中1的个数。

为了方便多组测试添加的while(scanf(“%d”,&num) != EOF)可删去。

#include <stdio.h>
//求一个整数存储在内存中的⼆进制中1的个数。
int main()
{
    int num = 10;
    while(scanf("%d",&num) != EOF)//为了方便多次测试不同数值
    {
        int i = 0;
        int count = 0;
        for (i = 1; i <= 32; i++) {		//32个比特位
            if (1 == ((num >> i) & 1)) 
            //num是不会因为>>改变num的值的,所以每向右移 i 位,
            //让最右侧的值 & 1,来判断是否为1,为1 则进入if语句,count++
                count++;
        }//该语句要循环32次,知道num的32个比特位都移动至最右侧
        printf("%d\n", count);
    }
    return 0;
}
#include <stdio.h>
//求一个整数存储在内存中的⼆进制中1的个数。
int main()
{
    int num = 10;
    while(scanf("%d",&num) != EOF)//为了方便多次测试不同数值
    {
        int i = 0;
        int count = 0;
        for (i = 1; i <= 32; i++)
        {
            count += num & 1;
            num = num >> 1;//同上一代码原理一致,当num最右侧值和1相同时,
            //num & 1是1,否则为0,以此count进行计数的功能,
            //每次循环num的值都会变化,因此是 num >> 1,而不是num >> i
            //同样需要循环32次进行判断
        }
        printf("%d\n", count);
    }
    return 0;
}
#include <stdio.h>
int main()
{
 int num = -1;
 int i = 0;
 int count = 0;//计数
 for(i=0; i<32; i++)//循环32次
 {
 if( num & (1 << i) )//上列是num 右移,现在思维转换,将1 的二进制左移进行判断
 count++; //计数
 }	//原理一致
 printf("⼆进制中1的个数 = %d\n",count);
 return 0; }

上列方法都必须循环32次的,思考是否可以优化

#include <stdio.h>
int main()
{
 int num = -1;
 int count = 0;//计数
 while(num)//非0为真,0为假
 {
 count++;
 num = num&(num-1);//按位与
 //&按位与: 对应的二进制位都为1,才为1,否则为0
 //&运算符可以用来检查一个数的二进制表示中是否包含1
 //通过将原数与1进行位与运算,并将结果加到计数器上,可以计算出二进制中1的个数。 
 }
 printf("⼆进制中1的个数 = %d\n",count);
 return 0;
}

练习2:⼆进制位置0或者置1

编写代码将13⼆进制序列的第5位修改为1,然后再改回0

//13的2进制序列: 00000000000000000000000000001101
//将第5位置为1后:00000000000000000000000000011101
//将第5位再置为0:00000000000000000000000000001101
#include <stdio.h>
//编写代码将13⼆进制序列的第5位修改为1,然后再改回0
int main()
{
 int a = 13;
//13的2进制序列: 00000000000000000000000000001101
 
 a = a | (1<<4);
 printf("a = %d\n", a);
 //1左移4个二进制位00000000 00000000 00000000 00000001
 //00000000 00000000 00000000 00010000  →  第5位变为1
 //‘|’按位或 :两个数对应的二进制位都为0才为0,否则为1
 //a修改为00000000 00000000 00000000 00011101  →  16+8+4+1=29

 a = a & ~(1<<4);
 printf("a = %d\n", a);
 //~按位取反 
 //11111111 11111111 11111111 11101111 
 //& 按位与 : 对应的二进制位都为1才为1,否则为0 
 //00000000 00000000 00000000 00011101  修改第五位的a的2进制序列
 //00000000 00000000 00000000 00001101  第五位改回为0 → 13

 return 0; 
}

image.png

六、下标访问[]、函数调用()

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

操作数:一个数组名 + 一个索引值

int arr[10];//创建数组
arr[9] = 10;//实用下标引用操作符。
[]的两个操作数是arr和9。

(二)函数调用操作符

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

#include <stdio.h>
void test1()
{
 printf("hehe\n");
}
void test2(const char *str)
{
 printf("%s\n", str);
}
int main()
{
 test1(); 							//这⾥的()就是作为函数调用操作符。
 test2("hello bit.");		//这⾥的()就是函数调用操作符。
 return 0; 
} 

七、结构成员访问操作符

(一)结构体

C语言已经提供了内置类型,如:char、short、int、long、float、double等,但是只有这些内置类型还是不够的。
假设我想描述学生,描述一本书,这是单一的内置类型是不行的。描述一个学生需要 名字、年龄、学号、身高、体重等;描述一本书需要 作者、出版社、定价等。C语言为了解决这个问题,增加了结构体这种自定义的数据类型,让程序员可以自己创造适合的类型。

结构是一些值的集合,这些值称为成员变量。
结构的每个成员可以是不同类型的变量,如: 标量、数组、指针,甚至是其他结构体。

结构的声明:

struct tag
{
 member-list;
}variable-list;

描述一个学生:

struct Stu
{
 char name[20];		//名字
 int age;					//年龄 1234
 char sex[5];			//性别
 char id[20];			//学号
}; 								//分号不能丢

结构体变量的定义和初始化

//代码1:变量的定义
struct Point
{
  int x;
	int y;
}p1; 									//声明类型的同时定义变量p1
struct Point p2; 			//定义结构体变量p2

//代码2:初始化。
struct Point p3 = {10, 20};

struct Stu 									//类型声明 
{
   char name[15];							//名字
   int age; 									//年龄
};

struct Stu s1 = {"zhangsan", 20};//初始化
struct Stu s2 = {.age=20, .name="lisi"};		//指定顺序初始化

//代码3
struct Node
{
	int data;
	struct Point p;
	struct Node* next; 
}n1 = {10, {4,5}, NULL}; 								//结构体嵌套初始化

struct Node n2 = {20, {5, 6}, NULL};		//结构体嵌套初始化

(二)结构成员访问操作符

结构体成员的直接访问

结构体成员的直接访问是通过点操作符(.)访问的。点操作符接受两个操作数。如下所示:

#include <stdio.h>
struct Point
{
 int x;
 int y;
}p = {1,2};
int main()
{
 printf("x: %d y: %d\n", p.x, p.y);
 return 0; 
}

//使用⽅式:结构体变量.成员名 
//					p.x, p.y

结构体成员的间接访问

有时候我们得到的不是一个结构体变量,而是得到了一个指向结构体的指针。如下所示:

#include <stdio.h>
struct Point
{
 int x;
 int y;
};
int main()
{
 struct Point p = {3, 4};
 struct Point *ptr = &p;
 ptr->x = 10;
 ptr->y = 20;
 printf("x = %d y = %d\n", ptr->x, ptr->y);
 return 0; 
}

//使用⽅式:结构体指针->成员名
//				 ptr->x, ptr->y

综合举例:

#include <stdio.h>
#include <string.h>

struct Stu
{
 char name[15];					//名字
 int age; 							//年龄
};

void print_stu(struct Stu s)
{
 printf("%s %d\n", s.name, s.age);
}

void set_stu(struct Stu* ps)
{
 strcpy(ps->name, "李四");
 ps->age = 28; 
}
 
int main()
{
 struct Stu s = { "张三", 20 };
 print_stu(s);
 set_stu(&s);
 print_stu(s);
 return 0; 
}

八、表达式求值

(一)整型提升 (隐式类型转换)

C语言中整型算术运算总是至少以缺省整型类型的精度来进行的。
为了获得这个精度,表达式中的字符和短整型操作数在使用之前被转换为普通整型,这种转换称为整型提升
整型提升的意义:

表达式的整型运算要在CPU的相应运算器件内执行,CPU内整型运算器(ALU)的操作数的字节长度一般就是int的字节长度,同时也是CPU的通用寄存器的长度。
因此,即使两个char类型的相加,在CPU执行时实际上也要先转换为CPU内整型操作数的标准长度。
通用CPU(general-purpose CPU)是难以直接实现两个8比特字节直接相加运算(虽然机器指令中 可能有这种字节相加指令)。所以,表达式中各种长度可能⼩于int长度的整型值,都必须先转换为 int或unsigned int,然后才能送⼊CPU去执行运算。

//实例1
char a,b,c;
...
a = b + c;

b和c的值被提升为普通整型,然后再执执行加法运算
加法运算完成之后,结果将被截断,然后再存储于a中。
如何进行整体提升呢?

  1. 有符号整数提升是按照变量的数据类型的符号位来提升的
  2. 无符号整数提升,高位补0
//负数的整形提升
char c1 = -1; 
//变量c1的⼆进制位(补码)中只有8个⽐特位:1111111
//因为 char 为有符号的 char
//所以整形提升的时候,⾼位补充符号位,即为1 
//提升之后的结果是:
//								11111111111111111111111111111111

//正数的整形提升
char c2 = 1; 
//变量c2的⼆进制位(补码)中只有8个⽐特位:00000001
//因为 char 为有符号的 char
//所以整形提升的时候,⾼位补充符号位,即为0 
//提升之后的结果是:
//								00000000000000000000000000000001

//⽆符号整形提升,⾼位补0

(二)算术转换

如果某个操作符的各个操作数属于不同的类型,那么除非其中一个操作数的转换为另一个操作数的类型,否则操作就无法进行。下面的层次体系称为寻常算术转换。

long double
double
float
unsigned long int
long int
unsigned int
int

如果某个操作数的类型在上面这个列表中排名靠后,那么首先要转换为另外一个操作数的类型后执行运算。
小的数据类型转换成大的数据类型。再执行运算。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

基于YOLOv8深度学习的西红柿成熟度检测系统【python源码+Pyqt5界面+数据集+训练代码】目标检测、深度学习实战

《博主简介》 小伙伴们好&#xff0c;我是阿旭。专注于人工智能、AIGC、python、计算机视觉相关分享研究。 ✌更多学习资源&#xff0c;可关注公-仲-hao:【阿旭算法与机器学习】&#xff0c;共同学习交流~ &#x1f44d;感谢小伙伴们点赞、关注&#xff01; 《------往期经典推…

linux下查看进程资源ulimit

ulimit介绍与使用 ulimit命令用于查看和修改进程的资源限制。下面是ulimit命令的使用方法&#xff1a; 查看当前资源限制&#xff1a; ulimit -a 这将显示当前进程的所有资源限制&#xff0c;包括软限制和硬限制。查看或设置单个资源限制&#xff1a; ulimit -<option> …

2023年【陕西省安全员C证】新版试题及陕西省安全员C证复审模拟考试

题库来源&#xff1a;安全生产模拟考试一点通公众号小程序 陕西省安全员C证新版试题参考答案及陕西省安全员C证考试试题解析是安全生产模拟考试一点通题库老师及陕西省安全员C证操作证已考过的学员汇总&#xff0c;相对有效帮助陕西省安全员C证复审模拟考试学员顺利通过考试。…

C++初阶-stack的使用与模拟实现

stack的使用与模拟实现 一、stack的介绍和使用二、stack的使用三、stack的模拟实现3.1 成员变量3.2 成员函数3.2.1 push入栈3.2.2 pop出栈3.2.3 返回栈顶数据3.2.4 返回栈的大小3.2.5 判断栈是否为空 四、完整代码4.1 stack.h4.2 test.h 一、stack的介绍和使用 1.stack是一种容…

03_Web开发基础之综合应用

web开发基础之综合使用 学习目标和内容 1、能够描述jQuery的作用 2、能够使用jQuery的选择器获取元素 3、能够使用jQuery对HTML标签元素注册事件 4、能够使用jQuery对HTML元素的属性进行操作 5、能够描述Bootstrap的作用 6、能够使用Bootstrap创建简单网页 7、能够描述AJAX的作…

Java----新手一步一步安装 Java 语言开发环境

查看原文 文章目录 一、基于 Windows 10 系统 安装配置 JDK8二、基于 CentOS7 系统安装配置 JDK8 一、基于 Windows 10 系统 安装配置 JDK8 &#xff08;1&#xff09;打开 JDK下载网站&#xff0c;根据系统配置选择版本&#xff0c;这里选择windows 64位的版本&#xff0c;点…

人工智能-A*算法-最优路径搜索实验

上次学会了《A*算法-八数码问题》&#xff0c;初步了解了A*算法的原理&#xff0c;本次再用A*算法完成一个最优路径搜索实验。 一、实验内容 1. 设计自己的启发式函数。 2. 在网格地图中&#xff0c;设计部分障碍物。 3. 实现A*算法&#xff0c;搜索一条最优路径。 二、A*算法实…

DDA 算法

CAD 算法是计算机辅助设计的算法&#xff0c;几何算法是解决几何问题的算法 CAD 算法是指在计算机辅助设计软件中使用的算法&#xff0c;用于实现各种设计和绘图功能&#xff0c;CAD 广泛应用于建筑、机械、电子等领域&#xff0c;可以大大提高设计效率和精度 绘图算法是 CAD…

关于多重背包的笔记

多重背包可以看作01背包的拓展&#xff0c; 01背包是选或者不选。多重背包是选0个一直到选s个。 for (int i 1; i < n; i) {for (int j m; j > w[i]; --j){f[j] max(f[j], f[j - 1*w[i]] 1*v[i], f[j - 2*w[i]] 2*v[i],...f[j - s*w[i]] s*v[i]);} } 由上述伪代码…

FL Studio2024mac电脑版本下载步骤教程

FL Studio2024是款专业的音频录制编辑软件&#xff0c;可以针对作曲者的要求编辑出不同音律的节奏&#xff0c;例如鼓、镲、锣、钢琴、笛、大提琴等等任何乐器的节奏律动。FL Studio目前在中国已经受到广大制作人喜爱&#xff0c;使用它制作的音乐作品也已经数不胜数&#xff0…

Leetcode的AC指南 —— 链表:24. 两两交换链表中的节点

摘要&#xff1a; Leetcode的AC指南 —— 链表&#xff1a;24. 两两交换链表中的节点。题目介绍&#xff1a;给你一个链表&#xff0c;两两交换其中相邻的节点&#xff0c;并返回交换后链表的头节点。你必须在不修改节点内部的值的情况下完成本题&#xff08;即&#xff0c;只能…

计算机服务器中了360后缀勒索病毒怎么处理,勒索病毒解密数据恢复

网络技术的不断发展与应用&#xff0c;越来越多的企业开始走向数字化办公模式&#xff0c;极大地方便了企业的生产运营。但随之而来的网络安全威胁也在不断增加&#xff0c;在本月&#xff0c;云天数据恢复中心陆续接到很多企业的求助&#xff0c;企业的计算机服务器遭到了360后…

文心一言 VS 讯飞星火 VS chatgpt (158)-- 算法导论12.3 5题

五、用go语言&#xff0c;假设为每个结点换一种设计&#xff0c;属性 x.p 指向 x 的双亲&#xff0c;属性 x.succ 指向 x 的后继。试给出使用这种表示法的二叉搜索树 T 上 SEARCH、INSERT 和DELETE 操作的伪代码。这些伪代码应在 O(h) 时间内执行完&#xff0c;其中 h 为树 T 的…

函数式编程 h函数

<template><div><table border><tr><th>id</th><th>name</th><th>age</th><th>操作</th></tr><tr v-for"item in list" :key"item.id"><td>{{ item.id }}<…

wsl kafka的简单应用

安装并配置单机版kafka所需环境 wsl2 环境可用性较高&#xff0c;如下介绍在该环境中安装单机版本kafka的详细过程。 启动命令行工具启动wsl&#xff1a;wsl --user root --cd ~&#xff0c;&#xff08;以root用户启动&#xff0c;进入wsl后当前路径为~“用户主目录”&#…

3.3【窗口】窗口的几何形状(二,窗口属性)

写在前面 应用程序使用窗口来显示内容。一些属性决定了窗口及其内容的大小和位置。其他属性决定了窗口内容的外观和解释。 了解窗口属性引用的两个坐标系非常重要。如果你知道你正在使用的坐标系,那么为你的窗口属性选择设置值会容易得多,并且会更有意义。 一,显示相关属…

SpringBoot零基础入门到项目实战——学习路线规划与目录结构

文章目录 第一部分&#xff1a;Spring Boot基础第二部分&#xff1a;Web开发与RESTful API第三部分&#xff1a;数据访问与持久化第四部分&#xff1a;安全与身份验证第五部分&#xff1a;高级主题第六部分&#xff1a;测试总结与扩展实战项目练习 &#x1f389;欢迎来到Spring…

Storage engine MyISAM is disabled (Table creation is disallowed)

如何解决Storage engine MyISAM is disabled (Table creation is disallowed&#xff09; 在开发中&#xff0c;需要把mysql5.7的数据库&#xff0c;迁移到mysql8.0 的阿里云数据库上 把Mysql5.7的数据导入到8.0时&#xff0c;出现 解决方法 1、使用指令找出那些表是MyISAM引擎…

求解最大子段和问题

求解最大子段和问题。 对于给定序列a1,a2,a3……an,寻找它的某个连续子段&#xff0c;使得其和最大。如( -2,11,-4,13,-5,-2 )最大子段是{ 11,-4,13 }其和为20。 要求&#xff1a;分别用教材所给的三种方法求解&#xff08;简单方法、分治法、动态规划&#xff09;&#xff0…

【Redis】AOF 基础

因为 Redis AOF 的实现有些绕, 就分成 2 篇进行分析, 本篇主要是介绍一下 AOF 的一些特性和依赖的其他函数的逻辑,为下一篇 (Redis AOF 源码) 源码分析做一些铺垫。 AOF 全称: Append Only File, 是 Redis 提供了一种数据保存模式, Redis 默认不开启。 AOF 采用日志的形式来记…