[java各种疑问解答]数值表达式

http://sishuok.com/forum/posts/list/775.html

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方法。

3.      int整数相乘溢出

我们计算一天中的微秒数:

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

 

从上面可以看出,十六进制的字面常量表示的是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

+0x0000000100000000L

-----------------------------

 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值(10010000à111111111111111111111111 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

 

条件表达式结果类型的规则:

(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";

              }

}

在5.0或以上版本中,条件操作符在延续二个和第三个操作数是引用类型时总是合法的。其结果类型是这两种类型的最小公共超类。公共超类总是存在的,因为Object是每一个对象类型的超类型,上面的最小公共超类是T,所以能编译。

8.      +=复合赋值问题

x+=i与x=x+i等效吗,许多程序员都会认为第一个表达式x+=i只是第二个表达式x=x+i的简写方式,但这并不准确。

 

Java语言规范中提到:复合赋值 E1 op= E2等价于简单赋值 E1 = (T)((E1) op (E2)),其中T是E1的类型。

 

复合赋值表达式自动地将所执行计算的结果转型为其左侧变量的类型。如果结果的类型与该变量的类型相同,那么这个转型不会造成任何影响,然而,如果结果的类型比该变量的类型要宽,那么复合赋值操作符将悄悄地执行一个窄化原生类型转换,这样就会导致结果不正确:

short x=0;

int i = 123456;

x +=i;

System.out.println(x);//-7616

 

使用简单的赋值方式就不会有这样的问题了,因为宽类型不能自动转换成窄的类型,编译器会报错,这时我们就会注意到错误:x = x + i;//编译通不过

 

请不要将复合赋值操作符作用于byte、short或char类型的变量;在将复合赋值操作符作用于int类型的变量时,要确保表达式右侧不是long、float或double类型;在将复合赋值操作符作用于float类型的变量时,要确保表达式右侧不是double类型。其实一句:不要将让左侧的类型窄于右侧的数字类型。

 

总之,不要在short、byte或char类型的变量之上使用复合赋值操作符,因为这一过程会伴随着计算前类型的提升与计算后结果的截断,导致最后的计算结果不正确。

9.      i =++i;与i=i++;的区别

int i = 0;

i = i++;

System.out.println(i);

上面的程序会输出什么?大部分会说是 1,是也,非也。运行时正确结果为0。

 

i=++i;相当于以下二个语句(编译时出现警告,与i=i;警告相同):

i=i+1;

i=i;

 

i = i++;相当于以下三个语句:

int tmp = i;

i = i + 1;

i = tmp;

 

下面看看下面程序片段:

int i = 0, j = 0, y = 0;

i++;//相当于:i=i+1;

System.out.println("i=" + i);// i=1

++i;//相当于:i=i+1;

System.out.println("i=" + i);// i=2

i = i++;//相当于:int tmp=i;i=i+1;i=tmp;

System.out.println("i=" + i);// i=2

i = ++i;//编译时出现警告,与i=i;警告相同。相当于:i=i+1;i=i;

System.out.println("i=" + i);// i=3

j = i++;//相当于:int tmp=i;i=i+1;j=tmp;

System.out.println("j=" + j);// j=3

System.out.println("i=" + i);// i=4

y = ++i;//相当于:i=i+1;y=i;

System.out.println("y=" + y);// y=5

System.out.println("i=" + i);// i=5

10. Integer.MAX_VALUE + 1=?

System.out.println(Integer.MAX_VALUE + 1);

上面的程序输出多少?2147483647+1=2147483648?答案为-2147483648。

 

查看源码Integer.MAX_VALUE 为MAX_VALUE = 0x7fffffff;所以加1后为0x80000000,又0x80000000为整型字面常量,满了32位,且最位为1,所以字面上等于 -0,但又由于 -0就是等于0,所以-0这个编码就规定为最小的负数,32位的最小负数就是-2147483648。

11. -1<<32=?、-1<<65=?

如果左操作数是int(如果是byte、short、char型时会提升至int型再进行位操作)型,移位操作符只使用其右操作数的低5位作为移位长度(也就是将右操作数除以32取余);如果左操作数是long型,移位操作符只使用其右操作数的低6位作为移位长度(也就是将右操作数除以64取余);

 

再看看下面程序片段就会知道结果:

System.out.println(-1 << 31);// -2147483648 向左移31%32=31位

System.out.println(-1 << 32);// -1 向左移32%32=0位

System.out.println(-1 << 33);// -2 向左移33%32=1位

System.out.println(-1 << 1);// -2 向左移1%32=1位

 

System.out.println(-1L << 63);// -9223372036854775808 向左移63%64=63位

System.out.println(-1L << 64);// -1 向左移64%64=0位

System.out.println(-1L << 65);// -2 向左移65%64=1位

System.out.println(-1L << 1);// -2 向左移1%64=1位

 

byte b = -1;// byte型在位操作前类型提升至int

System.out.println(b << 31);// -2147483648 向左移31%32=31位

System.out.println(b << 63);// -2147483648 向左移63%32=31位

 

short s = -1;// short型在位操作前类型提升至int

System.out.println(s << 31);// -2147483648 向左移31%32=31位

System.out.println(s << 63);// -2147483648 向左移63%32=31位

 

char c = 1;// char型在位操作前类型提升至int

System.out.println(c << 31);// -2147483648 向左移31%32=31位

System.out.println(c << 63);// -2147483648 向左移63%32=31位

12. 一个数永远不会等于它自己加1吗?i==i+1

一个数永远不会等于它自己加1,对吗?如果数字是整型,则对;如果这个数字是无穷大或都是浮点型足够大(如1.0e40),等式就可能成立了。

 

Java强制要求使用IEEE 754浮点数算术运算,它可以让你用一个double或float来表示无穷大。

 

浮点型分为double型、float型。

 

无穷分为正无穷与负无穷。

 

无穷大加1还是无穷大。

 

一个浮点数值越大,它和其后继数值之间的间隔就越大。

 

对一个足够大的浮点数加1不会改变它的值,因为1不足以“填补它与其后者之间的空隙”。

 

浮点数操作返回的是最接近其精确数学结果的浮点数值。

 

一旦毗邻的浮点数值之间的距离大于2,那么对其中的一个浮点数值加1将不会产生任何效果,因为其结果没有达到两个数值之间的一半。对于float类型,加1不会产生任何效果的最小数是2^25,即33554432;而对于double类型,最小数是2^54,大约是1.8*10^16。

 

33554432F转二进制过程:

33554432的二进制为:10000000000000000000000000,将该二进制化成规范的小数二进制,即小数从右向左移25位1.0000000000000000000000000,化成浮点数二进制0,25+127, 00000000000000000000000 00(丢弃最后两位),即0, 10011000, 00000000000000000000000,最后的结果为1.00000000000000000000000*2^25

 

 

毗邻的浮点数值之间的距离被称为一个ulp,它是最小单位(unit in the last place)的首字母缩写。在5.0版本中,引入了Math.ulp方法来计算float或double数值的ulp。

 

二进制浮点算术只是对实际算术的一种近似。

 

// 注,整型数不能被 0 除,即(int)XX/0运行时抛异常

double i = 1.0 / 0.0;// 正无穷大

double j = -1.0 / 0.0;// 负无穷大

// Double.POSITIVE_INFINITY定义为:POSITIVE_INFINITY = 1.0 / 0.0;

System.out.println(i + " " + (i == Double.POSITIVE_INFINITY));//Infinity true

// Double.NEGATIVE_INFINITY定义为:NEGATIVE_INFINITY = -1.0 / 0.0;

System.out.println(j + " " + (j == Double.NEGATIVE_INFINITY));//-Infinity true

System.out.println(i == (i + 1));// true

System.out.println(0.1f == 0.1);// false

float f = 33554432;

System.out.println(f + " " + (f==(f+1)));//3.3554432E7 true

13. 自己不等于自己吗?i!=i

NaN(Not a Number)不等于任何数,包括它自身在内。

 

double i = 0.0/0.0;可表示NaN。

 

float和double类型都有一个特殊的NaN值,Double.NaN、Float.NaN表示NaN。

 

如果一个表达式中产生了NaN,则结果为NaN。

 

System.out.println(0.0 / 0.0);//NaN

System.out.println(Double.NaN + " " + (Double.NaN == (0.0 / 0.0)));//NaN false

14. 自动拆箱

// 為了兼容以前版本,1.5不會自動拆箱

System.out.println(new Integer(0) == new Integer(0));// false

// 1.4编译非法,1.5会自动拆箱

System.out.println(new Integer(0) == 0);// true

15. 为什么-0x00000000==0x00000000、-0x80000000== 0x80000000

为了取一个整数类型的负值,要对其每一位取反(如果是对某个十六进制形式整数求负,如:-0x00000000则直接对这个十六进制数进行各位取反操作——但不包括前面的负号;如果是对某个十进制求负,如-0,则需先求其绝对值的十六进制的原码后,再各位取反),然后再加1。

注:如果是对某个十进制数求负,如-1(0xffffffff),实质上按照平时求一个负数补码的方式来处理也是一样的,求某个负数的补码规则为:先求这个数绝对值的原码,然后从该二进制的右边开始向左找第一个为1的位置,最后将这个1前的各位取反(包括最高位符号位,即最高位0取反后为1),其他位不变,最终所得的二进制就为这个负数的补码,也就是最终在内存中负数所表示的形式。不过在找这个第一个为1时可能找不到或在最高位,比如-0,其绝对值为0(0x00000000);也有可能最高位为1,比如-2147483648,其绝对值为2147483648(0x80000000),如果遇到绝对值的原码为0x00000000或0x80000000的情况下则不变,即为绝对值的原码本身。

 

-0x00000000的运算过程:对0x00000000先取反得到0xffffffff,再加1,-0x00000000的最后结果就为 0xffffffff+1,其最后的结果还是0x00000000,所以-0x00000000 == 0x00000000。前面是对0x00000000求负的过程,如果是对0求负呢?先求0的十六进制形式0x00000000,再按前面的过程来即可。或者根据前面规则对0x00000000求负不变,即最后结果还是0x00000000。

 

-0x80000000的运算过程:对0x80000000先取反得到0x7fffffff,再加1,-0x80000000的最后结果就为 0x7fffffff+1,其最后的结果还是0x80000000,即-0x80000000 == 0x80000000。前面是对0x80000000求负的过程,如果是对2147483648求负呢?先求2147483648的十六进制形式0x80000000,再按前面的过程来即可。或者根据前面规则对0x80000000求负不变,即最后结果还是0x80000000。

 

-0x00000001的运算过程,实质上就是求-1的补码过程,即对其绝对值的十六进制0x00000001求补码,即为0xffffffff,即-1的补码为0xffffffff。

 

System.out.println(Integer.MIN_VALUE == -Integer.MIN_VALUE);// true      

/*

 *  0x80000000取反得0x7fffffff,再加1得0x80000000,因为负数是

 *  以补码形式存储于内存中的,所以推导出结果原码为:0x80000000,

 *  即为-0,又因为-0是等于0的,所以不需要-0这个编码位,那就多了

 *  一个0x80000000编码位了,所以最后就规定0x80000000为最小负数

 */

System.out.println(-0x80000000);// -2147483648

/*

 *  0x7fffffff取反得0x80000000,再加1得0x80000001,因为负数是

 *  以补码形式存储于内存中的,所以推导出结果原码为:0xffffffff,

*  第一位为符号位,所以最后的结果就为 -0x7fffffff = -2147483647

 */

System.out.println(-0x7fffffff);// -2147483647

 

 

另外,还发现有趣现象:最大整数加1后会等于最小整数:

// MAX_VALUE = 0x7fffffff; MIN_VALUE = 0x80000000;

System.out.println((Integer.MAX_VALUE + 1) == Integer.MIN_VALUE);// true

// MIN_VALUE = 0x8000000000000000L; MIN_VALUE = 0x8000000000000000L;

System.out.println((Long.MAX_VALUE + 1) == Long.MIN_VALUE);// true

 

当然,-Byte. MIN_VALUE==Byte.MIN_VALUE、-Short.MIN_VALUE== Short.MIN_VALUE、-Long.MIN_VALUE== Long.MIN_VALUE,也是成立的。

16. Math.abs结果一定为非负数吗?

System.out.println(Math.abs(Integer.MIN_VALUE));// -2147483648

上面的程序不会输出2147483648,而是-2147483648,为什么?

 

其实我们看一下Math.abs源码就知道为什么了,源码:(a < 0) ? -a : a;,结合上面那个迷题,我们就发现-Integer.MIN_VALUE= Integer.MIN_VALUE,所以上面的答案就是最小整数自己。

 

另外我们也可以从API文档看到对Math.abs()方法的解释:如果参数等于 Integer.MIN_VALUE 的值(即能够表示的最小负 int 值),则结果与该值相同且为负。

 

所以Math.abs不能保证一定会返回非负结果。

 

当然,Long.MIN_VALUE也是这样的。

17. 不要使用基于减法的比较器

Comparator<Integer> c = new Comparator<Integer>() {

              public int compare(Integer i1, Integer i2) {

                            return i1 - i2;// 升序

              }

};

List<Integer> l = new ArrayList<Integer>();

l.add(new Integer(-2000000000));

l.add(new Integer(2000000000));

Collections.sort(l, c);

System.out.println(l);// [2000000000, -2000000000]

上面程序的比较器是升序,结果却不是这样,比较时出现了什么问题?

 

先看看下面程序片断:

int x = -2000000000;

int y = 2000000000;

/*

 * -2000000000 即 -(01110111001101011001010000000000)

 * 的补码为:                10001000110010100110110000000000

 *

 * 计算过程使用竖式表示:

 * 10001000110010100110110000000000

 * 10001000110010100110110000000000

 * --------------------------------

 * 00010001100101001101100000000000

 *

 * 计算结果溢出,结果为294967296

 */

System.out.println(x - y);// 294967296

 

所以不要使用减法的比较器,除非能确保要比较的数值之间的距离永远不会大于Intger. MAX_VALUE。

 

基于整型的比较器的实现一般使用如下的方式来比较:

public int compare(Integer i1, Integer i2) {

              return (i1 < i2 ? -1 : (i1 == i2 ? 0 : 1));

}

18.  int i=-2147483648与int i=-(2147483648)?

int i=-(2147483648);

编译通不过!为什么

 

int字面常量2147483638只能作为一元负操作符的操作数来使用。

 

类似的还有最大long:

long i=–(9223372036854775808L);

字符串

19. char类型相加

System.out.println('a' + 'A');//162

上面的结果不是 aA ,而是 162。

当且仅当+操作符的操作数中至少有一个是String类型时,才会执行字符串连接操作;否则,执行加法。如果要连接的数值没有一个是字符串类型的,那么你可以有几种选择:预置一个空字符串("" + 'a' + 'A');将第一个数值用String.valueOf()显示地转换成一个字符串(String.valueOf('a') + 'A');使用一个字符串缓冲区(sb.append('a');sb.append('A'););或者如果使用的是JDK5.0,可以用printf(System.out.printf("%c%c",'a','A'));

20. 程序中的Unicode转义字符

//\u0022是双引号的Unicode编码表示

System.out.println("a\u0022.length() + \u0022b".length());// 2

 

Unicode编码表示的字符是在编译期间就转换成了普通字符,它与普通转义字符(如:\")是不一样的,它们是在程序被解析为各种符号之后才处理转义字符序列。

21. 注释中的Unicode转义字符

如果程序中含有以下的注释:// d:\a\b\util ,程序将不能编译通过,原因是\u后面跟的不是四个十六进制数字,但编译器在编译时却要把\u开头的字符的字符看作是Unicode编码表示的字符。

 

所以要注意:注释中也是支持Unicode转义字符的。

 

另外一个问题是不能在注释行的中间含有 \u000A 表示换行的Unicode字符,因为这样在编译时读到 \u000A 时,表示行结束,那么后面的字符就会当作程序代码而不在是注释了。

22. Windows与Linux上的行结束标示符

String line = (String)System.getProperties().get("line.separator");

for(int i =0; i < line.length();i++){

              System.out.println((int)line.charAt(i));

}

 

在Windows上运行结果:

13

10

在Linux上运行的结果:

10

 

在Windows平台上,行分隔符是由回车(\r)和紧其后的换行(\n)组成,但在Unix平台上通常使用单独的换行(\n)表示。

23. 输出0-255之间的ISO8859-1符

byte bts[] = new byte[256];

for (int i = 0; i < 256; i++) {

              bts[i] = (byte) i;

}

// String str = new String(bts,"ISO8859-1");//正确的做法

String str = new String(bts);//使用操作系统默认编码方式编码(XP GBK)

for (int i = 0, n = str.length(); i < n; i++) {

              System.out.print((int) str.charAt(i) + " ");

}

上面不会输出0-255之间的数字串,正确的方式要使用new String(bts," ISO8859-1") 方式来解码。

 

ISO8859-1是唯一能够让该程序按顺序打印从0到255的整数的缺少字符集,这也是唯一在字符和字节之间一对一的映射字符集。

 

通过java获取操作系统的默认编码方式:

System.getProperty("file.encoding");//jdk1.4或之前版本

java.nio.charset.Charset.defaultCharset();//jdk1.5或之后版本

24. String的replace()与replaceAll()

System.out.println(".".replaceAll(".class", "\\$"));

上面程序将 . 替换成 \$,但运行时报异常,主要原replaceAll的第二参数有两个字符(\ $)是特殊字符,具有特殊意思(\用来转移 \ 与 $,$后面接数字表示反向引用)。另外,replaceAll的第一参数是正则表达式,所以要注意特殊字符,正确的作法有以下三种:

System.out.println(".class".replaceAll("\\.", "\\\\\\$"));

System.out.println(".class".replaceAll("\\Q.\\E", "\\\\\\$"));

System.out.println(".class".replaceAll(Pattern.quote("."), Matcher.quoteReplacement("\\$")));

 

API对\、\Q与\E的解释:

\  引用(转义)下一个字符

\Q引用所有字符,直到 \E

\E结束从 \Q 开始的引用

 

JDK5.0新增了一些解决此问题的新方法:

java.util.regex.Pattern.quote(String s):使用\Q与\E将参数引起来,这些被引用的字符串就是一般的字符,哪怕含有正则式特殊字符。

java.util.regex.Matcher.quoteReplacement(String s):将\与$转换成能应用于replaceAll第二个参数的字符串,即可作为替换内容。

 

String的replace(char oldChar, char newChar)方法却不使用正则式,但它们只支持字符,而不是字符串,使用起来受限制:

System.out.println(".".replace('.','\\'));//能将 . 替换成 \

System.out.println(".".replace('.','$')); //能将 . 替换成 $

25. 一段程序的三个Bug

Random rnd = new Random();

StringBuffer word = null;

switch (rnd.nextInt(2)) {

case 1:

              word = new StringBuffer('P');

case 2:

              word = new StringBuffer('G');

default:

              word = new StringBuffer('M');

}

word.append('a');

word.append('i');

word.append('n');

System.out.println(word);

上面的程序目的是等概率的打印 Pain、Gain、Main 三个单词,但多次运行程序却发现永远只会打印 ain,这是为什么?

 

第一个问题在于:rnd.nextInt(2)只会返回0、1 两个数字,所以上面只会走case 1: 的分支语句,case 2: 按理是永远不会走的。

 

第二个问题在于:如果case语句不以break结束时,则一直会往向运行,即直到执行到break的case语句止,所以上面的的语句每次都会执行default分支语句。

 

第三个问题在于:StringBuffer的构造函数有两种可接受参数的,一个是StringBuffer(int capacity)、另一个是StringBuffer(String str),上面用的是StringBuffer(char)构造函数,实质上运行时将字符型转换成了int型,这样将字符当作StringBuffer的初始容量了,而不是字符本身。

 

以下是修改后的程序片段:

Random rnd = new Random();

StringBuffer word = null;

switch (rnd.nextInt(3)) {

case 1:

              word = new StringBuffer("P");

              break;

case 2:

              word = new StringBuffer("G");

              break;

default:

              word = new StringBuffer("M");

              break;// 可以不要

 

}

word.append('a');

word.append('i');

word.append('n');

System.out.println(word);

异常

26. finally与中断

//该方法返回false

static boolean f() {

              try {

                            return true;

              } finally {

                            return false;

              }

}

 

不要用return、break、continue或throw来退出finally语句块,并且千万不要允许受检查的异常传播到finally语句块之外。也就是说不要在finally块内终止程序,而是执行完finally块后,要将控制权移交给try块,由try最终决定怎样结束方法的调用。

 

对于任何在finally语句块中可能抛出的受检查异常都要进行处理,而不是任其传播,下面流拷贝程序在关闭流时没有防止异常的传播,这会有问题:

static void copy(String src, String dest) throws IOException {

              InputStream in = null;

              OutputStream out = null;

              try {

                            in = new FileInputStream(src);

                            out = new FileOutputStream(dest);

                            byte[] buf = new byte[1024];

                            int n;

                            while ((n = in.read(buf)) >= 0) {

                                          out.write(buf, 0, n);

                            }

              } finally{

                            //这里应该使用try-catch将每个close包装起来

                            if(in != null){in.close();}

                            if(in != null){out.close();}

              }

}

 

catch块中的return语句是不会阻止finally块执行的,那么catch块中的continue和break能否阻止?答案是不会的,与return一样,finally语句块是在循环被跳过(continue)和中断(break)之前被执行的:

int i = 0;

System.out.println("--continue--");

while (i++ <= 1) {

              try {

                            System.out.println("i=" + i);

                            continue;

              } catch (Exception e) {

              } finally {

                            System.out.println("finally");

              }

}

System.out.println("--break--");

while (i++ <= 3) {

              try {

                            System.out.println("i=" + i);

                            break;

              } catch (Exception e) {

              } finally {

                            System.out.println("finally");

              }

}

27. catch捕获异常规则

捕获RuntimeException、Exception或Throwable的catch语句是合法,不管try块里是否抛出了这三个异常。但如果try块没有抛出或不可能抛出检测性异常,则catch不能捕获这些异常,如IOException异常:

public class Test {

              public static void main(String[] args) {

                            try{

                                          //...

                            }catch (Exception e) {

                                         

                            }catch (Throwable e) {

                                         

                            }

                           

                            /* !! 编译出错

                                          try{

                                                        //...

                                          }catch (IOException e) {

                                                       

                                          }

                             */

              }

}

28. 重写时方法异常范围

重写或实现时不能扩大异常的范围,如果是多继承,则异常取所有父类方法异常的交集或不抛出异常:

interface I1 {

              void f() throws Exception;

}

 

interface I2 {

              void f() throws IOException;

}

 

interface I3 extends I1, I2 {}

 

class Imp implements I3 {

              // 不能编译通过,多继承时只能取父类方法异常交集,这样就不会扩大异常范围

              // !! void f () throws Exception;

              // void f();// 能编译通过

              // 能编译通过,Exception与IOException的交集为IOException

              public void f() throws IOException {

              }

}

29. 静态与非静态final常量不能在catch块中初始化

静态与非静态块中如果抛出了异常,则一定要使用try-catch块来捕获。

 

public class Test {

              static final int i;

              static {

                            try {

                                          i = f();

                            } catch (RuntimeException e) {

                                          i = 1;

                            }

              }

 

              static int f() {

                            throw new RuntimeException();

              }

}

上面的程序编译不能通过。表面上是可以的,因为i第一次初始化时可能抛出异常,所以抛异常时可以在catch块中初始化,最终还是只初始化一次,这正是空final所要求的,但为什么编译器不知道这些呢?

 

要确定一个程序是否不止一次地对一个空final进行赋值是很困难的问题。语言规范在这一点上采用了保守的方式。

30. System.exit()与finally

try {

              System.out.println("Hello world");

              System.exit(0);

              // 或者使用Runtime退出系统

              // Runtime.getRuntime().exit(0);

} finally {

              System.out.println("Goodbyte world");

}

上面的程序会打印出"Goodbyte world"吗?不会。

 

System.exit将立即停止所有的程序线程,它并不会使finally语句块得到调用,但是它在停止VM之前会执行关闭挂钩操作(这此挂钩操作是注册到Runtime.addShutdownHook上的线程),这对于释放VM之外的资源很有帮助。使用挂钩程序修改上面程序:

System.out.println("Hello world");

Runtime.getRuntime().addShutdownHook(new Thread() {

              public void run() {

                            System.out.println("Goodbyte world");

              }

});

System.exit(0);

 

对象回收时,使用VM调用对象的finalize()方法有两种:

System.runFinalization():该方法让虚拟机也只是尽最大努力去完成所有未执行的finalize()终止方法,但不一定会执行。

System.runFinalizersOnExit(true):该方法一定会回收,但不安全,已被废弃。因为它可能对正在使用的对象调用终结方法,而其他线程同时正在操作这些对象,从而导致不正确的行为或死锁。

 

为了加快垃圾回收,使用System.gc(),但不一定马上执行加收动作,由虚拟机决定,实质上是调用Runtime.getRuntime().gc()。

 

System的很多方法都是调用Runtime类的相关方法来实现的。

 

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值