简述:
《Java 解惑》 第二章 表达式之谜
谜题1:奇数性
package com.anialy.test.java_puzzlers.chapter_1;
public class 奇数性 {
public static void main(String[] args) {
System.out.println("-1: " + isOdd(-1));
}
private static boolean isOdd(int a){
return a % 2 == 1;
}
}
输出:
原因:
Java中对于所有int数值a和所有的非零int数值b,都满足下面的恒等式
(a / b) * b + (a % b) == a
当取余操作返回一个非零的结果时,它与左操作数,也就是a具有相同的正负符号
解决方式:
1)右数使用零
private static boolean isOdd(int a){
return a % 2 != 0;
}
2)用“&”与操作
private static boolean isOdd(int a){
return (a & 1) != 0;
}
谜题2: 找零时刻
package com.anialy.test.java_puzzlers.chapter_1;
public class 找零时刻 {
public static void main(String[] args) {
System.out.println(2.0 - 1.1);
}
}
输出:
原因:
不是所有的小数都可以用二进制浮点数精确表示, 对于货币计算尽量使用int, long, BigDecimal
解决方式:
1)通过printf设置输出精度
package com.anialy.test.java_puzzlers.chapter_1;
public class 找零时刻 {
public static void main(String[] args) {
System.out.printf("%.1f", 2.0 - 1.1);
}
}
2)使用BigDecimal
package com.anialy.test.java_puzzlers.chapter_1;
import java.math.BigDecimal;
public class 找零时刻 {
public static void main(String[] args) {
System.out.println(new BigDecimal("2.0")
.subtract(new BigDecimal("1.1")));
}
}
谜题3: 长整除
package com.anialy.test.java_puzzlers.chapter_2;
public class 长整除 {
public static void main(String[] args) {
final long MICROS_PER_DAY = 24 * 60 * 60 * 1000 * 1000;
final long MILLIS_PER_DAY = 24 * 60 * 60 * 1000;
System.out.println(MICROS_PER_DAY / MILLIS_PER_DAY);
}
}
输出:
原因:
右侧的操作均为int型,从int提升为long是一种拓宽基本类型转换
解决方式:
1) 增加L标示符
谜题4:初级问题
在使用过程中,如果标示为long的数字,不要用小写,以防看不清
谜题5:十六进制的趣事
package com.anialy.test.java_puzzlers.chapter_2;
public class 十六进制的趣事 {
public static void main(String[] args) {
System.out.println(Long.toHexString(
0x100000000L + 0xcafebabe));
}
}
输出:
原因:
左操作数, 0x100000000L
右操作数,0xcafebabe
其中,两个数字相加时, 由于十六进制的数字并不是通过负号来标示正负值的,所以对于最高位为c的右操作数, 在运算时,从int拓宽基本类型至long后,高位
都用 f 填充(编译器判断为负数的缺省操作),即0xffffffffcafebabeL 之后这个数字加上左操作数0x0000000100000000L 得到的就是0x00000000cafebabeL
谜题6:多重转型
package com.anialy.test.java_puzzlers.chapter_2;
public class 多重转型 {
public static void main(String[] args) {
System.out.println((int)(char)(byte) -1);
}
}
输出:
原因:
byte是有符号类型,在byte数值-1转换为char时,会发生符号扩展,char(为2个byte)数值的16位都被置位了,等于2^16 - 1
此外,如果最初的数值类型是有符号的,就执行符号扩展;如果是char, 那么不管要被转换为什么类型,一律是零扩展(高位用0补足)
解决方式:
1)char 不希望进行符号扩展,则需要使用一个位掩码来明确表示
(int)((char) 'A' & 0xffff)
2)
如果一个byte数值转换为char时, 无符号扩展,那就必须使用一个位掩码做限制
(byte) -1 & 0xff
谜题7: 互换内容
package com.anialy.test.java_puzzlers.chapter_2;
public class 互换内容 {
public static void main(String[] args) {
int x = 1984;
int y = 2001;
x^= y^= x^= y;
System.out.printf("x: %d , y: %d", x, y);
}
}
输出:
原因:
Java操作数是从左向右求值的
解决方式:
1)改变结算顺序呢,从左向右
package com.anialy.test.java_puzzlers.chapter_2;
public class 互换内容 {
public static void main(String[] args) {
int x = 1984;
int y = 2001;
x = (y ^= (x ^= y)) ^ x;
System.out.printf("x: %d , y: %d", x, y);
}
}
谜题8: Dos Equis
package com.anialy.test.java_puzzlers.chapter_2;
public class DosEquis {
public static void main(String[] args) {
char x = 'X';
int i = 0;
System.out.println(true ? x : 0);
System.out.println(false ? i : x);
}
}
输出:
原因:
1)如果第二个和第三个操作数具有相同的类型,那么它就是条件表达式的类型。
2)如果一个操作输的类型是T, T表示byte、short或char, 而另一个操作数是一个int类型的常量表达式,它的值可以用类型T表示,那么条件表达式的类型就是T
3)否则,将对操作数类型进行二进制数字提升,而条件表达式的类型就是第二个和第三个操作数被提升之后的类型
谜题9:半斤
x += i 合法,x = x + i 不合法
原因:
对于复合赋值,简单的赋值是非法的,这需要显示的转型。
谜题10: 八两
x = x + i 合法,x += i 不合法
原因:
右侧是String类型的情况下,“+=” 操作符不允许左侧是Object类型