1. 程式人生 > >基本型別和引用型別區別

基本型別和引用型別區別

1.基本型別和引用型別在記憶體中的儲存

Java中資料型別分為兩大類,基本型別和物件型別。相應的,變數也有兩種型別:基本型別和引用型別。
基本型別的變數儲存原始值,即它代表的值就是數值本身;
而引用型別的變數儲存引用值,"引用值"指向記憶體空間的地址,代表了某個物件的引用,而不是物件本身,
物件本身存放在這個引用值所表示的地址的位置。

基本型別包括:byte,short,int,long,char,float,double,Boolean,returnAddress,
引用型別包括:類型別,介面型別和陣列。

相應的,變數也有兩種型別:基本型別和引用型別。

2.變數的基本型別和引用型別的區別

基本資料型別在宣告時系統就給它分配空間:

1

2

int a;

a=10;//正確,因為宣告a時就分配了空間

引用則不同,它宣告時只給變數分配了引用空間,而不分配資料空間:

1

2

3

4

5

6

7

Date date;

//執行例項化,開闢資料空間存放Date物件,然後把空間的首地址傳給today變數 

//date=new Date();

//如果註釋掉上一步操作

//The local variable date may not have been initialized

//也就是說物件的資料空間沒有分配

date.getDate();

看一下下面的初始化過程,注意"引用"也是佔用空間的,一個空Object物件的引用大小大概是4byte

1

2

3

Date a,b; //在記憶體開闢兩個引用空間

a = new Date();//開闢儲存Date物件的資料空間,並把該空間的首地址賦給a

b = a; //將a儲存空間中的地址寫到b的儲存空間中

3.引用傳遞和值傳遞

這裡要用實際引數和形式引數的概念來幫助理解,

值傳遞:

方法呼叫時,實際引數把它的值傳遞給對應的形式引數,函式接收的是原始值的一個copy,此時記憶體中存在兩個相等的基本型別,即實際引數和形式引數

後面方法中的操作都是對形參這個值的修改,不影響實際引數的值

引用傳遞:

也稱為傳地址。方法呼叫時,實際引數的引用(地址,而不是引數的值)被傳遞給方法中相對應的形式引數,函式接收的是原始值的記憶體地址;
在方法執行中,形參和實參內容相同,指向同一塊記憶體地址,方法執行中對引用的操作將會影響到實際物件

看一個例子:

1

2

3

class MyObj{

public int b=99;

}

分別傳參int和物件型別:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

public class ReferencePkValue2 {

public static void main(String[] args) { 

ReferencePkValue2 t = new ReferencePkValue2(); 

int a=99

t.test1(a);//這裡傳遞的引數a就是按值傳遞 

System.out.println(a);

MyObj obj=new MyObj(); 

t.test2(obj);//這裡傳遞的引數obj就是引用傳遞

System.out.println(obj.b);

public void test1(int a){ 

a=a++;

System.out.println(a);

public void test2(MyObj obj){ 

obj.b=100;

System.out.println(obj.b);

}

}

輸出是:
99         
99            
100          
100           

可以看到,int值沒有發生變化,但是在test2方法中對obj類做的修改影響了obj這個物件。

這裡要特殊考慮String,以及Integer、Double等幾個基本型別包裝類,它們都是immutable型別,
因為沒有提供自身修改的函式,每次操作都是新生成一個物件,所以要特殊對待,可以認為是和基本資料型別相似,傳值操作。

看下面的例子:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

public class ReferencePkValue1 {

public static void main(String[] args){

ReferencePkValue1 pk=new ReferencePkValue1();

//String類似基本型別,值傳遞,不會改變實際引數的值

String test1="Hello";

pk.change(test1);

System.out.println(test1);

//StringBuffer和StringBuilder等是引用傳遞

StringBuffer test2=new StringBuffer("Hello");

pk.change(test2);

System.out.println(test2.toString());

}

public void change(String str){

str=str+"world";

}

public void change(StringBuffer str){

str.append("world");

}

}