五月的深圳空气中弥漫起初夏的味道,淡淡的,暖暖的。春日里不太张扬的阳光也掺入这股气息...(烟哥好文采!)

这天,烟哥愉快的喝着霸气芝士莓莓莓。一边东张西望,寻找着可以装13的机会。一切正如下面这张图这样



这时,小刘出现了!没错,就是那个大家期待的小刘出现了!她拿着一本《XXXjava笔试指南》来找烟哥。

只见小刘娴熟的打开这本书,望着整本书满满的笔记,烟哥不禁猜测道:"小刘如此热衷于学习,一定还是单身。"想到这里,烟哥不禁更有回答的动力了(没错,我就是这种人!)。

缓存问题

小刘翻到某一页后,指出下面这样一道题

public static void main(String[] args){
Integer a = 50;
Integer b = 50;
Integer c = 150;
Integer d = 150;
System.out.println(a==b);
System.out.println(c==d);
}

输出结果为

true
false

然后询问烟哥具体缘由。

烟哥看完以后,内心正(wei)直(suo)的笑了笑,心里想道:"这不是几年前的老题目了么,怎么现在还在考!"

烟哥回答道:"其实很简单,原理是下面这样滴!"

JAVA编译器编译Integer a = 50的时候,被翻译成-> Integer a = Integer.valueOf(50);

valueOf的源码是下面这样的

 public static Integer valueOf(int i) {
if (i >= IntegerCache.low && i <= IntegerCache.high)
return IntegerCache.cache[i + (-IntegerCache.low)];
return new Integer(i);
}

看到了嘛,Integer内部有一个IntegerCache缓存。对于值范围在-128到127之间的数,会进行缓存。因此a和b范围在-128到127之间,所以指向的是同一个对象,所以判断结果是true。而c和d在128之外,所以每次都是返回一个新对象,所以判断结果是false

听到这里,小刘似乎很满意的准备打道回府。

"不行,怎么能这样让小刘离开。我得多和小刘说说话!"

于是,烟哥说道:"小刘阿,其实java笔试里关于数值方面的陷阱,可以玩出很多花样,你了解多少?"

很自然的,小刘的表情是下面这样的

越界问题

烟哥说道:"这样吧,先问你一个问题。Math.abs(Integer.MIN_VALUE))的结果是正数还是负数?"

小刘:"Math.abs是绝对值运算,结果应该是正数吧。"

烟哥:"不对,还是负数。Integer的范围为-2147483648~2147483647。不过我们先来看一眼abs函数的源码,如下所示"

public static int abs(int a) {
return (a < 0) ? -a : a;
}

烟哥:"看了源码,其实很明显,绝对值运算的原理是判断这个数是否大于零,如果小于零则取负值。OK,回到我们题目。Integer.MIN_VALUE,它的十六进制表示是 0x80000000。其符号位为1,其余所有的位都是0。取负数(反码+1)则为 0x7fffffff+1,也就是 0x80000000。你会发现对Integer.MIN_VALUE取负值还是本身。因此,结果还是负数。"

小刘:"那你这套理论对LongShortByte都成立么?"

烟哥:"是的,都是成立的,原理都一样。你可以回去测试一下如下代码"

Short num =(short)Math.abs(Short.MIN_VALUE));
System.out.println(num);

小刘望着这些代码,陷入了思考...

突然,小刘回答道:"好像,之前我有看到一个题目是这样的。是否存在一个数i,可以使得i+1<i,这样看来,这个i就是Integer.MAX_VALUE,因为加完1就溢出了变为负值了。"

听小刘说道这里,烟哥换了一个角度问:"是否存在一个数,满足i != 0 && i == -i"

小刘想了下,机智的回答道:"其实还是Integer.MIN_VALUE,原因你刚才说过了!"

唉,没想到小刘领悟如此的快!

浮点奥秘

烟哥感慨小刘领悟速度的同时,加大难度问道:"是否存在一个数,满足i==i+1?"

小刘突然懵了,答道:"好像..似乎..应该一个数永远不会等于自己加一。"

看见小刘懵圈的眼神,烟哥只见自己装13的目的已经达到,便不再卖关子...

烟哥回答道:"有没听过一句话,无穷大加一个常数还是无穷大!所以,下面的例子输出为true"

double i = Double.POSITIVE_INFINITY;
System.out.println(i == i+1);

小刘反怼烟哥,说道:"其实,无穷大减去一个常数也是无穷大。所以下面例子也是输出为true,而且无穷小也有类似的特性。"

double i = Double.POSITIVE_INFINITY;
System.out.println(i == i-1);

烟哥满意的点了点头,感慨小刘成长真快!

突然,灵光一闪,烟哥补充道:"你知不知道有一个数可以满足i!=i?"

小刘再次陷入了深思...嘴里嘟囔道:"奇怪,还有一个数可以自己不等于自己么?"

望着小刘愁眉苦脸的眼神 ,烟哥答道:"对,有一个不是数值的值,它就是NaN,翻译过来就是(Not a Number),因此下面的输出是为true!"

double i = Double.NaN;
System.out.println(i != i);

结局

在烟哥一阵装13后,决定暴露自己的本性。问道:"小刘,你有对象了么?"

小刘答道:"烟哥,你是个好人,然而我已经有对象了!"

"Boom!"



(本文完!)

【原创】JAVA中令人眼花撩乱的数字魔法的更多相关文章

  1. Java中过滤出字母、数字和中文的正则表达式

    1.Java中过滤出字母.数字和中文的正则表达式 (1)过滤出字母的正则表达式 [^(A-Za-z)] (2)过滤出数字的正则表达式 [^(0-9)] (3)过滤出中文的正则表达式 [^(\\u4e0 ...

  2. AJPFX总结关于Java中过滤出字母、数字和中文的正则表达式

    1.Java中过滤出字母.数字和中文的正则表达式 (1)过滤出字母的正则表达式      [^(A-Za-z)] (2) 过滤出 数字 的正则表达式   [^(0-9)] (3) 过滤出 中文 的正则 ...

  3. (转载)java中判断字符串是否为数字的方法的几种方法

    java中判断字符串是否为数字的方法: 1.用JAVA自带的函数 public static boolean isNumeric(String str){ for (int i = 0; i < ...

  4. 字符串--java中判断字符串是否为数字的方法的几种方法?

    ava中判断字符串是否为数字的方法: 1.用JAVA自带的函数 public static boolean isNumeric(String str){ for (int i = 0; i < ...

  5. JAVA中令人疑惑的字符串

    Java中不同的字符串存在于同一个存储池中,字符串变量将指向存储池中相应的位置,也就是字符串变量里面包含的并不是字符串而是这个字符串对象的内存地址. String a = "123" ...

  6. java中判断字符串是否为数字的方法的几种方法

    1.用JAVA自带的函数 public static boolean isNumeric(String str){ for (int i = 0; i < str.length(); i++){ ...

  7. java中Number Type Casting(数字类型强转)的用法

    4.5 Number Type Casting(数字类型强转)隐式 casting(from small to big) byte a = 111; int b = a;显式 casting(from ...

  8. [原创]Java中的字符串比较,按照使用习惯进行比较

    java中的字符串比较一般可以采用compareTo函数,如果a.compareTo(b)返回的是小于0的数,那么说明a的unicode编码值小于b的unicode编码值. 但是很多情况下,我们开发一 ...

  9. Java中判断字符串是否为数字

    转载:https://blog.csdn.net/u013066244/article/details/53197756 用JAVA自带的函数 public static boolean isNume ...

随机推荐

  1. 数据库官方在线文档列表(mysql, postgreSQL)

    1. mysql http://dev.mysql.com/doc/ 2. postgreSQL https://www.postgresql.org/docs/

  2. 系统重装 WIN7如何创建和使用VHD文件

    1 在磁盘管理中,点击操作-创建VHD,然后可以创建一个空的VHD文件   2 右击这个磁盘,点击初始化磁盘,然后可以新建简单卷   3 右击这个磁盘,设置为脱机或者联机就可以在计算机中显示和隐藏这个 ...

  3. eclipse Kepler tomcat内存溢出解决方式

    使用eclipse开发ssh项目,本机8G内存,可是在打开一个表格后再打开一个页面.立即就内存溢出,网上搜到下面解决方式,未解决: 1.改动eclipse.ini參数 -vmargs -Xms1024 ...

  4. openERP邮件(发信、收信)

    openERP里的邮件处理主要有个2个模块处理 mail -核心 fetchmail -接收邮件     Alias domain和alias name     配置domain alias. Set ...

  5. centos Linux 常用命令汇总

    CentOS 关闭防火墙 1) 永久性生效,重启后不会复原 开启: chkconfig iptables on 关闭: chkconfig iptables off 2) 即时生效,重启后复原 开启: ...

  6. Elasticsearch 学习笔记 Elasticsearch及Elasticsearch head安装配置

    一.安装与配置 1.到官网下载Elasticsearch,https://artifacts.elastic.co/downloads/elasticsearch/elasticsearch-5.6. ...

  7. 全志Android SDK编译详解(一)

    1.编译环境说明 Linux 服务器(任意电脑都可以,建议选择配置高的)安装ubuntu 10.04 作为编译的主机,其他开发者在xp 下,通过SSH/Telent 等协议登录linux 主机进行编. ...

  8. A JavaScript library for reading EXIF meta data from image files.

    exif-js/exif-js: JavaScript library for reading EXIF image metadata https://github.com/exif-js/exif- ...

  9. dataTables-details 1.9

    本文共四部分:官网 | 基本使用|遇到的问题|属性表 一:官方网站:[http://www.datatables.net/] 二:基本使用:[http://www.guoxk.com/node/jqu ...

  10. CodeForces 559C Gerald and Gia (格路+容斥+DP)

    CodeForces 559C Gerald and Gia 大致题意:有一个 \(N\times M\) 的网格,其中有些格子是黑色的,现在需要求出从左上角到右下角不经过黑色格子的方案数(模 \(1 ...