《Java Puzzlers》(中文譯名《Java解惑》),一句話,好書,有趣,短小精悍的行文,主要是介紹JAVA編程一些必須注意小心的陷阱(trap)。第一章:運算式之謎,俺只記下一些有趣的例子,很多關於16進位的趣事不多說,反正盡量不要使用16進位。
1。對於精確答案的地方,要盡量避免使用float和double;對於貨幣計算,要使用int,long和BigDecimal,並且使用 BigDecimal時千萬不要使用BigDecimal(double)建構函式。,因為此建構函式將用它的參數的精確值來構建一個執行個體。
如:
public class Change {
public static void main(String args[]) {
System.out.println(2.00 - 1.10);
}
}
我們希望列印0.9,可是列印的卻是0.899999999999999,貨幣計算最好還是採用BigDecimal:
public class Change {
public static void main(String args[]) {
System.out.println(new BigDecimal("2.00").subtract(new BigDecimal(" 1.10")));
}
}
2。長整除:當你操作很大的數字時,千萬要提防溢出:
public class LongDivision {
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);
}
}
你猜測列印什嗎?1000?不,列印的是5。因為第一個MICRO_PER_DAY的計算完全是以int類型來進行,雖然計算結果是可以放進long的,但在放進之前已經溢出了。改為如此,顯示地轉為long:
final long MICROS_PER_DAY = 24L * 60 * 60 * 1000 * 1000;
3。基本類型的轉型操作:
public class Multicast {
public static void main(String[] args) {
System.out.println((int) (char) (byte) -1);
}
}
輸出可能讓你很吃驚:65535。其實問題的關鍵是JAVA不顯式區分有符號數與無符號數。在執行char的轉換時,一條規則必須記住,如果最初的 數實值型別是有符號的,那麼就執行符號擴充;如果它是char,那麼不管它轉化成什麼類型,都執行零擴充。因為byte是有符號的類型,所以在-1轉化為 char時,發生符號擴充,作為結果的char的16位都被置位了,所以結果為2的16次方再加上-1。
4。使用條件操作符最好是使用類型相同的第二和第三運算元。請猜測下下面這個例子的輸出結果:
public class DosEquis {
public static void main(String[] args) {
char x = 'X';
int i = 0;
System.out.print(true ? x : 0);
System.out.print(false ? i : x);
}
}
你可能以為應該列印XX,可並非如此,結果是X88,why?這與條件操作符的規則有關:
A。如果第2和第三運算元的類型相同,那麼它也是條件操作符結果的類型
B。如果一個運算元的類型是T(T包括byte,short,char)時,另一個運算元的類型是int的常量(注意是常量)運算式,並且可以用T類型表示,那麼條件運算式的結果為T
C。否則,將對錶達式的運算元進行二進位提升,結果即為提升後的類型。
因此第2個輸出語句把X轉為int型即88。很複雜對不?呵呵,所以還是最好使用相同類型吧!!
5。另一個需要注意的是複合操作符,如+=,*=,/+等,複合賦值運算式將自動把所執行計算的結果轉化為其左側變數的類型。所以要防止窄化轉型。