当前位置: 首页 > news >正文

【C语言】8.C语言操作符详解(1)

文章目录

  • 1.操作符的分类
  • 2.⼆进制和进制转换
  • 3.原码、反码、补码
  • 4.移位操作符
    • 4.1 左移操作符
    • 4.2 右移操作符
  • 5.位操作符:&、|、^、~
    • 5.1 &:按位与
    • 5.2 |:按位或
    • 5.3 ^:按位异或
    • 5.4 ~:按位取反
    • 5.5 例题
      • 例题1
      • 例题2
      • 例题3
      • 例题4


1.操作符的分类

  1. 算术操作符: +-*/%

  2. 移位操作符: << ,>>

  3. 位操作符: &, |, ^

  4. 赋值操作符: =+=-=*=/=%=<<=>>=&=|=^=

  5. 单目操作符: ++--&*+-~sizeof(类型)

  6. 关系操作符: >>=<<===!=

  7. 逻辑操作符: &&||

  8. 条件操作符: ? :

  9. 逗号表达式: ,

  10. 下标引用: []

  11. 函数调用: ()

  12. 结构成员访问: .->


2.⼆进制和进制转换

进制转换的教程网上比比皆是,而且视频看起来肯定比文字清晰。这里就不过多赘述了。

我们需要掌握一下几个:

  1. 2进制10进制
  2. 10进制2进制
  3. 2进制8进制和16进制
  4. 8进制和16进制2进制
  5. 8进制16进制
  6. 16进制8进制

3.原码、反码、补码

整数的2进制表示方法有三种:

  1. 原码

  2. 反码

  3. 补码

有符号整数的三种表示方法均有符号位和数值位两部分,2进制序列中,最高位的1位是被当做符号位,剩余的都是数值位。

符号位都是用0表示“正”,用1表示“负”。

正整数的原、反、补码都相同。

原码:直接将数值按照正负数的形式翻译成二进制得到的就是原码。

反码:正数反码和原码一样。

补码:正数补码和原码一样。

负整数的三种表示方法各不相同。

原码:直接将数值按照正负数的形式翻译成二进制得到的就是原码。(注意符号位)

反码:将原码的符号位不变,其他位依次按位取反就可以得到反码。

补码:反码+1就得到补码。


4.移位操作符

只有整数才能运用移位操作符。

4.1 左移操作符

例1:

#include <stdio.h>
int main()
{int num = 10;int n = num<<1;//10补码:  00000000 00000000 00000000 00001010//10<<1补码:0 00000000 00000000 00000000 00010100//这里最前面的0被丢掉了printf("n= %d\n", n);printf("num= %d\n", num);return 0;
}

打印:

n= 20
num= 10

例2:

#include <stdio.h>
int main()
{int num = -1;int n = num<<1;//-1原码: 10000000 00000000 00000000 00000001//-1反码: 11111111 11111111 11111111 11111110//-1补码: 11111111 11111111 11111111 11111111
//-1<<1补码:0 11111111 11111111 11111111 11111110//这里最前面的0被丢掉了printf("n= %d\n", n);printf("num= %d\n", num);return 0;
}

打印:

n= -2
num= -1

4.2 右移操作符

右移运算分两种:

  1. 逻辑右移:左边用0填充,右边丢弃。

  2. 算术右移:左边用原该值的符号位填充(正数左边补0,负数左边补1),右边丢弃。

    右移是算术右移还是逻辑右移是取决于编译器的。通常采用的都是算数右移。

例1:

#include <stdio.h>
int main()
{int num = -10;//-10原码:10000000 00000000 00000000 00001010//-10反码:11111111 11111111 11111111 11110101//-10补码:11111111 11111111 11111111 11110110//逻辑右移:01111111 11111111 11111111 11111011 |0(这个0被丢弃了)//算术右移:11111111 11111111 11111111 11111011 |0(这个0被丢弃了)//这里采用算数右移//右移补码:11111111 11111111 11111111 11111011//右移反码:10000000 00000000 00000000 00000100//右移原码:10000000 00000000 00000000 00000101--->-5int n = num>>1;printf("n= %d\n", n);printf("num= %d\n", num);return 0;
}

打印:

n= -5
num= -10

5.位操作符:&、|、^、~

双目操作符:

  1. &:按位与
  2. |:按位或
  3. ^:按位异或

单目操作符:

  1. ~:按位取反

5.1 &:按位与

有0则0。

#include <stdio.h>
int main()
{int a = 6;// 6的补码:00000000 00000000 00000000 00000110int b = -7;//-7的原码:10000000 00000000 00000000 00000111//-7的反码:11111111 11111111 11111111 11111000//-7的补码:11111111 11111111 11111111 11111001int c = a & b;// 6的补码:00000000 00000000 00000000 00000110//-7的补码:11111111 11111111 11111111 11111001//a & b补码::00000000 00000000 00000000 00000000-->0printf("n= %d\n", c);return 0;
}

打印:

n= 0

5.2 |:按位或

有1则1。

#include <stdio.h>
int main()
{int a = 6;// 6的补码:00000000 00000000 00000000 00000110int b = -7;//-7的原码:10000000 00000000 00000000 00000111//-7的反码:11111111 11111111 11111111 11111000//-7的补码:11111111 11111111 11111111 11111001int c = a | b;// 6的补码:00000000 00000000 00000000 00000110//-7的补码:11111111 11111111 11111111 11111001//a | b补码:11111111 11111111 11111111 11111111//a | b反码:10000000 00000000 00000000 00000000//a | b原码:10000000 00000000 00000000 00000001-->-1printf("n= %d\n", c);return 0;
}

打印:

n= -1

5.3 ^:按位异或

相同为0,不同为1。

#include <stdio.h>
int main()
{int a = 6;// 6的补码:00000000 00000000 00000000 00000110int b = -7;//-7的原码:10000000 00000000 00000000 00000111//-7的反码:11111111 11111111 11111111 11111000//-7的补码:11111111 11111111 11111111 11111001int c = a ^ b;// 6的补码:00000000 00000000 00000000 00000110//-7的补码:11111111 11111111 11111111 11111001//a ^ b补码:11111111 11111111 11111111 11111111//a ^ b反码:10000000 00000000 00000000 00000000//a ^ b原码:10000000 00000000 00000000 00000001-->-1printf("n= %d\n", c);return 0;
}

打印:

n= -1

5.4 ~:按位取反

按2进制位取反

#include <stdio.h>
int main()
{int a = 0;//0的原码:00000000 00000000 00000000 00000000//~a补码: 11111111 11111111 11111111 11111111//~a反码: 10000000 00000000 00000000 00000000 //~a原码: 10000000 00000000 00000000 00000001-->-1printf("n= %d\n", ~a);return 0;
}

打印:

n= -1

5.5 例题

例题1

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

方法1:

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

打印:

交换前:a=3 b=5
交换后:a=5 b=3

上面这个方法有点问题,如果a+b的和超过了一个整型数据的存储大小,那么就计算不了了。


方法2:

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

打印:

交换前:a=3 b=5
交换后:a=5 b=3

为什么呢?

因为异或操作符,相同两个数异或为0。

3^3=0

a^a=0

0异或任何数都为数的本身 。

0^3=3

0^a=a

异或是支持交换率的。


例题2

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

int count_bit_onr(unsigned int n) {int count = 0;while (n) {if ((n % 2) == 1) {count++;}n = n / 2;}return count;
}int main() {int num = 0;scanf("%d", &num);int ret = count_bit_onr(num);printf("%d\n", ret);return 0;
}

输入:

15

打印:

4

也可以这么写:

int count_bit_onr(int n) {int i = 0;int count = 0;for (i = 0; i < 32; i++) {if ((n >> 1) & 1 == 1) {count++;}}return count;
}int main() {int num = 0;scanf("%d", &num);int ret = count_bit_onr(num);printf("%d\n", ret);return 0;
}

输入-1会打印32

这个算法的原理:

-1补码:11111111 11111111 11111111 11111111
1补码: 00000000 00000000 00000000 00000001
-1&1=  00000000 00000000 00000000 00000001

不论-1的补码前面多少个1,只要和1按位与后就只看最后一位是不是1。

然后运用移位操作符,将每一位都和1按位与,统计出一共多少个1。


其实还有更加巧妙地算法:

n=n&(n-1)

例如:n=11

n       = 1011
n-1     = 1010
n&(n-1) = 1010

因为n=n&(n-1),所以现在新的n是1010

n       = 1010
n-1     = 1001
n&(n-1) = 1000

因为n=n&(n-1),所以现在新的n是1000

n       = 1000
n-1     = 0111
n&(n-1) = 0000

从这三次变化里面我们可以看到,我们执行了一次n=n&(n-1),那么n最右面那个1就会消失。把所有的1都去掉后就变成0了。


例题3

写一个代码,判断n是否为2的次方数。

首先我们先看看2的次方数:

000010
000100
001000
010000
100000

可以看出2的次方数里面只有1个1。

if (n & (n - 1) == 0) {printf("yes")
}

例题4

编写代码,将13二进制序列的第五位修改为1,然后改回0。

第5位改成1,第5位就和1或|,其他位都是0。

第5位改为0,第5位就和0与&,其他位都是1。

int main() {int a = 13;//13原码:00000000 00000000 00000000 00001101//13补码:00000000 00000000 00000000 00001101//16补码:00000000 00000000 00000000 00010000//16|13:00000000 00000000 00000000 00011101-->29int n = 5;a = a | (1 << (n - 1));printf("%d\n", a);a &= ~(a << (n - 1));printf("%d\n", a);return 0;
}

打印:

29
13

相关文章:

【C语言】8.C语言操作符详解(1)

文章目录 1.操作符的分类2.⼆进制和进制转换3.原码、反码、补码4.移位操作符4.1 左移操作符4.2 右移操作符 5.位操作符&#xff1a;&、|、^、~5.1 &&#xff1a;按位与5.2 |&#xff1a;按位或5.3 ^&#xff1a;按位异或5.4 ~&#xff1a;按位取反5.5 例题例题1例题2例…...

Buzz库网络爬虫实例:快速爬取百度搜索实时热点

前言 随着互联网的发展&#xff0c;信息获取已经成为了人们日常生活和工作中的重要一环。而在信息获取的过程中&#xff0c;网络爬虫作为一种自动化的数据采集工具&#xff0c;为我们提供了极大的便利。本文将介绍如何利用PHP编写一个简单而高效的网络爬虫&#xff0c;实现快速…...

SQL注入:pikachu靶场中的SQL注入通关

目录 1、数字型注入&#xff08;post&#xff09; 2、字符型注入&#xff08;get&#xff09; 3、搜索型注入 4、XX型注入 5、"insert/update"注入 Insert&#xff1a; update&#xff1a; 6、"delete"注入 7、"http header"注入 8、盲…...

springsecurity入门登录授权

①我们需要自定义登陆接口&#xff0c;也就是在controller目录新建LoginController类&#xff0c;在controller方法里面去调用service接口&#xff0c;在service接口实现AuthenticationManager去进行用户的认证&#xff0c;注意&#xff0c;我们定义的controller方法要让Spring…...

医学科技查新中对查新点的撰写方法!附案例讲解!

我国的科技查新工作最早是从医学领域开始的&#xff0c;始于1985年中国科学院医学情报所&#xff0c;后来逐步发展到工、农等其 他各个领域。医学科技查新包括立项查新和成果查新两个部分&#xff0c;其中医学立项查新&#xff0c;它是指在医学科研项目申报开题之前&#xff0c…...

2024最新流媒体在线音乐系统网站源码| 音乐社区 | 多语言 | 开心版

2024最新流媒体在线音乐系统网站源码| 音乐社区 | 多语言 | 开心版 https://download.csdn.net/download/huayula/89347742...

回溯算法05(leetcode491/46/47)

参考资料&#xff1a; https://programmercarl.com/0491.%E9%80%92%E5%A2%9E%E5%AD%90%E5%BA%8F%E5%88%97.html 491. 非递减子序列 题目描述&#xff1a; 给你一个整数数组 nums &#xff0c;找出并返回所有该数组中不同的递增子序列&#xff0c;递增子序列中 至少有两个元素…...

Transformer,革命性的深度学习架构

Transformer 是一种革命性的深度学习架构,专门设计用于处理序列数据,特别是在自然语言处理(NLP)任务中表现卓越。它由 Vaswani 等人在 2017 年发表的论文《Attention is All You Need》中首次提出,打破了当时基于循环神经网络(RNN)和卷积神经网络(CNN)的序列建模常规,…...

实验五:实现循环双链表各种基本运算的算法

实验五&#xff1a;实现循环双链表各种基本运算的算法 一、实验目的与要求 目的:领会循环双链表存储结构和掌握循环双链表中各种基本运算算法设计。 内容:编写一个程序cdinklist.cpp,实现循环双链表的各种基本运算和整体建表算法(假设循环双链表的元素类型ElemType为char),并…...

ElasticSearch IK分词器的安装、词典扩展与停用

&#x1f3f7;️个人主页&#xff1a;牵着猫散步的鼠鼠 &#x1f3f7;️系列专栏&#xff1a;云原生与服务部署-专栏 &#x1f3f7;️个人学习笔记&#xff0c;若有缺误&#xff0c;欢迎评论区指正 目录 ​编辑 1. 前言 2. IK分词器安装 3. IK分词器词典扩展与停用 4. 总…...

代码随想录训练营总结

为期两个月的代码随想录训练营今天结束了&#xff0c;我想我的收获是非常大的。进到训练营的大群里&#xff0c;令我有种安心的感觉&#xff0c;原来世界各地有这么多与我一起努力的伙伴。更令人安心的是知识星球对于学习进度的规划&#xff0c;细化到每一天每道题&#xff0c;…...

深度学习-转置卷积

转置卷积 转置卷积&#xff08;Transposed Convolution&#xff09;&#xff0c;也被称为反卷积&#xff08;Deconvolution&#xff09;&#xff0c;是深度学习中的一种操作&#xff0c;特别是在卷积神经网络&#xff08;CNN&#xff09;中。它可以将一个低维度的特征图&#x…...

Unity性能优化工具介绍

文章目录 一.Stats组件1.Audio音频的数据组件:2.图形数据 二.Profiler 性能分析器 一.Stats组件 Unity自带Statistics(统计数据),Game视窗中点击Stats打开 1.Audio音频的数据组件: 1):Level 声音强度 单位是分贝(dB) 表示音频听声音的大小,是闪烁波动的. 2):SDPload 数据信…...

Math之向上向下取整

有时我们会遇到向上和向下取整的操作&#xff0c;这时我们可以使用Math类来进行操作。 1、向上取整 Math.ceil() 方法返回大于或等于指定表达式的最小整数&#xff08;即向上取整&#xff09;。如果参数是一个整数&#xff0c;那么结果就是这个整数本身。 示例&#xff1a; …...

MPP架构

MPP架构&#xff0c;即Massively Parallel Processing&#xff08;大规模并行处理&#xff09;架构&#xff0c;是一种用于处理大规模数据的并行计算架构。它通过将数据和计算能力分布在多个处理节点上&#xff0c;利用并行处理技术来加速数据处理和分析的速度。 在MPP架构中&…...

These relative modules were not found:* ../../../constant in

这个错误信息表明&#xff0c;你的项目在尝试加载一个相对路径模块 ../../../constant 时遇到了问题。具体来说&#xff0c;它在 ./node_modules/cache-loader/dist/cj 这个路径下找不到这个模块。 这里有几个可能的原因和相应的解决方案&#xff1a; 路径错误&#xff1a;首…...

2024最新彩虹聚合DNS管理系统源码v1.3 全开源

2024最新彩虹聚合DNS管理系统源码v1.3 全开源 聚合DNS管理系统可以实现在一个网站内管理多个平台的域名解析&#xff0c;目前已支持的域名平台有&#xff1a;阿里云、腾讯云、华为云、西部数码、DNSLA、CloudFlare。 本系统支持多用户&#xff0c;每个用户可分配不同的域名解…...

在Go语言中如何实现变参函数和函数选项模式

在Go语言编程中,我们经常会遇到需要给函数传递可选参数的情况。传统的做法是定义一个结构体,将所有可选参数作为结构体字段,然后在调用函数时创建该结构体的实例并传递。这种方式虽然可行,但是当可选参数较多时,创建结构体实例的代码就会变得冗长และ不太直观。 Go语言的一个…...

Spring Boot中的 6 种API请求参数读取方式

使用Spring Boot开发API的时候&#xff0c;读取请求参数是服务端编码中最基本的一项操作&#xff0c;Spring Boot中也提供了多种机制来满足不同的API设计要求。 接下来&#xff0c;就通过本文&#xff0c;为大家总结6种常用的请求参数读取方式。如果你发现自己知道的不到6种&a…...

Linux基础命令[27]-gpasswd

文章目录 1. gpasswd 命令说明2. gpasswd 命令语法3. gpasswd 命令示例3.1 不加参数3.2 -a&#xff08;将用户加入组&#xff09;3.3 -d&#xff08;从组中删除用户&#xff09;3.4 -r&#xff08;删除组密码&#xff09;3.5 -M&#xff08;多个用户一起加入组&#xff09;3.6 …...

MPNet:旋转机械轻量化故障诊断模型详解python代码复现

目录 一、问题背景与挑战 二、MPNet核心架构 2.1 多分支特征融合模块(MBFM) 2.2 残差注意力金字塔模块(RAPM) 2.2.1 空间金字塔注意力(SPA) 2.2.2 金字塔残差块(PRBlock) 2.3 分类器设计 三、关键技术突破 3.1 多尺度特征融合 3.2 轻量化设计策略 3.3 抗噪声…...

深入剖析AI大模型:大模型时代的 Prompt 工程全解析

今天聊的内容&#xff0c;我认为是AI开发里面非常重要的内容。它在AI开发里无处不在&#xff0c;当你对 AI 助手说 "用李白的风格写一首关于人工智能的诗"&#xff0c;或者让翻译模型 "将这段合同翻译成商务日语" 时&#xff0c;输入的这句话就是 Prompt。…...

k8s从入门到放弃之Ingress七层负载

k8s从入门到放弃之Ingress七层负载 在Kubernetes&#xff08;简称K8s&#xff09;中&#xff0c;Ingress是一个API对象&#xff0c;它允许你定义如何从集群外部访问集群内部的服务。Ingress可以提供负载均衡、SSL终结和基于名称的虚拟主机等功能。通过Ingress&#xff0c;你可…...

STM32F4基本定时器使用和原理详解

STM32F4基本定时器使用和原理详解 前言如何确定定时器挂载在哪条时钟线上配置及使用方法参数配置PrescalerCounter ModeCounter Periodauto-reload preloadTrigger Event Selection 中断配置生成的代码及使用方法初始化代码基本定时器触发DCA或者ADC的代码讲解中断代码定时启动…...

是否存在路径(FIFOBB算法)

题目描述 一个具有 n 个顶点e条边的无向图&#xff0c;该图顶点的编号依次为0到n-1且不存在顶点与自身相连的边。请使用FIFOBB算法编写程序&#xff0c;确定是否存在从顶点 source到顶点 destination的路径。 输入 第一行两个整数&#xff0c;分别表示n 和 e 的值&#xff08;1…...

rnn判断string中第一次出现a的下标

# coding:utf8 import torch import torch.nn as nn import numpy as np import random import json""" 基于pytorch的网络编写 实现一个RNN网络完成多分类任务 判断字符 a 第一次出现在字符串中的位置 """class TorchModel(nn.Module):def __in…...

Unsafe Fileupload篇补充-木马的详细教程与木马分享(中国蚁剑方式)

在之前的皮卡丘靶场第九期Unsafe Fileupload篇中我们学习了木马的原理并且学了一个简单的木马文件 本期内容是为了更好的为大家解释木马&#xff08;服务器方面的&#xff09;的原理&#xff0c;连接&#xff0c;以及各种木马及连接工具的分享 文件木马&#xff1a;https://w…...

[免费]微信小程序问卷调查系统(SpringBoot后端+Vue管理端)【论文+源码+SQL脚本】

大家好&#xff0c;我是java1234_小锋老师&#xff0c;看到一个不错的微信小程序问卷调查系统(SpringBoot后端Vue管理端)【论文源码SQL脚本】&#xff0c;分享下哈。 项目视频演示 【免费】微信小程序问卷调查系统(SpringBoot后端Vue管理端) Java毕业设计_哔哩哔哩_bilibili 项…...

渗透实战PortSwigger靶场:lab13存储型DOM XSS详解

进来是需要留言的&#xff0c;先用做简单的 html 标签测试 发现面的</h1>不见了 数据包中找到了一个loadCommentsWithVulnerableEscapeHtml.js 他是把用户输入的<>进行 html 编码&#xff0c;输入的<>当成字符串处理回显到页面中&#xff0c;看来只是把用户输…...

从物理机到云原生:全面解析计算虚拟化技术的演进与应用

前言&#xff1a;我的虚拟化技术探索之旅 我最早接触"虚拟机"的概念是从Java开始的——JVM&#xff08;Java Virtual Machine&#xff09;让"一次编写&#xff0c;到处运行"成为可能。这个软件层面的虚拟化让我着迷&#xff0c;但直到后来接触VMware和Doc…...