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