第9天----【位运算进阶之----按位取反(~)】(附补码,原码讲解)

news2025/1/16 10:55:38

今天我们来谈谈按位取反这件事。
简单来说,按位取反就是先将一个数写成其二进制表达形式,然后1变0,0变1。下面就让我们展开深入地讨论吧!
在这里插入图片描述


文章目录

  • 一、预备知识:
    • 1. 原码:
      • 定义:
      • 优缺点:
    • 2. 补码:
      • 定义:
      • 优缺点:
  • 二、按位取反:
  • 三、拓展应用:
    • 1. 巧求相反数:(不用`-`)
    • 2. 代替减法(不用`-`实现减法)
    • 3. 代替加法(不用`+`实现加法)


一、预备知识:

1. 原码:

定义:

✨原码是一种用二进制表示有符号整数的编码方式。其中,最高位表示符号位0为正1为负其余位是数值位,表示数值的绝对值。

  • 举个“栗子”来说:
    • 1的原码是:001 ;
    • -1的原码是:101 ;

✨我们可以看出,互为相反数的两个数的原码,除了第一位的符号位不一样以外,其余位都相同。(那么,可能有聪明的小伙伴们就想到了0,是不是0也有两种表达方式呢?果然聪明!在原码中,0-0的表示是不一样的。)

优缺点:

  • ❤️优点:简单直观,易于理解和计算。
  • 💔缺点:加减法运算复杂,需要考虑符号位的处理。-0多占了一个数的表达位置,导致原码也可表示的总数减少了1个。

✨那可能又有小伙伴要问了,一个数很多吗?哈哈,一个-0单单看起来不多,但是一群-0不就多了吗?(这就告诉了我们"人多力量大"的道理,一个人可以走得很快,但一群人才能走得更远)。另外,-0的存在还会带来一些其他的问题:比如1️⃣(0)+(-0)=?2️⃣ 0和-0哪个大?

😢怎么一个原码事这么多?烦呐!下面就让我们来看看更为合适的编码方式–补码。


2. 补码:

❤️计算机中存储数据时,通常采用补码编码方式。

定义:

✨补码也是一种用于表示有符号整数的编码方式。其中,最高位(最左侧位)是符号位0表示正数,1表示负数其余位是数值位

  • 对于正数,补码就是其二进制原码本身。(看来不讲原码还不能开补码),如:数值+3的补码就是0011。
  • 对于负数,补码是其对应正数的原码的各位取反,末位加一,符号位取1。

如:-3的补码可以通过三步得到:(因为+3的原码是0011)
1️⃣各位取反:1100
2️⃣末位加一:1101
3️⃣符号位取1:1101


优缺点:

  • ❤️优点:
    1️⃣加减法运算可以直接按位运算,无需特殊处理符号位。(相对原码来说更加方便)
    2️⃣解决了原码中存在的正零和负零的问题。(-0的位置让给了最小的数)
    3️⃣表示范围更加均衡。(补码的最小值的绝对值比最大值的绝对值多1)

  • 💔缺点:
    1️⃣加减法时可能溢出。
    2️⃣对称不明显。(看吧,上帝为你打开一扇窗的同时,也会为你关闭一扇门)

🐒当然,补码的知识宝库偌大无比,我现在只不过是揭开了其冰山一角,但对我们解决一般的问题来说,足够了。


二、按位取反:

✨将一个数的二进制表达的每一位取反,即0变为11变为0取反符号用 "~" 表示。(从最低位(最右侧位)开始,逐位进行取反操作。)

例如,对于一个8位的二进制数10101010,按位取反后的结果是01010101。(注意,这是补码的取反哦)

下面来看几个具体的代码:
1️⃣1的按位取反:

#include<stdio.h>

int main(void)
{
	int a = 0b1;
	printf("%d", ~a);
	return 0;
}

😄猜猜它的输出是多少? 输出0是不是?哈哈哈,是0你就上当了!
在这里插入图片描述
咦❓为什么❓为什么不是0❓ 1->0 没有问题啊❗️到底是哪里出错了❓会是哪里错了呢❓
在这里插入图片描述
阁下莫慌,我们慢慢来解释。

  1. 首先,a是一个int类型,一般是4个字节(Byte),而一个字节是8位(8个bit),所以总共是32位。0b1是1的二进制表达,具体可以写为:00000000 00000000 00000000 00000001.
  2. 下面我们对它进行取反操作,得:11111111 11111111 11111111 11111110,但它是一个补码,我们以%d的形式输出就要得到它的十进制表达(可以理解为真值),那么问题来了,补码怎么转换成真值呢?考虑到具体的定义过于枯燥乏味,所以我直接告诉你转换的方法:可以先将补码转换为机器数,而计算机中的机器数一般用原码来表示,所以问题就转化为了补码->原码
  3. 将补码的符号位不变,其余各位取反,末位加一。(这好像和原码转补码有点类似啊!)也就是:10000000 00000000 00000000 00000001+1---> 10000000 00000000 00000000 00000010,再转换成十进制整数就是-2。

2️⃣0的按位取反:(%d的形式输出)

#include<stdio.h>

int main(void)
{
	printf("%d", ~0);
	return 0;
}

在这里插入图片描述

2️⃣0的按位取反:(%u的形式输出)

#include<stdio.h>

int main(void)
{
	printf("%u\n", ~0);
	printf("%lld", ((long long)1 << 32) - 1);
	return 0;
}

在这里插入图片描述
咦❓为啥同样是0,但取反后的输出结果不一样呢?

  • %d:有符号十进制整数的输出 (signed int): -2 ^ 31 -- 2 ^ 31 - 1
  • %u:无符号十进制整数的输出 (unsigned int) : 0 -- 2 ^ 32 - 1

这个有无符号具体就体现在是否有符号位,有符号位就少一位数值位,没有符号位就都是数值位。

  • 0的补码:00000000 00000000 00000000 00000000
  • ~0的补码:11111111 11111111 11111111 11111111

✨对于有符号整数来说,有正有负,且第一位是符号位。~0第一位是1,所以要转换成原码再输出:
10000000 00000000 00000000 00000001,其值为-1;
✨对于无符号整数来说,没有符号位,32个位都是数值位,所以可以直接计算。因此,两者输出不同。


三、拓展应用:

1. 巧求相反数:(不用-

✨相反数相必大家都知道,1的相反数是-1,0的相反数是0,简单来说一个数的相反数就是再其前面加个-

那么,不用这种方法还可以求相反数吗?

哈哈😄,其实这个问题我们上面已经说过了,直接利用补码的定义,求一个数的相反数,就是各位取反,末位加一。即-x=~x+1

#include<stdio.h>
int main(void)
{
	int x = -18;
	printf("%d的相反数是%d\n",x, ~x + 1);

	int y = 18;
	printf("%d的相反数是%d",y, ~y + 1);
	return 0;
}

在这里插入图片描述
(当然了,要注意数据的溢出问题。)


2. 代替减法(不用-实现减法)

✨现给定int类型的两个正数,不用-如何实现减法操作?
根据减法定义,减去一个数就等于加上这个数的相反数;
所以a - b = a + (-b) = a + (~b + 1);

#include<stdio.h>
int main(void)
{
	int a=1,b=2;
	printf("%d-%d=%d\n",a, b,a+~b+1);
	return 0;
}

在这里插入图片描述


3. 代替加法(不用+实现加法)

✨给定两个int类型的正数,不用+实现加法。根据加法定义,加上一个数等于减去一个数的相反数。即:a + b = a - (-b) = a - (~b +1)=a - ~b - 1;

#include<stdio.h>
int main(void)
{
	int a=1,b=2;
	printf("%d+%d=%d\n",a, b,a-(~b + 1));
	return 0;
}

在这里插入图片描述


好了,今天的讲解就到这里了,相信你也是收获满满吧!
终于肝完了,好累!!!

在这里插入图片描述

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

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

相关文章

PyCharm下安装配置PySide6开发环境(Qt Designer(打开,编辑)、PyUIC和PyRCC)

一.准备工作 1.安装python和pycharm并配置好环境变量 python安装路径 pycharm安装路径&#xff1a; python系统变量&#xff1a; pycharm环境变量&#xff1a; 注意&#xff1a;正常安装&#xff0c;并勾选ADD PATH一般会自动配好 2.在pycharm创建一个新的python的虚拟环境 …

lEC 61068-2-14_2023环境试验.第2-14部分:试验.试验N:温度变化, 最新版发布

https://download.csdn.net/download/m0_67373485/88251313 lEC 61068-2-14_2023环境试验.第2-14部分:试验.试验N:温度变化 A change of temperature test is intended to determine the effect on the specimen of a changeof temperature or a succession of changes of tem…

postgre 基本操作

给已经创建好的table 添加主键,在postgrep中添加主键&#xff0c;才可以在pgAdmin中修改表中内容 ALTER TABLE user ADD PRIMARY KEY (id); 这个编辑的按钮才会显示&#xff0c;不然就是小锁 给表添加json数据 INSERT INTO POWER (tool,METRICS) VALUES (powerpro,[{"n…

stm32之11.USART串口通信

可以添加上拉电阻&#xff0c;但会增加功耗&#xff0c;传输距离变长 要添加库函数USART 官方参考文档说明书位置 ALT&#xff0b;左键可实现整体删除&#xff08;如下图&#xff09; 输出模式第三种模式AF ---------------------- 源码 远程控制pc端 #include <stm32f4x…

[ACL2023] Exploring Lottery Prompts for Pre-trained Language Models

Exploring Lottery Prompts for Pre-trained Language Models 文章链接 清深的工作&#xff0c;比较有意思的一篇。作者先给出假设&#xff0c;对于分类问题&#xff0c;在有限的语料空间内总能找到一个prompt让这个问题分类正确&#xff0c;作者称之为lottery prompt。为此&…

windows窗口背景色修改方法

windows也不知道什么时候将notepad, word, vs的背景色由白色变成了浅绿色&#xff0c;用了一段时间没去改觉得麻烦&#xff0c;就一直用下去了&#xff0c;今天有时间找了下方法改回了白色。 1. 被动默认浅绿色效果 2. 修改注册表. 保存&#xff0c;重启电脑. 3. 恢复如初.

[LeetCode周赛复盘] 第 111 场双周赛20230819

[LeetCode周赛复盘] 第 111 场双周赛20230819 一、本周周赛总结2824. 统计和小于目标的下标对数目1. 题目描述2. 思路分析3. 代码实现 2825. 循环增长使字符串子序列等于另一个字符串1. 题目描述2. 思路分析3. 代码实现 2826. 将三个组排序1. 题目描述2. 思路分析3. 代码实现 …

element纯前端table分页

<template><div class"boxs"><div class"close" click"closeShow()"><img src"./image/close.png"></div><div class"title">一舟储能峰谷收益统计数据</div><div class"…

计算机安全学习笔记(II):自主访问控制 - DAC

书接上篇博客&#xff0c;自主访问方案是指一个实体可以被授权按其自己的意志使另一个实体能够访问某些资源。DAC的一种通常访问方式是在操作系统或数据库管理系统中运用的访问矩阵(access matrix)。 矩阵中的一维由试图访问资源的被标识的主体组成。这个列表一般由用户或用户…

【核磁共振成像】方格化重建

目录 一、缩放比例二、方格化变换的基础三、重建时间四、方格化核 一、缩放比例 对于笛卡尔K空间直线轨迹数据可直接用FFT重建&#xff0c;而如果K空间轨迹的任何部分都是非均匀取样的 可用DFT直接重建&#xff0c;有时称为共轭相位重建&#xff0c;但此法太慢不实用。把数据再…

js、PHP连接外卖小票机打印机方案(调用佳博、芯烨等)

前言&#xff1a; 目前开发需要用到电脑直接连接外卖小票机打印小票&#xff0c;查阅各种资料&#xff0c;使用 6612345浏览器 终于解决了这个问题。 效果&#xff1a; PHP、js直接连接小票机并且自动出票。 支持的小票机&#xff1a; 目前测试可以的有&#xff1a;电脑A4打印…

【TI毫米波雷达笔记】SOC外设初始化配置及驱动(以IWR6843AOP为例)

【TI毫米波雷达笔记】SOC外设初始化配置及驱动&#xff08;以IWR6843AOP为例&#xff09; 最基本的工程建立好以后 需要给SOC进行初始化配置 SOC_Cfg socCfg; //SOC配置结构体Task_Params taskParams; //任务参数SOC_Handle socHandle;ESM_init(0U); …

网络安全工程师岗位分类-徐庆臣(黑客洗白者)

研发系&#xff1a; 安全研发、安全攻防研究、逆向分析、云计算研究、机器安全 工程系&#xff1a; 安全工程师、安全运维工程师、安全服务工程师、安全技术支持、安全售后、Web渗透测试工程师、Web安全工程师、应用安全审计、移动安全工程师 销售系&#xff1a; 安全销售…

数字孪生:重塑制造、医疗和能源等领域的未来

数字孪生技术&#xff0c;作为虚拟仿真的重要领域&#xff0c;正以其强大的能力在各个行业中创造前所未有的创新。本文带大家一起深入探讨数字孪生技术在不同领域的广泛应用场景&#xff0c;展示其在实现效率、可靠性和智能化方面的积极影响。 制造业与工业领域 数字孪生技术在…

力扣 337. 打家劫舍 III

题目来源&#xff1a;https://leetcode.cn/problems/house-robber-iii/description/ C题解1&#xff08;来源代码随想录&#xff09;&#xff1a;本题一定是要后序遍历&#xff0c;因为通过递归函数的返回值来做下一步计算。本题关键是要讨论当前节点抢还是不抢。如果抢了当前节…

Java--abstract class 与 interface的区别

在Java语言中&#xff0c;abstract class和interface是支持抽象类定义的两种机制。正是由于这两种机制的存在&#xff0c;才赋予了Java强大的面向对象能力。abstract class和interface之间在对于抽象类定义的支持方面具有很大的相似性&#xff0c;甚至可以相互替换&#xff0c;…

【2023钉钉杯复赛】A题 智能手机用户监测数据分析 Python代码分析

【2023钉钉杯复赛】A题 智能手机用户监测数据分析 Python代码分析 1 题目 一、问题背景 近年来&#xff0c;随着智能手机的产生&#xff0c;发展到爆炸式的普及增长&#xff0c;不仅推动了中 国智能手机市场的发展和扩大&#xff0c;还快速的促进手机软件的开发。近年中国智能…

Linux操作系统--linux环境搭建(2)

在上一小节中,我们已经把CentOS和VMware下载和安装做好了,下面我们使用VMware创建一个虚拟机,如下 1.Vmware创建虚拟机 下面我们开始创建一个新的虚拟机。这里操作就类似于我们攒一个电脑,我们呢先把电脑攒出来,然后再给电脑安装操作系统,这样就可以开始使用这一台电脑了…

【C++】—— C++11新特性之 “右值引用和移动语义”

前言&#xff1a; 本期&#xff0c;我们将要的介绍有关 C右值引用 的相关知识。对于本期知识内容&#xff0c;大家是必须要能够掌握的&#xff0c;在面试中是属于重点考察对象。 目录 &#xff08;一&#xff09;左值引用和右值引用 1、什么是左值&#xff1f;什么是左值引用…

如何在不使用任何软件的情况下将 PDF 转换为 Excel

通常&#xff0c;您可能会遇到这样的情况&#xff1a;您需要的数据不在 Excel 工作表中&#xff0c;而是以数据表形式出现在 PDF 文件中。为了将此数据放入 Excel 工作表中&#xff0c;如果您尝试将数字复制并粘贴到电子表格中&#xff0c;则列/行将无法正确复制和对齐。因此&a…