c语言算术运算符越界问题解决方案

2016-02-19 11:52 20 1 收藏

下面图老师小编要跟大家分享c语言算术运算符越界问题解决方案,简单的过程中其实暗藏玄机,还是要细心学习,喜欢还请记得收藏哦!

【 tulaoshi.com - 编程语言 】

大量的安全漏洞是由于计算机算术运算的微妙细节引起的, 具体的C语言, 诸如符号数和无符号数之间转换, 算术运算的越界都会导致不可预知的错误和安全漏洞, 具体的案例数不胜数.

作为一个系统程序员, 有必要对这些细节有深入的了解. 本篇参考csapp, 主要介绍如何判断算术运算的越界问题.

(虽然本篇的代码经过大量的测试, 但本人仍然无法保证代码的正确性, 希望大家纠错).
讲解的原则是"摆定理, 不证明, 写代码". 具体的证明过程在csapp中有详细的讲解, 也不是太难. 主要使用关键定理来写代码. Go~
问题一: 无符号数的加法越界问题
[定理]


[理解]
这个定理比较容易, 也比较能让人接受. 不解释啦.
代码如下:

/* Determine whether arguments can be added without overflow */
int uadd_ok(unsigned int x, unsigned int y)
{
return !(x+y x);
}

问题二: 无符号数的减法越界问题
[定理]


[理解]
1. 计算机中没有减法, x-y = x+(-y), 这里的-y就是上述的y的加法逆元. 不管是有符号还是无符号, 都是转换为加法运算. 只是加法逆元的定义不同.

3. C语言保证 -x = ~x+1; 可以验证这种方式与上面公式等价.
4. s=x-y = x+(-y). 那么 不会溢出 等价于 y不为0 或者 !uadd_ok(x, -y).


代码如下:

/* Determine whether argumnts can be substracted without overflow */
int usub_ok(unsigned int x, unsigned int y)
{
return !y || !uadd_ok(x, -y);
}

问题三: 无符号数的乘法越界问题
[定理]


[理解]
等价条件可以相互推导即可.
代码如下:

/* Determine whether arguments can be multiplied without overflow */
int umul_ok(unsigned int x, unsigned int y)
{
unsigned int p = x * y;
return !x || p/x==y;
}

问题四: 有符号数的加法越界问题
[定理]
对于两个有符号数x, y. 越界的等价条件是x,y为负数, x+y为正数或者x,y为正数, x+y为负数.
[理解]
这个定理比较容易.
代码如下:

/* Determine whether arguments can be added without overflow */
int tadd_ok(int x, int y)
{
return !(x0&&y0&&x+y0 || x0&&y0&&x+y0);
}

问题五: 有符号数的减法越界问题
[定理]

(本文来源于图老师网站,更多请访问http://www.tulaoshi.com/bianchengyuyan/)


[理解]
同无符号的减法一样, 只是加法逆元的定义不同, 但是位模式是一样的. C语言可以保证-x=~x+1. 同样也分两种情况讨论.见代码.

代码如下:

/* Determine whether arguments can be subtracted without overflow */
int tsub_ok(int x, int y)
{
  #if 0
  if (y == INT_MIN)
  return x0;
  else
  return tadd_ok(x, -y);
  #endif
  return y==INT_MIN&&x0 || y!=INT_MIN&&tadd_ok(x, -y);
}

问题六: 有符号数的乘法越界问题
[定理]
完全同无符号的乘法一样.
代码如下:

/* Determine whether arguments can be multiplied without overflow. */
int tmul_ok(int x, int y)
{
#if 0
int p = x * y;
return !x || p/x==y;
#endif
return umul_ok(x, y); /* 直接调用 */
}

(本文来源于图老师网站,更多请访问http://www.tulaoshi.com/bianchengyuyan/)

来源:http://www.tulaoshi.com/n/20160219/1598946.html

延伸阅读
标签: 电脑入门
空间发表的日志没有在QQ上更新 尝试按以下步骤解决: 确认您是否有在空间“个人档——权限管理——访问权限设置”中将“将QQ空间日志更新显示在QQ上”前面的打上勾?如果您没打勾,那么日志是不会更新显示在QQ上的。 若打上勾后仍没有更新显示,有可能是网络数据不同步引起的。建议您再上传一篇日志,然后点击空间中的“个...
标签: 电脑入门
内存做为电脑三大件配件之一,担负着数据的临时存取等任务。在使用过程中,难免会出现一些问题,如启动电脑却无法正常启动、无法进入操作系统或运行应用软件、无故经常死机等故障,这些问题的产生常会因为内存出现异常故障而导致操作失败。内存出现问题一部分是因为升级内存,由于内存种类的不匹配,往往也会遇到一些问题;另一部分是内存在使用...
标签: Web开发
=================================== 开发java应用出现乱码是很常见的,毕竟现在unicode的使用还不是很广泛,在使用gb2312(包含了gbk简体,big5繁体)的系统中要正确实现 中文的display和数据库的存储是最基本的要求。 ============================== 1,首先developer要明确自己为什么会遇到乱码,遇到什么样的乱码...
标签: 孕期
职场孕妈咪常见问题解决方案 现在的女性都比较独立,有自己的事业,很多女性在怀孕以后,出于这样那样的原因,依然无法远离工作岗位,不愿意在家里安心地养胎。这个时候在工作过程当中要注意什么,才能够更好地保护自己和保护肚子里的宝宝呢?我们一起来关注一下吧! 如果已经由医院方面确认说怀孕的话,应该将这件...
标签: 电脑入门
苹果电脑用bootcamp安装双系统,win7下,如果安装苹果自带的 bootcamp程序,或者bootcamp64. 则重启后将会蓝屏,报告驱动不兼容的问题。 网络上的解决思路是: 安好,重启,等待蓝屏重启。然后进入MacOSX系统,安装NTFS读取驱动paragon ntfs for mac。 然后重命名 C:/window/Systam32/driver32下的 applemtp.sys 以及 applemtn.sys 为其他...

经验教程

65

收藏

68
微博分享 QQ分享 QQ空间 手机页面 收藏网站 回到头部