- 浏览: 286965 次
- 性别:
- 来自: 杭州
文章分类
最新评论
-
sjx19871109:
有一个疑问,博主在做循环的时候,for(int i=0;i&l ...
ArrayList:用add代替remove -
剑锋凛冽:
不错,看了很有帮助。但有个概念不是很清楚,锁投票是什么?
java中的lock和synchronized -
星期扒的幻想:
学习了,了解了
Solr增删改查 -
programming:
很蛋痛的webx 工程与jarsource编码不一直,相关 ...
Webx3 -
xjt927:
...
Solr增删改查
转自:http://jiangzhengjun.iteye.com/blog/652623
数值表达式
1. 奇偶判断
不要使用 i % 2 == 1 来判断是否是奇数,因为i为负奇数时不成立,请使用 i % 2 != 0 来判断是否是奇数,或使用
高效式 (i & 1) != 0来判断。
2. 小数精确计算
- System.out.println(2.00 -1.10);//0.8999999999999999
上面的计算出的结果不是 0.9,而是一连串的小数。问题在于1.1这个数字不能被精确表示为一个double,因此它被表
示为最接近它的double值,该程序从2中减去的就是这个值,但这个计算的结果并不是最接近0.9的double值。
一般地说,问题在于并不是所有的小数都可以用二进制浮点数精确表示。
二进制浮点对于货币计算是非常不适合的,因为它不可能将1.0表示成10的其他任何负次幂。
解决问题的第一种方式是使用货币的最小单位(分)来表示:
- System.out.println(200-110);//90
第二种方式是使用BigDecimal,但一定要用BigDecimal(String)构造器,而千万不要用BigDecimal(double)来构造(
也不能将float或double型转换成String再来使用BigDecimal(String) 来构造,因为在将float或double转换成String
时精度已丢失)。例如new BigDecimal(0.1),它将返回一个BigDecimal,也即
0.1000000000000000055511151231257827021181583404541015625 ,正确使用BigDecimal,程序就可以打印出我们所期
望的结果0.9:
- System.out.println(new BigDecimal("2.0").subtract(new BigDecimal("1.10")));// 0.9
另外,如果要比较两个浮点数的大小,要使用BigDecimal的compareTo方法。
如果你还想更深入了解下,请参考《 Java中的浮点数剖析 》!
3. int整数相乘溢出
我们计算一天中的微秒数:
- long microsPerDay = 24 * 60 * 60 * 1000 * 1000 ;// 正确结果应为:86400000000
- System.out.println(microsPerDay);// 实际上为:500654080
- long microsPerDay = 24 * 60 * 60 * 1000 * 1000;// 正确结果应为:86400000000
- System.out.println(microsPerDay);// 实际上为:500654080
问题在于计算过程中溢出了。这个计算式完全是以int运算来执行的,并且只有在运算完成之后,其结果才被提升为
long,而此时已经太迟:计算已经溢出。
解决方法使计算表达式的第一个因子明确为long型,这样可以强制表达式中所有的后续计算都用long运算来完成,这
样结果就不会溢出:
- long microsPerDay = 24L * 60 * 60 * 1000 * 1000;
4. 负的十六进制与八进制字面常量
“数字字面常量”的类型都是int型,而不管他们是几进制,所以“2147483648”、“0x180000000(十六进制,共33
位,所以超过了整数的取值范围)”字面常量是错误的,编译时会报超过int的取值范围了,所以要确定以long来表示
“2147483648L”、“0x180000000L”。
十进制字面常量只有一个特性,即所有的十进制字面常量都是正数,如果想写一个负的十进制,则需要在正的十进制
字面常量前加上“-”即可。
十六进制或八进制字面常量可就不一定是正数或负数,是正还是负,则要根据当前情况看:如果十六进制和八进制字
面常量的最高位被设置成了1,那么它们就是负数:
- System.out.println(0x80 );//128
- //0x81看作是int型,最高位(第32位)为0,所以是正数
- System.out.println(0x81 );//129
- System.out.println(0x8001 );//32769
- System.out.println(0x70000001 );//1879048193
- //字面量0x80000001为int型,最高位(第32位)为1,所以是负数
- System.out.println(0x80000001 );//-2147483647
- //字面量0x80000001L强制转为long型,最高位(第64位)为0,所以是正数
- System.out.println(0x80000001L);//2147483649
- //最小int型
- System.out.println(0x80000000 );//-2147483648
- //只要超过32位,就需要在字面常量后加L强转long,否则编译时出错
- System.out.println(0x8000000000000000L);//-9223372036854775808
- System.out.println(0x80);//128
- //0x81看作是int型,最高位(第32位)为0,所以是正数
- System.out.println(0x81);//129
- System.out.println(0x8001);//32769
- System.out.println(0x70000001);//1879048193
- //字面量0x80000001为int型,最高位(第32位)为1,所以是负数
- System.out.println(0x80000001);//-2147483647
- //字面量0x80000001L强制转为long型,最高位(第64位)为0,所以是正数
- System.out.println(0x80000001L);//2147483649
- //最小int型
- System.out.println(0x80000000);//-2147483648
- //只要超过32位,就需要在字面常量后加L强转long,否则编译时出错
- System.out.println(0x8000000000000000L);//-9223372036854775808
从上面可以看出,十六进制的字面常量表示的是int型,如果超过32位,则需要在后面加“L”,否则编译过不过。如
果为32,则为负int正数,超过32位,则为long型,但需明确指定为long。
- System.out.println(Long.toHexString(0x100000000L + 0xcafebabe));// cafebabe
结果为什么不是0x1cafebabe?该程序执行的加法是一个混合类型的计算:左操作数是long型,而右操作数是int类型
。为了执行该计算,Java将int类型的数值用拓宽原生类型转换提升为long类型,然后对两个long类型数值相加。因为
int是有符号的整数类型,所以这个转换执行的是符号扩展。
这个加法的右操作数0xcafebabe为32位,将被提升为long类型的数值0xffffffffcafebabeL,之后这个数值加上了左操
作0x100000000L。当视为int类型时,经过符号扩展之后的右操作数的高32位是-1,而左操作数的第32位是1,两个数
值相加得到了0:
0x 0xffffffffcafebabeL
+0x 0000000100000000L
-----------------------------
0x 00000000cafebabeL
如果要得到正确的结果0x1cafebabe,则需在第二个操作数组后加上“L”明确看作是正的long型即可,此时相加时拓
展符号位就为0:
- System.out.println(Long.toHexString(0x100000000L + 0xcafebabeL));// 1cafebabe
5. 窄数字类型提升至宽类型时使用符号位扩展还是零扩展
- System.out.println((int)(char)(byte)-1);// 65535
结果为什么是65535而不是-1?
窄的整型转换成较宽的整型时符号扩展规则:如果最初的数值类型是有符号的,那么就执行符号扩展(即如果符号位
为1,则扩展为1,如果为零,则扩展为0);如果它是char,那么不管它将要被提升成什么类型,都执行零扩展。
了解上面的规则后,我们再来看看迷题:因为byte是有符号的类型,所以在将byte数值-1(二进制为:11111111)提
升到char时,会发生符号位扩展,又符号位为1,所以就补8个1,最后为16个1;然后从char到int的提升时,由于是
char型提升到其他类型,所以采用零扩展而不是符号扩展,结果int数值就成了65535。
如果将一个char数值c转型为一个宽度更宽的类型时,只是以零来扩展,但如果清晰表达以零扩展的意图,则可以考虑
使用一个位掩码:
- int i = c & 0xffff;//实质上等同于:int i = c ;
如果将一个char数值c转型为一个宽度更宽的整型,并且希望有符号扩展,那么就先将char转型为一个short,它与
char上个具有同样的宽度,但是它是有符号的:
- int i = (short)c;
如果将一个byte数值b转型为一个char,并且不希望有符号扩展,那么必须使用一个位掩码来限制它:
- char c = (char)(b & 0xff);// char c = (char) b;为有符号扩展
6. ((byte)0x90 == 0x90)?
答案是不等的,尽管外表看起来是成立的,但是它却等于false。为了比较byte数值(byte)0x90和int数值0x90,Java
通过拓宽原生类型将byte提升为int,然后比较这两个int数值。因为byte是一个有符号类型,所以这个转换执行的是
符号扩展,将负的byte数值提升为了在数字上相等的int值(10010000111111111111111111111111 10010000)。在本例中,该转换将(byte)0x90提升为int数值-112,它不等于int数值的0x90,即+144。
解决办法:使用一个屏蔽码来消除符号扩展的影响,从而将byte转型为int。
- ((byte)0x90 & 0xff)== 0x90
7. 三元表达式(?:)
- char x = 'X' ;
- int i = 0 ;
- System.out.println(true ? x : 0 );// X
- System.out.println(false ? i : x);// 88
- char x = 'X';
- int i = 0;
- System.out.println(true ? x : 0);// X
- System.out.println(false ? i : x);// 88
条件表达式结果类型的规则:
(1) 如果第二个和第三个操作数具有相同的类型,那么它就是条件表达式的类型。
(2) 如果一个操作的类型是T,T表示byte、short或char,而另一个操作数是一个int类型的“字面常量”,并且
它的值可以用类型T表示,那条件表达式的类型就是T。
(3) 否则,将对操作数类型进行提升,而条件表达式的类型就是第二个和第三个操作被提升之后的类型。
现来使用以上规则解上面的迷题,第一个表达式符合第二条规则:一个操作数的类型是char,另一个的类型是字面常
量为0的int型,但0可以表示成char,所以最终返回类型以char类型为准;第二个表达式符合第三条规则:因为i为int
型变量,而x又为char型变量,所以会先将x提升至int型,所以最后的结果类型为int型,但如果将i定义成final时,
则返回结果类型为char,则此时符合第二条规则,因为final类型的变量在编译时就使用“字面常量0”来替换三元表
达式了:
- final int i = 0;
- System.out.println(false ? i : x);// X
在JDK1.4版本或之前,条件操作符 ?: 中,当第二个和延续三个操作数是引用类型时,条件操作符要求它们其中一个
必须是另一个的子类型,那怕它们有同一个父类也不行:
- public class T {
- public static void main(String[] args) {
- System.out.println(f());
- }
- public static T f() {
- // !!1.4不能编译,但1.5可以
- // !!return true?new T1():new T2();
- return true ? (T) new T1() : new T2();// T1
- }
- }
- class T1 extends T {
- public String toString() {
- return "T1" ;
- }
- }
- class T2 extends T {
- public String toString() {
- return "T2" ;
- }
- }
- public class T {
- public static void main(String[] args) {
- System.out.println(f());
- }
- public static T f() {
- // !!1.4不能编译,但1.5可以
- // !!return true?new T1():new T2();
- return true ? (T) new T1() : new T2();// T1
- }
- }
- class T1 extends T {
- public String toString() {
- return "T1";
- }
- }
- class T2 extends T {
- public String toString() {
- return "T2";
- }
- }
在5.0或以上版本中,条件操作符在延续二个和第三个操作数是引用类型时总是合法的。其结果类型是这两种类型的最
小公共超类。公共超类总是存在的,因为Object是每一个对象类型的超类型,上面的最小公共超类是T,所以能编译。
发表评论
-
多线程 Future 异常处理
2017-03-14 19:15 1374http://www.blogjava.net/xylz/ar ... -
问题排查记录-linux
2016-08-05 12:02 854### 以下所有命令都要先sudo su admin ... -
打完整含依赖包的jar包
2015-04-10 13:32 667如下,可以打出带依赖包的jar package.ba ... -
打印线程详细信息并按CPU占用排序
2014-05-28 17:20 739好久没来写了,不过太好用了,记录下 #!/bin/ksh ... -
java hashCode详解
2013-05-17 21:12 2821=============================== ... -
PO、DO、VO的区别
2013-05-17 16:53 2773Rational Mapping( ... -
java.util.concurrent 包
2013-04-03 11:09 1076java.util.concurrent 包含许多线程安全、 ... -
BTrace使用简介
2013-03-27 21:05 836很多时候在online的应用出现问题时,很多时候我们需要知道 ... -
BTrace使用总结
2013-03-27 21:01 1117一、背景 在生产环境中可能经常 ... -
java Singleton 的使用详解
2013-03-21 16:15 833概念: 在Java应用程序中,一个类Class只有一个实 ... -
java中的lock和synchronized
2013-03-21 10:04 9451、ReentrantLock 拥有Synchronized相 ... -
Java 理论与实践: JDK 5.0 中更灵活、更具可伸缩性的锁定机制
2013-03-21 09:58 693多线程和并发性并不是什么新内容,但是 Java 语言设计中的 ... -
Java5中lock
2013-03-20 11:24 998说明:Java多线程相关相关几篇文章转自http://lav ... -
栈与堆的困惑--java内存解析
2013-03-19 19:12 697----对这两个概念的不明好久,终于找到一篇好文,拿来共享1. ... -
Java内存之"栈"与"堆"
2013-01-30 20:13 757引用: http://1615926449.iteye.co ... -
Java性能优化技巧之尽可能使用局部变量
2012-08-03 20:10 990如果你频繁存取变量,就需要考虑从何处 ... -
velocity 循环set null
2012-08-03 20:10 1100今天很不幸又看到中站一个故障,又是因为在velocity模板中 ... -
Hash碰撞/ Hash Collision
2012-01-10 19:30 1363最近一个Hash Collision DoS(Hash碰撞的拒 ... -
java回调函数的超形象例子
2011-10-20 10:20 1195无意间看到这个哥们的牛X解释: 下面举个通俗的例子: ... -
JDK中的URLConnection参数详解
2011-09-03 18:36 726JDK中的URLConnection参数详解 针对JD ...
相关推荐
1. **类型转换**:Java中的类型转换包括自动类型提升和强制类型转换。例如,`double`到`int`的转换可能涉及到数值的丢失,而`int`到`double`则会自动进行。 2. **运算符优先级**:理解运算符的优先级是解决表达式...
9. **泛型**:泛型用于创建类型安全的容器,限制了只能存储特定类型的元素,避免了强制类型转换。 10. **反射**:Java反射机制允许程序在运行时检查类的信息,如字段、方法、构造器等,并能在运行时动态创建对象和...
另一方面,Java 丢弃了 C++ 中很少使用的、很难理解的、令人迷惑的那些特性,如操作符重载、多继承、自动的强制类型转换。特别地,Java 语言不使用指针,而是引用。并提供了自动分配和回收内存空间,使得程序员不必...
这些谜题涵盖了从基本语法到高级特性的各种主题,包括但不限于类型转换、对象引用、内存管理、多线程、异常处理、集合框架以及Java的API使用等。 1. 类型转换:Java是一种静态类型语言,但类型转换时可能出现问题,...
【Java解惑PPT1】深入探讨Java编程...5. 理解不同数据类型在运算时的类型转换规则,特别是与十六进制数字的结合。 通过对这些谜题的深入理解和解决,Java程序员可以提高代码质量,减少潜在的错误,提升程序的可靠性。
Java 语言丢弃了 C++ 中很少使用的、很难理解的、令人迷惑的那些特性,如操作符重载、多继承、自动的强制类型转换。特别地,Java 语言不使用指针,并提供了自动的废料收集,使得程序员不必为内存管理而担忧。 Java ...
Java IO 是Java编程语言中一个至关重要的部分,它不仅提供了丰富的API来处理各种数据源和目的地的输入输出操作,还通过细致的类设计帮助开发者高效地管理数据流。掌握Java IO 不仅能够提升编程技能,还能在处理文件...
- **强类型检查**:Java对数据类型的检查严格,不允许数据类型的向下转换和可能导致数据丢失的赋值操作。 - **int与Integer的区别**:int是基本数据类型,Integer是其对应的封装类,提供了更多的方法和功能。 - *...
2. Java语言丢弃了C++中很少使用的、很难理解的、令人迷惑的那些特性,如操作符重载、多继承、自动的强制类型转换。 3. Java语言提供类、接口和继承等原语,支持类之间的单继承和接口之间的多继承,并支持类与接口...
2. **类型转换**:C语言允许不同类型的变量之间进行转换,这可能导致数据类型的混淆,从而产生难以预见的行为。 3. **预处理器宏**:宏在C语言中是一种强大的工具,但过度使用或滥用可能导致宏地狱,使代码变得难以...
1.5.4 通用类型转换系统和属性格式化系统 1.5.5 数据访问层新增OXM功能 1.5.6 Web层的增强 1.5.7 其他 1.6 Spring对Java版本的要求 1.7 如何获取Spring 1.8 小结 第2章 快速入门 2.1 实例功能概述 2.1.1 比Hello ...
1.5.4 通用类型转换系统和属性格式化系统 1.5.5 数据访问层新增OXM功能 1.5.6 Web层的增强 1.5.7 其他 1.6 Spring对Java版本的要求 1.7 如何获取Spring 1.8 小结 第2章 快速入门 2.1 实例功能概述 2.1.1 比Hello ...
JavaScript 学习技巧是指在学习和使用 JavaScript 过程中需要掌握的一些重要技巧和知识点,涵盖从基本数据类型到高级应用场景。本文将总结和分享六个有用的 JavaScript 学习技巧,帮助开发者更好地掌握这门语言。 ...
- **基本型别偏执 (Primitive Obsession)**:过度依赖基本数据类型,而不是定义更高级的数据类型。 - **Switch语句 (Switch Statements)**:使用大量的条件分支语句,导致代码难以维护。 - **平行继承体系 (Parallel...
- **解释**: 尝试进行类型转换时,如果目标类型与源类型之间不存在继承关系,则转换是不可能的。 - **建议**: 检查类型转换逻辑,确保转换的可行性。 #### rule.findbugs.XSS_REQUEST_PARAMETER_TO_SEND_ERROR **...
1. **文本文件(Text File)**:文本文件是一种常见的文件类型,它以人类可读的形式存储数据,通常包括文字、数字和其他字符。与二进制文件不同,文本文件可以被各种文本编辑器打开和修改,如Notepad、Visual Studio...