关于C语言除0引发的思考

代码如下:

<SPAN style="BACKGROUND-COLOR: rgb(241,254,221)">
<SPAN style="FONT-FAMILY: Microsoft YaHei">   
进行浮点数编程时,如果没有注意,常常会出现输出类似 1.#IND, 1.#INF 或者 nan, inf 之类奇怪的输出。这通常隐含了浮点数操作的异常。
</SPAN></SPAN>

进行整数除0的时候,程序会检查出一个错误,比如对于这个代码:


代码如下:

#include <stdio.h>
#include <math.h>

int main()
{
   int m;
   m=1/0;
   printf("%d",m);
    return 0;
}

运行完以后程序就会警告:warning: division by zero [-Wdiv-by-zero] 
但是对于浮点数,就没有类似的检查:


代码如下:

#include <stdio.h>
#include <math.h>

int main()
{
   double m;
   m=1.0/0.0;
   printf("%lf",m);
    return 0;
}

不会有警告或者报错,但是运行完以后会出现这样的值:-1.#IND00,不知所云。于是上网查了一下,原来是这样!

特殊浮点数的含义
1.#INF / inf:这个值表示“无穷大 (infinity 的缩写)”,即超出了计算机可以表示的浮点数的最大范围(或者说超过了 double 类型的最大值)。
例如,当用 0 除一个整数时便会得到一个1.#INF / inf值;相应的,如果用 0 除一个负整数也会得到 -1.#INF / -inf 值。    -1.#IND / nan:这个的情况更复杂,一般来说,它们来自于任何未定义结果(非法)的浮点数运算。"IND"是 indeterminate 的缩写,而"nan"是 not a number 的缩写。产生这个值的常见例子有:对负数开平方,对负数取对数,0.0/0.0,0.0*∞, ∞/∞ 等。  简而言之,如果遇到 1.#INF / inf,就检查是否发生了运算结果溢出除零,而遇到 1.#IND / nan,就检查是否发生了非法的运算。

特殊浮点数的判断
很多 C 库都提供了一组函数用来判断一个浮点数是否是无穷大或 NaN。int _isnan(double x) 函数用来判断一个浮点数是否是 NaN,而 int _finite(double x) 用以判断一个浮点数是否是无穷大。

你可能已经注意到了,上面两个函数都是以下划线开头的,因此在可移植性上可能是存在问题的,那么如何实现一个通用的判断版本呢?
首先,对于 Nan,可以用下面的代码实现: 


代码如下:

bool IsNumber(double x)
{
    // 这里的比较操作看上去总是会得到 true
    // 但有趣的是对于 IEEE 754 浮点数 NaN 来说总会得到 false!
    return (x == x);
}

而下面的代码可以判断一个浮点数是否是有限的(finite, 即既不是 NaN 又不是 infinite):


代码如下:

bool IsFiniteNumber(double x)
{
    return (x <= DBL_MAX && x >= -DBL_MAX);
}

其中,DBL_MAX 是 <float.h> 中预定义的常量。   
把上面两个函数结合起来,还可以实现一个浮点数是否是 Inf 的判断。

(0)

相关推荐

  • 关于C语言除0引发的思考

    复制代码 代码如下: <SPAN style="BACKGROUND-COLOR: rgb(241,254,221)"><SPAN style="FONT-FAMILY: Microsoft YaHei">    进行浮点数编程时,如果没有注意,常常会出现输出类似 1.#IND, 1.#INF 或者 nan, inf 之类奇怪的输出.这通常隐含了浮点数操作的异常.</SPAN></SPAN> 进行整数除0的时候,程序会

  • 有关.NET参数传递的方式引发的思考

    下面就简单的介绍一下.NET的一些常用参数用法,如有不足还望指正,也欢迎大家在下面留言讨论,分享自己的见解. 一.DotNet参数概述: .NET中参数(形式参数)变量是方法或索引器声明的一部分,而实参是调用方法或索引器时使用的表达式. 在CLR中,默认的情况下所有的方法参数都是传值的.在传递引用类型的对象时,对一个对象的引用会传递给方法.这里的船引用本身是以传值的方式传给方法的.这也意味着方法能够修改对象,而调用者能看到这些修改.对于值类型的实例,传给方法的实例的一个副本.意味着方法将获得它专

  • 关于ORACLE通过file_id与block_id定位数据库对象遇到的问题引发的思考

    在ORACLE中,我们可以通过file_id(file#)与block_id(block#)去定位一个数据库对象(object).例如,我们在10046生成的trace文件中file#=4 block#=266 blocks=8,那么我可以通过下面两个SQL去定位对象 SQL 1:此SQL效率较差,执行时间较长. SELECT OWNER, SEGMENT_NAME, SEGMENT_TYPE, TABLESPACE_NAME FROM DBA_EXTENTS WHERE FILE_ID =&F

  • C语言数组越界引发的死循环问题解决

    目录 一.引入 二.代码缺陷 三.为什么会死循环? 四.补充说明 五.总结 一.引入 下面的程序在VS编译器会出现什么问题?运行结果是什么?为什么? #include <stdio.h> int main() { int i = 0; int arr[] = {1,2,3,4,5,6,7,8,9,10}; for(i=0; i<=12; i++) { arr[i] = 0; printf("hello\n"); } return 0; } 运行结果: 如下图:代码死循

  • 由php的call_user_func传reference引发的思考

    问题的提出 网友bercmisir在院内留言,针对php手册中的call_user_func函数的文档一事,大致如下: http://php.net/manual/en/function.call-user-func.php 其中parameter下有这样一句话: Note: Note that the parameters for call_user_func() are not passed by reference. 简单地翻译一下,是说这个函数的参数是不能依靠引用来传递的. 还有一个例子

  • .Net页面局部更新引发的思考

    最近在修改以前做的模块,添加一个新功能.整理了下才发现重用率很低,大部分的东西还是需要重新写.功能里用到了局部更新,所有整理一下一路来实现局部更新的解决方案及改进. 我接触的项目开发大多是以Asp.net WebForm开发的,自然会用到UpdatePanel,好处就是开发快.方便,当然产生的问题也是一大堆.然后是Ajax和一般处理程序配合实现异步请求更新.最后就是利用第三方绑定插件优化Ajax请求.  一.UpdatePanel  将需要更新的模块放入UpdatePanel的ContentTe

  • 基于Log4j2阻塞业务线程引发的思考

    目录 问题描述 问题1 问题2:异常线程栈打印使用讨论 ThrowableProxy使用错误的CCL原因分析 异步Appender追加日志 创建log4j日志事件 创建ThrownProxy代理 为什么同一个类会加载多次? GeneratedMethodAccessor类 问题总结 问题1 问题2 问题描述 问题1 异步日志打印在ringbuffer满了之后2.7版本的log4j2会默认使用当前线程进行打印日志. 即使不使用默认的策略,2.9之后已经改为默认的为enqueue方式,也会因为最后队

  • php 无法加载mysql的module的时候的配置的解决方案引发的思考

    之后看phpinfo() 里 确实也没找到mysql 模块, 之后所谓的解决方案如"将php.ini" 放入C:\Windows 环境变量等不靠谱说法..... 甚至拷贝ext的文件夹的dll 到System32 中等 统统不靠谱 直到看到这篇帖子,我才发现问题 http://www.haosblog.com/index.php?mod=article_read&id=322 "mysql无法找到的原因是mysql的运行库无法找到,打开mysql的安装文件夹,在bin

  • 基于React.js实现原生js拖拽效果引发的思考

    一.起因&思路 一直想写一个原生js拖拽效果,又加上近来学react学得比较嗨.所以就用react来实现这个拖拽效果. 首先,其实拖拽效果的思路是很简单的.主要就是三个步骤: 1.onmousedown的时候,启动可拖拽事件,记录被拖拽元素的原始坐标参数. 2.onmousemove的时候,实时记录鼠标移动的距离,结合被拖拽元素第一阶段的坐标参数,计算并设置新的坐标值. 3.onmouseup的时候,关闭可拖拽事件,记录新的坐标值. 注意:这里主要是通过绝对定位的top和left来确定元素的位置

  • javascript 一段代码引发的思考第1/2页

    在2008年的最后一天,在此祝愿大家元旦快乐!!! 郑重声明:此问题根本不是问题,现在看来就是本人知识匮乏,庸人自扰,望广大朋友勿喷!! 细心发现问题,耐心解决问题,信心面对问题. 作者:白某人 长话短说:"服务员,上代码...." 测试代码: function init(){ var tpl = new Ext.Template(' this is div{id} '); tpl.append('div1',{id:'2'}); tpl.insertAfter('div2',{id:

随机推荐