盡管 Java 語言是典型的面向?qū)ο缶幊陶Z言砂豌,但其中的八種基本數(shù)據(jù)類型并不支持面向?qū)ο缶幊蹋椿绢愋偷臄?shù)據(jù)不具備“對象”的特性——不攜帶屬性光督、沒有方法可調(diào)用阳距。
Java同時(shí)又為每種基本數(shù)據(jù)類型分別設(shè)計(jì)了對應(yīng)的類,稱之為包裝類结借。
基本數(shù)據(jù)類型 | 包裝類 | 取值范圍 |
---|---|---|
byte | Byte | -128~127 |
short | Short | -32768~32767 |
int | Integer | -231 ~ 2(31-1) |
long | Long | -263 ~ 2(63-1) |
char | Character | 0~65535 |
float | Float | 1.4E-45~3.4028235E38 |
double | Double | 4.9E-324~1.7976931348623157E308 |
boolean | Boolean | true或false |
每個包裝類的對象可以封裝一個對應(yīng)的基本類型的數(shù)據(jù)筐摘,并提供了其它一些有用的方法。
為什么有基本類型還要有包裝類型?
首先蓄拣,基本類型并不具有對象的性質(zhì)扬虚,為了讓基本類型也具有對象的特征,就出現(xiàn)了包裝類型(如我們在使用集合類型Collection時(shí)就一定要使用包裝類型而非基本類型)球恤,它相當(dāng)于將基本類型“包裝起來”辜昵,使得它具有了對象的性質(zhì),并且為其添加了屬性和方法咽斧,豐富了基本類型的操作堪置。
另外,集合類(ArrayList张惹,HashMap等)里面保存的必須是Object類型的對象舀锨,無法保存基本類型。
那只有包裝類型不就可以了宛逗?
我們知道Java中坎匿,new一個對象會存儲在堆里,我們通過棧中的引用來使用這些對象雷激。由于基本數(shù)據(jù)類型很常用替蔬,如果我們用new將其存儲在堆里效率明顯很低,尤其是基本數(shù)據(jù)類型都很簡單屎暇,僅僅只是一個變量而已承桥。
關(guān)于默認(rèn)值
整型byte、short根悼、int凶异、long的默認(rèn)值都為0,浮點(diǎn)型float、double的默認(rèn)值為0.0,boolean默認(rèn)值為false,char默認(rèn)值為空挤巡。對應(yīng)的包裝類型默認(rèn)值都為null剩彬。
好多博客里都有這段話,但一旦自己動手實(shí)踐過就會發(fā)現(xiàn)這里是有個坑的:
局部變量不會有默認(rèn)值玄柏,必須要初始化才能使用襟衰。例如如下代碼:
int a;
System.out.println(a);
會在第二行報(bào)錯:
The local variable a may not have been initialized
只有當(dāng)基本類型作為作為類成員使用時(shí),Java則會確保給定其初始值:
public static void main(String[] args) {
Person person = new Person();
System.out.println(person.age);
}
static class Person{
public int age;
}
可以正常運(yùn)行粪摘,輸出結(jié)果為0瀑晒。
這一功能主要是為了防止程序運(yùn)行時(shí)錯誤。但是這些初始值對我們的程序來說絕大多數(shù)時(shí)候是不正確的徘意。所以使用時(shí)必須明確指定初始值苔悦。
自動拆箱和裝箱
Java從jdk1.5開始引入了自動裝箱和拆箱,使得基本數(shù)據(jù)類型與包裝類之間相互轉(zhuǎn)換變得非常簡單椎咧。
自動裝箱: java自動將基本類型轉(zhuǎn)化為包裝類的過程玖详,自動裝箱時(shí)編譯器會調(diào)用valueOf方法,將基本類型轉(zhuǎn)化為包裝類把介。
自動拆箱: java自動將包裝類轉(zhuǎn)化為基本類型的過程,自動拆箱時(shí)編譯器會調(diào)用intValue(),doubleValue()這類的方法將對象轉(zhuǎn)換成基本類型值蟋座。
自動裝箱拗踢、拆箱帶來的問題
由于裝箱會隱式地創(chuàng)建對象,多余的對象會增加GC的壓力,影響程序的性能。因此向臀,最好不要在循環(huán)中使用自動裝箱巢墅。
另外,如下代碼:
public static void main(String[] args) {
Integer t1 = 128;
Integer t2 = 128;
System.out.println(t1 == t2);//false
/**
* 在編譯階段自動裝箱后券膀,代碼變?yōu)? * Integer t1 = new Integer(128);
* Integer t2 = new Integer(128);
* 會創(chuàng)建兩個對象君纫,因此不相等。
*/
Integer t3 = 127;
Integer t4 = 127;
System.out.println(t3 == t4);//true
/**
* t3與t4之所以相等芹彬,是因?yàn)樵贘VM中有數(shù)字常量池蓄髓,
* 緩存的范圍是-128~127,如果Integer指向這個范圍內(nèi)的數(shù)字舒帮,
* 會在編譯的時(shí)候會直接指向常量池中的數(shù)字会喝,而不會創(chuàng)建新的對象
*/
}
除Float和Double之外的基本類型在-128~127之間都會指向數(shù)組常量池。