《Java 解惑》 第二章 表达式之谜

简述:

《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类型









  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值