“第五十四天” 溢出判断,标志位 ,有问题

2023-10-30 04:28

本文主要是介绍“第五十四天” 溢出判断,标志位 ,有问题,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

之前说过在运算的时候可能发生溢出,这种情况只会出现在 正正相加  ,负负相加 的时候才会出现前者是上溢,正正加得负,后者是下溢,负负加得正。(减法变成加法,再看)

       加减运算&溢出判断

方法一:利用符号位进行判断是否溢出

假设 a+b=c;a,b,c的符号分别视为as,bs,cs,则将     as与bs与(cs的非)的结果      或       ( as的非)与(bs的非)与cs的结果,最终的到的结果记为v 。如果v为0则表示无溢出,如果v为1则表示溢出。其实就是看看abc三者的符号一样不一样,不过这逻辑实现我可能还想不到,前面半部分用来看负负的溢出,如果前半部分结果为 1 则说明abc应该是 1 1 0,则 数值和结果的符号不一样,就是溢出了,后面半部分就是用来看正正的溢出,和上面一样类比一下就行。

方法二:利用符号位的进位 和 最高数值位的进位情况来判断是否溢出(也就是最前面那两位的进位,这个最前是百十个的百)

当符号位和最高位进位不一样的时候表示发生溢出。

如果最高数值位的进位为 1 而符号位的进位为 0 的时候表示上溢;

如果最高数值位的进位为 0 而符号位的进位为 1 的时候表示下溢;

之前说过,只有正正和负负才会发生溢出,对于正正来说,符号位的进位一定是 0 ,那最高数值为的进位是1,最后符号位就变成 1 了,那就上溢了,负负和这个差不多。

这种方法的实现可以借助逻辑运算 异或  (相同为 0,不同为1).

方法三:采用双符号位,正数的符号位是 00 ,负数的符号位是 11.

(这里数据在内存中存储的还是一个符号位,不过在参与运算的时候,会将符号位复制一次)

这种方法的话,如果最后的结果,两位的符号如果最后不一样就表示发生了溢出,01表示发生了上溢,10表示发生了下溢,前面的符号位原本正确的符号位,后面的应该是数值位。

这个也可以使用逻辑运算 异或 来实现 ,不同为 1 表示发生溢出。

双符号位补码又称:模4补码(双符号位 可以表示的数是 00 ,01,10,11也就是 0,1,2,3,模4的余数)                  单符号位补码又称:模2补码 ,解释和上面一样。

        符号扩展;

如果将短数据变成长数据,如int 的数据扩展成 long 型 

对于定点整数的符号扩展:在原符号位和数值位中间添加新位的时候,正数都补 0 ,负数原码补 0 ,反码,补码补 1。 这里我感觉就是原码都是补 0 ,补码按符号位决定补 1/0;

定点小数,定点小数是在最后面添加新位的。而且扩展的方式和整数不一样,注意,定点小数正数都补 0 ,负数的原,补码补的是 0,反码补的是 1.除了反码以外剩下的都补 0

        标志位的生成:这个感觉图清楚一点

OF:在硬件上的实现方法是将最高位的进位和次高位的进位进行异或处理,为1则表示溢出,就是上面判断溢出的方法二,根据符号位的进位和最高数值位的进位来判断是否溢出。需要注意的是OF位对无符号数的加减法是没有意义的。

SF:表示有符号加减运算结果的正负性,0表示正,1表示负,(但这里我不太理解存在的意义,因为符号位不是在运算结果中就已经体现了吗,为什么这里还要单独出来一个SF标志位?

在硬件上的计算方法SF的值就是最高位的本位和,同样这个也对无符号数的加减法无意义。

ZF:用来判断运算结果是否为 0 ,其值为 1 的时候 表示结果为 0 ,其值为 0 的时候 ,表示运算结果非 0,在硬件上的计算方法是运算结果的全部位 为 0 的时候 ZF的值为1,这里我和上面保有一样的疑问,不过这里再用来判断的时候是不是好点。

CF:用来判断无符号数的加减法(注意是无符号)是否发生进位/借位,其值为1时,说明发生了进位/借位,也就是发生了溢出。硬件的计算方法,其值是最高位的产生的进位和sub的异或结果(sub为1时表示减法,sub为0表示加法)。这里应该是站在最高位的角度来看的,如果最高位发生进位,则结果超出上限,表示溢出;如果最高位发生进位,则表示被减数小于减数,然后被减数的最高位向更高位借 1(实际上没有更高位了),也就是说本来应该是一个负值,但无符号数无法表示负值,出了问题。

前面两个 OF , SF 仅对有符号数有用,最后一个 CF 只对无符号数有用,第三个 ZF  通用。

        移位运算:

移位:通过改变各个数码位和小数点的相对位置,从而改变各数码位的位权。可用移位运算实现乘除。

这里的右移或者左移是对于数码位而言,小数点不动。

原码的算术移位 —— 符号位保持不变,仅对数值位进行移位。

右移:高位补0,低位舍弃,若舍弃的位 = 0 ,则相当于 除以 2 ;若舍弃的位 不等于 0 ,则会丢失精度。

左移:低位补0,高位舍弃,若舍弃的位 = 0,则相当于 乘以 2;若舍弃的位 不等于 0,则会出现严重的误差,因为这里丢失的高位 ,如果是1001的话,就只剩 1 了。

对于正数而言,反码的算数移位和原码一样,对于负数而言,右移和左移的时候,高低位补的是1.

补码的话,正数的补码与原码的算数移位相同;

对于负数而言,负数的补码的算术移位,右移的时候,高位补1 ,低位舍弃,对于左移的话,低位补0,高位舍弃。规律:负数补码中,最右边的 1 及其右边同原码一样,最右边 1 的左边同反码一样。

左移的效果相当于 扩大二倍 ,右移的效果相当于缩小二倍;但由于位数有限,因此有时候无法用算数移位精确地等效乘除法。

        逻辑移位:

逻辑右移:高位补 0 ,低位舍弃;

逻辑左移:低位补 0 ,高位舍弃;

可以把逻辑移位看作是对“无符号数”的算数移位。

        循环移位

循环移位有带进位位的和不带进位位的两种;

对于不带进位位的循环移位,用移出的位不上空缺的位,比如右移,这样最低位被移除,最高位变成次高位,最高位空缺,然后移出的最低位就会补到因为右移而空缺出的最高位。

对于带进位位的循环移位,会把移出的位放到进位位,原进位位补上空缺,也是右移,这样最低位(假如是0)被移除,然后把原进位位(假如是1)已到由于右移而空缺的最高位,然后把最低位移出的 0 放入进位位。

这一题确实不难,但要注意20的阶乘对于int型和long型会溢出
实际上还有一个就是 0! 的问题,不过这种情况好像不在测试之中。
int main()
{int n = 0;scanf("%d", &n);int i = 0;unsigned long sum = 1, m = 1;for (i = 2; i <= n; i++){m *= i;sum += m;}printf("%lu", sum);return 0;
}

这篇关于“第五十四天” 溢出判断,标志位 ,有问题的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/305573

相关文章

好题——hdu2522(小数问题:求1/n的第一个循环节)

好喜欢这题,第一次做小数问题,一开始真心没思路,然后参考了网上的一些资料。 知识点***********************************无限不循环小数即无理数,不能写作两整数之比*****************************(一开始没想到,小学没学好) 此题1/n肯定是一个有限循环小数,了解这些后就能做此题了。 按照除法的机制,用一个函数表示出来就可以了,代码如下

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

购买磨轮平衡机时应该注意什么问题和技巧

在购买磨轮平衡机时,您应该注意以下几个关键点: 平衡精度 平衡精度是衡量平衡机性能的核心指标,直接影响到不平衡量的检测与校准的准确性,从而决定磨轮的振动和噪声水平。高精度的平衡机能显著减少振动和噪声,提高磨削加工的精度。 转速范围 宽广的转速范围意味着平衡机能够处理更多种类的磨轮,适应不同的工作条件和规格要求。 振动监测能力 振动监测能力是评估平衡机性能的重要因素。通过传感器实时监

poj 3259 uva 558 Wormholes(bellman最短路负权回路判断)

poj 3259: 题意:John的农场里n块地,m条路连接两块地,w个虫洞,虫洞是一条单向路,不但会把你传送到目的地,而且时间会倒退Ts。 任务是求你会不会在从某块地出发后又回来,看到了离开之前的自己。 判断树中是否存在负权回路就ok了。 bellman代码: #include<stdio.h>const int MaxN = 501;//农场数const int

缓存雪崩问题

缓存雪崩是缓存中大量key失效后当高并发到来时导致大量请求到数据库,瞬间耗尽数据库资源,导致数据库无法使用。 解决方案: 1、使用锁进行控制 2、对同一类型信息的key设置不同的过期时间 3、缓存预热 1. 什么是缓存雪崩 缓存雪崩是指在短时间内,大量缓存数据同时失效,导致所有请求直接涌向数据库,瞬间增加数据库的负载压力,可能导致数据库性能下降甚至崩溃。这种情况往往发生在缓存中大量 k

6.1.数据结构-c/c++堆详解下篇(堆排序,TopK问题)

上篇:6.1.数据结构-c/c++模拟实现堆上篇(向下,上调整算法,建堆,增删数据)-CSDN博客 本章重点 1.使用堆来完成堆排序 2.使用堆解决TopK问题 目录 一.堆排序 1.1 思路 1.2 代码 1.3 简单测试 二.TopK问题 2.1 思路(求最小): 2.2 C语言代码(手写堆) 2.3 C++代码(使用优先级队列 priority_queue)

zoj 1721 判断2条线段(完全)相交

给出起点,终点,与一些障碍线段。 求起点到终点的最短路。 枚举2点的距离,然后最短路。 2点可达条件:没有线段与这2点所构成的线段(完全)相交。 const double eps = 1e-8 ;double add(double x , double y){if(fabs(x+y) < eps*(fabs(x) + fabs(y))) return 0 ;return x + y ;

POJ1269 判断2条直线的位置关系

题目大意:给两个点能够确定一条直线,题目给出两条直线(由4个点确定),要求判断出这两条直线的关系:平行,同线,相交。如果相交还要求出交点坐标。 解题思路: 先判断两条直线p1p2, q1q2是否共线, 如果不是,再判断 直线 是否平行, 如果还不是, 则两直线相交。  判断共线:  p1p2q1 共线 且 p1p2q2 共线 ,共线用叉乘为 0  来判断,  判断 平行:  p1p

Codeforces Round #113 (Div. 2) B 判断多边形是否在凸包内

题目点击打开链接 凸多边形A, 多边形B, 判断B是否严格在A内。  注意AB有重点 。  将A,B上的点合在一起求凸包,如果凸包上的点是B的某个点,则B肯定不在A内。 或者说B上的某点在凸包的边上则也说明B不严格在A里面。 这个处理有个巧妙的方法,只需在求凸包的时候, <=  改成< 也就是说凸包一条边上的所有点都重复点都记录在凸包里面了。 另外不能去重点。 int

【VUE】跨域问题的概念,以及解决方法。

目录 1.跨域概念 2.解决方法 2.1 配置网络请求代理 2.2 使用@CrossOrigin 注解 2.3 通过配置文件实现跨域 2.4 添加 CorsWebFilter 来解决跨域问题 1.跨域概念 跨域问题是由于浏览器实施了同源策略,该策略要求请求的域名、协议和端口必须与提供资源的服务相同。如果不相同,则需要服务器显式地允许这种跨域请求。一般在springbo