一. 類型轉換
在Java中,將一種類型的值賦給另一種類型是很常見的,在這個賦值過程中有可能會進行類型的轉換,轉換分為自動轉換和強制轉換。
自動類型轉換(隱式轉換)無需進行任何操作,而強制類型轉換則需要顯式轉換,即需要使用強制轉換操作符(type)。
注意: boolean類型與其他所有的7種類型都不能進行類型轉換,而其他7種基本類型彼此之間都可以進行轉換,但是可能會出現精度損失或者其他的一些變化。
二. 自動轉換
1. 自動轉換時機
那么什么時候會進行自動轉換呢?一般當一個較"小"的數據與一個較"大"的數據一起運算時,系統會自動將"小"數據轉換成"大"數據,再進行運算,這時候就發生了自動轉換。
首先我們將7種類型按下面從小到大的順序排列一下: byte < (short=char) < int < long < float < double 這里我們所說的"大"與"小",并不是指占用的字節多少,而是指表示值的范圍的大小。
在上面的7種類型之間,如果數據類型是從小轉換到大,可以自動轉換,自動轉換時會發生擴寬(widening conversion);而從大到小,必須進行強制轉換;short和char兩種類型之間也必須強制轉換。這是因為較大的類型(如int) 要保存較小的類型(如byte),內存總是足夠的,不需要強制轉換。我們直接將整型字面值(常量)賦值到byte、short、char、long的時候,其實也是自動進行了類型轉換。
比如下面的源碼:
//“小”轉“大”,自動轉換 byte b = 100; int i; i = b; System.out.println("i=" + i);
//“大”轉“小”,強制轉換 int i; float f = 20000f; i = (int) f; System.out.println("i=" + i);
//short與char之間需要強制類型轉換 short s = 100; char c = 90; s=(short)c; System.out.println("s="+s);
2. 自動轉換時的精度損失
除了以下幾種情況可能會導致精度損失以外,其他的轉換都不會出現精度損失。int--> floatlong--> floatlong--> doublefloat -->double without strictfp 除了可能的精度損失外,自動轉換時不會出現任何運行時異常。
三. 強制轉換
1. 強制轉換語法
如果要把"大"的轉成"小"的,或者在short與char之間進行轉換,就必須進行強制轉換,也被稱作縮小轉換(narrowing conversion),因為必須顯式地使數值變得更小 以適應目標類型。
嚴格地說,將byte轉為char不屬于縮小轉換narrowing conversion,因為從byte到char的過程其實是byte-->int-->char,也就是既有擴寬操作widening,也有縮小操作narrowing。
強制轉換時需要采用轉換操作符(目標基本類型),格式如下:
(target-type) value
//強制轉換案例 int x=300; byte y; y = (byte)x;j
2. 強制轉換的問題
另外強制轉換除了可能的精度損失外,還可能使模(overall magnitude)發生變化。即如果整數的值超出了byte所能表示的范圍,結果將對byte類型的范圍取余數。
例如a=257超出了byte的[-128,127]的范圍,所以會將257除以byte的最大范圍(256),然后得到余數b=1。需要注意的是,當a=200時,此時除去256取得的余數應該為-56,而不是200。
//強制轉換時模發生了變化 int a = 257; byte b; b = (byte) a; System.out.println("b=" + b);
//b=1
//a=200,余數=-56 int a = 200; byte b; b = (byte) a; System.out.println("b=" + b);
//-56 將浮點類型賦給整數類型的時候,會發生截尾(truncation),也就是會把小數的部分去掉,只留下整數部分。此時如果整數超出目標類型范圍,一樣會對目標類型的范圍取余數。
//浮點賦值給整形,會截尾 float f=100.58f; int i; i=(int)f; System.out.println("i="+i);
//100 強制轉換時可能會導致溢出或精度的丟失。
3. 表達式中的數據類型自動提升
我們表達式中可能會有+、-、*、/等各種操作符,在執行這些表達式時,則可能會發生數據類型的自動提升,有以下規則:
1)所有的byte、short、char型的值將被提升為int型;
2)如果有一個操作數是long型,計算結果是long型;
3)如果有一個操作數是float型,計算結果是float型;
4)如果有一個操作數是double型,計算結果是double型。
//表達式中的類型自動提升 byte b; b = 10;
//因為此時b已經進行了類型提升,所以如果這里不進行強制類型轉換則會報錯
//b=b*2;
//此時必須進行強制類型轉換 b=(byte)(b*3);System.out.println("b="+b);
//b=30
//類型提升為long int i=400; long l=500;
//這樣會報錯
//int x=i+l;
//這樣沒問題 long x=i+l; System.out.println("x="+x);
//x=900