摘要:強引用執行結果如下,可知垃圾收集器寧愿拋出內存溢出異常,也不會回收正在使用中的強引用軟引用此時,對于這個數組對象,有兩個引用路徑,一個是來自對象的軟引用,一個來自變量的強引用,所以這個數組對象是強可及對象。
本文主要分三部分介紹 Java 中的值、指針與引用的概念。一、參數傳遞方式 1.1 值傳遞
第一部分從編程語言的三種參數傳遞方式入手,闡釋“為什么 Java 中只有值傳遞”。
第二部分排除自動裝箱和自動拆箱的干擾,理解 Integer 等封裝類作為參數傳值的情形。
第三部分通過簡單的示例,展示強引用、軟引用、弱引用和虛引用之間的區別。
形參是實參的拷貝,改變形參的值并不會影響外部實參的值。
從被調用函數的角度來說,值傳遞是單向的(實參->形參),參數的值只能傳入,不能傳出。
public class IntegerTest01 { private static void changeInt(int value) { ++value; } public static void main(String[] args) { int a = 1; changeInt(a); System.out.println("a = " + a); } }
執行結果為a = 1
1.2 指針傳遞Java 中沒有指針,為了直觀展示指針傳遞,這里使用了 C++ 的例子。
指針從本質上講是一個變量,變量的值是另一個變量的地址。因此可以說指針傳遞屬于值傳遞。
#includeusing namespace std; void fun(int *x) {// 聲明指針 *x += 5; // *x 是取得指針所指向的內存單元,即指針解引用 // x += 5; 則對實參沒有影響 } int main() { int y = 0; fun(&y);// 取地址 cout<< "y = "<< y < 執行結果為y = 5
Java 中的“指針”《Head First Java》中關于 Java 參數傳遞的說明:
Java 中所傳遞的所有東西都是值,但此值是變量所攜帶的值。引用對象的變量所攜帶的是遠程控制而不是對象本身,若你對方法傳入參數,實際上傳入的是遠程控制的拷貝。《深入理解 JVM 虛擬機》中關于 Sun HotSpot 虛擬機進行對象訪問的方式的說明:
如果使用直接指針,那么 Java 堆對象的布局中就必須考慮如何放置訪問對象類型數據的相關信息,而 reference 中存儲的直接就是對象地址。在 Java 中聲明并初始化一個對象Object object = new Object(),在堆中存儲對象實例數據,在棧中存儲對象地址,這里的變量 object 相當于 C/C++ 中的指針。
因此,可以通過 Java 對象的引用,達到指針傳遞的效果。
public class IntegerTest02 { private static void changeInt(int[] value) { ++value[0]; } public static void main(String[] args) { int[] a = {1}; changeInt(a); System.out.println("a[0] = " + a[0]); } }執行結果為a[0] = 2
1.3 引用傳遞既然 Java 中沒有引用傳遞,那么到底什么是引用傳遞呢,看下 C++ 中的例子。
#includeusing namespace std; void fun(int &x){// 聲明一個別名 x += 5; // 修改的是 x 引用的對象值 &x = y; } int main() { int y = 0; fun(y); cout<< "y = "<< y < 執行結果y = 5
C++ 中的引用就是某一變量(目標)的一個別名,對引用的操作與對變量直接操作完全一樣。
Java 中的引用
聲明一個引用,不是新定義了一個變量,它只表示該引用名是目標變量名的一個別名,它本身不是一種數據類型,因此引用本身不占存儲單元,系統也不給引用分配存儲單元。Java 中的引用是 reference 類型,類似于 C/C++ 中指針的概念,而跟 C/C++ 中引用的概念完全不同。
在 JDK 1.2 以前,Java 中的引用的定義:如果 reference 類型的數據中存儲的數值代表的是另外一塊內存的起始地址,就稱這塊內存代表著一個引用。
在JDK 1.2之后,Java對引用的概念進行了擴充,將引用分為強引用(Strong Reference)、軟引用(Soft Reference)、弱引用(Weak Reference)、虛引用(Phantom Reference)4種,這4種引用強度依次逐漸減弱。
進一步的介紹見 Java 中的 Reference 類型
二、Integer 參數傳遞問題回到開篇值傳遞的例子:
public class IntegerTest01 { private static void changeInt(int value) { ++value; } public static void main(String[] args) { int a = 1; changeInt(a); System.out.println("a = " + a); } }如果把代碼中的 int 類型換成 Integer 對象,結果會怎么樣?
public class IntegerTest02 { private static void changeInteger(Integer value) { ++value; } public static void main(String[] args) { Integer a = 1; changeInteger(a); System.out.println("a = " + a); } }首先需要排除自動裝箱和自動拆箱的干擾。
2.1 自動裝箱和自動拆箱package com.sumkor.jdk7.integer02; public class IntegerTest { public static void main(String[] args) { Integer a = 1; int b = a; } }使用命令javap -c IntegerTest.class進行反編譯:
Compiled from "IntegerTest.java" public class com.sumkor.jdk7.integer02.IntegerTest { public com.sumkor.jdk7.integer02.IntegerTest(); Code: 0: aload_0 1: invokespecial #1 // Method java/lang/Object."":()V 4: return public static void main(java.lang.String[]); Code: 0: iconst_1 1: invokestatic #2 // Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer; 4: astore_1 5: aload_1 6: invokevirtual #3 // Method java/lang/Integer.intValue:()I 9: istore_2 10: return } 由此可知:
自動裝箱實際調用的是Integer.valueOf
自動拆箱實際調用的是Integer.intValue因此,排除自動裝箱、自動拆箱,例子 IntegerTest02 等價于以下寫法:
public class IntegerTest03 { private static void changeInteger(Integer value) { value = Integer.valueOf(value.intValue() + 1); } public static void main(String[] args) { Integer a = Integer.valueOf(1); changeInteger(a); } }查看 Integer 源碼,可知valueOf()會將形參指向不同的 Integer 對象實例。
/** * Returns an {@code Integer} instance representing the specified * {@code int} value. If a new {@code Integer} instance is not * required, this method should generally be used in preference to * the constructor {@link #Integer(int)}, as this method is likely * to yield significantly better space and time performance by * caching frequently requested values. * * This method will always cache values in the range -128 to 127, * inclusive, and may cache other values outside of this range. * * @param i an {@code int} value. * @return an {@code Integer} instance representing {@code i}. * @since 1.5 */ public static Integer valueOf(int i) { assert IntegerCache.high >= 127; if (i >= IntegerCache.low && i <= IntegerCache.high) return IntegerCache.cache[i + (-IntegerCache.low)]; return new Integer(i); } /** * Cache to support the object identity semantics of autoboxing for values between * -128 and 127 (inclusive) as required by JLS. * * The cache is initialized on first usage. The size of the cache * may be controlled by the -XX:AutoBoxCacheMax=2.2 關于 IntegerCacheoption. * During VM initialization, java.lang.Integer.IntegerCache.high property * may be set and saved in the private system properties in the * sun.misc.VM class. */ private static class IntegerCache { static final int low = -128; static final int high; static final Integer cache[]; static { // high value may be configured by property int h = 127; String integerCacheHighPropValue = sun.misc.VM.getSavedProperty("java.lang.Integer.IntegerCache.high"); if (integerCacheHighPropValue != null) { int i = parseInt(integerCacheHighPropValue); i = Math.max(i, 127); // Maximum array size is Integer.MAX_VALUE h = Math.min(i, Integer.MAX_VALUE - (-low) -1); } high = h; cache = new Integer[(high - low) + 1]; int j = low; for(int k = 0; k < cache.length; k++) cache[k] = new Integer(j++); } private IntegerCache() {} } IntegerCache 在首次使用時被初始化,最小值為 -128,最大值默認為 127,也可以通過 VM 參數-XX:AutoBoxCacheMax=
設置最大值。 @Test public void test01() { Integer a = 1; Integer b = 1; System.out.println(a == b); Integer aa = 128; Integer bb = 128; System.out.println(aa == bb); }變量a和b指向的是同一個IntegerCache.cache,因此比較結果為true.
三、Java 中的 Reference 類型
變量aa和bb指向的是不同的 Integer 實例,因此比較結果為false.《深入理解 JVM 虛擬機》中對此的介紹為:
強引用就是指在程序代碼之中普遍存在的,類似Object object = new Object()這類的引用,只要強引用還存在,垃圾收集器永遠不會回收掉被引用的對象。
軟引用是用來描述一些還有用但并非必需的對象。對于軟引用關聯著的對象,在系統將要發生內存溢出異常之前,將會把這些對象列進回收范圍之中進行第二次回收。如果這次回收還沒有足夠的內存,才會拋出內存溢出異常。在JDK 1.2之后,提供了 SoftReference 類來實現軟引用。
弱引用也是用來描述非必需對象的,但是它的強度比軟引用更弱一些,被弱引用關聯的對象只能生存到下一次垃圾收集發生之前。當垃圾收集器工作時,無論當前內存是否足夠,都會回收掉只被弱引用關聯的對象。在JDK 1.2之后,提供了 WeakReference 類來實現弱引用。
虛引用也稱為幽靈引用或者幻影引用,它是最弱的一種引用關系。一個對象是否有虛引用的存在,完全不會對其生存時間構成影響,也無法通過虛引用來取得一個對象實例。為一個對象設置虛引用關聯的唯一目的就是能在這個對象被收集器回收時收到一個系統通知。在JDK 1.2之后,提供了 PhantomReference 類來實現虛引用。
Reference 類型的強度跟 JVM 垃圾回收有關,可惜書上沒有給出實例,本文對此進行補充。注意,以下例子中,使用 JDK 1.8,且均設置 JVM 參數為-verbose:gc -Xms20M -Xmx20M -Xmn10M -XX:SurvivorRatio=8
3.1 強引用
即堆大小為 20 m,其中新生代大小為 10 m,按照 1:8 比例分配,Eden 區大小為 8 m。/** * Created by Sumkor on 2018/9/10. */ public class StrongReferenceTest { public static void main(String[] args) { byte[] allocation01 = new byte[1024 * 1024 * 9]; byte[] allocation02 = new byte[1024 * 1024 * 9]; } }執行結果如下,可知垃圾收集器寧愿拋出內存溢出異常,也不會回收正在使用中的強引用:
[GC (Allocation Failure) 11197K->10032K(19456K), 0.0014301 secs] [Full GC (Ergonomics) 10032K->9851K(19456K), 0.0072375 secs] [GC (Allocation Failure) 9851K->9851K(19456K), 0.0004413 secs] [Full GC (Allocation Failure) 9851K->9833K(19456K), 0.0093839 secs] Exception in thread "main" java.lang.OutOfMemoryError: Java heap space at com.sumkor.reference.StrongReferenceTest.main(StrongReferenceTest.java:18)3.2 軟引用@Test public void test01() { byte[] allocation01 = new byte[1024 * 1024 * 8]; SoftReferencesoftReference = new SoftReference (allocation01); // 此時,對于這個byte數組對象,有兩個引用路徑,一個是來自SoftReference對象的軟引用,一個來自變量allocation01的強引用,所以這個數組對象是強可及對象。 System.out.println("softReference.get() = " + softReference.get()); allocation01 = null; // 結束變量allocation01對這個byte數組實例的強引用,此后該byte數組對象變成一個軟可及對象,可以通過softReference進行訪問 System.out.println("softReference.get() = " + softReference.get()); System.gc(); System.out.println("softReference.get() = " + softReference.get()); } 執行結果如下,可見在觸發 gc 時,內存空間充足,并不會回收軟引用:
softReference.get() = [B@5d6f64b1 softReference.get() = [B@5d6f64b1 [GC (System.gc()) 14584K->9644K(19456K), 0.0040375 secs] [Full GC (System.gc()) 9644K->9508K(19456K), 0.0115994 secs] softReference.get() = [B@5d6f64b1再來看內存不足的例子:
@Test public void test02() { byte[] allocation01 = new byte[1024 * 1024 * 8]; SoftReferencesoftReference = new SoftReference (allocation01); // 此時,對于這個byte數組對象,有兩個引用路徑,一個是來自SoftReference對象的軟引用,一個來自變量allocation01的強引用,所以這個數組對象是強可及對象。 System.out.println("softReference.get() = " + softReference.get()); allocation01 = null; // 結束變量allocation01對這個byte數組實例的強引用,此后該byte數組對象變成一個軟可及對象,可以通過softReference進行訪問 System.out.println("softReference.get() = " + softReference.get()); byte[] allocation02 = new byte[1024 * 1024 * 8]; System.out.println("softReference.get() = " + softReference.get()); } 可見在觸發 gc 時,內存空間不足,回收軟引用:
softReference.get() = [B@5d6f64b1 softReference.get() = [B@5d6f64b1 [GC (Allocation Failure) 14749K->9636K(19456K), 0.0056237 secs] [GC (Allocation Failure) 9636K->9684K(19456K), 0.0014787 secs] [Full GC (Allocation Failure) 9684K->9508K(19456K), 0.0128735 secs] [GC (Allocation Failure) 9508K->9508K(19456K), 0.0006353 secs] [Full GC (Allocation Failure) 9508K->1261K(19456K), 0.0107362 secs] softReference.get() = null3.3 弱引用package com.sumkor.reference; import java.lang.ref.WeakReference; /** * Created by Sumkor on 2018/9/10. */ public class WeakReferenceTest { public static void main(String[] args) { byte[] allocation01 = new byte[1024 * 1024 * 8]; WeakReferenceweakReference = new WeakReference (allocation01); System.out.println("weakReference.get() = " + weakReference.get());// [B@154ebadd allocation01 = null; System.out.println("weakReference.get() = " + weakReference.get());// [B@154ebadd System.gc(); System.out.println("weakReference.get() = " + weakReference.get());// null } } 執行結果如下,可見盡管內存空間充足,垃圾回收器工作時回收掉只被弱引用關聯的對象:
weakReference.get() = [B@14ae5a5 weakReference.get() = [B@14ae5a5 [GC (System.gc()) 10177K->9008K(19456K), 0.0011390 secs] [Full GC (System.gc()) 9008K->643K(19456K), 0.0069800 secs] weakReference.get() = null3.4 虛引用package com.sumkor.reference; import java.lang.ref.PhantomReference; import java.lang.ref.Reference; import java.lang.ref.ReferenceQueue; import java.lang.reflect.Field; /** * Created by Sumkor on 2018/9/10. */ public class PhantomReferenceTest { public static void main(String[] args) throws InterruptedException { ReferenceQueue執行結果如下,phantom.get()總是為 null,當 byte 數組對象被垃圾回收器回收時,垃圾收集器會把要回收的對象添加到引用隊列ReferenceQueue,即得到一個“通知”:
[GC (System.gc()) 14742K->9608K(19456K), 0.0025841 secs] [Full GC (System.gc()) 9608K->9510K(19456K), 0.0117227 secs] poll = java.lang.ref.PhantomReference@5d6f64b1 phantom.get() = null
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/77076.html
摘要:有種流行的觀點說的另外一個特殊之處在于,在方法調用傳參數時,是按值傳遞的,其他普通對象是引用傳遞。然而這種說法是大大錯誤的,至少是完全誤解了值傳遞和引用傳遞的概念。方法調用傳參只有一種傳遞就是值傳遞。 上篇文章說到Java的String是比較特殊的對象,它是不可變的。 有種流行的觀點說String的另外一個特殊之處在于,在方法調用傳參數時,String是按值傳遞的,其他普通對象是引用傳...
摘要:引用數據類型指針存放在局部變量表中,調用方法的時候,副本引用壓棧,賦值僅改變副本的引用。方法執行完畢,不再局部變量不再被使用到,等待被回收。 小方法大門道 小瓜瓜作為一個Java初學者,今天跟我說她想通過一個Java方法,將外部變量通過參數傳遞到方法中去,進行邏輯處理,方法執行完畢之后,再對修改過的變量進行判斷處理,代碼如下所示。 public class MethodParamsPa...
摘要:每個棧幀中包括局部變量表用來存儲方法中的局部變量非靜態變量函數形參。操作數棧虛擬機的解釋執行引擎被稱為基于棧的執行引擎,其中所指的棧就是指操作數棧。指向運行時常量池的引用存儲程序執行時可能用到常量的引用。 本篇文章轉自微信公眾號:Java后端技術 學過Java基礎的人都知道:值傳遞和引用傳遞是初次接觸Java時的一個難點,有時候記得了語法卻記不得怎么實際運用,有時候會的了運用卻解釋不出...
摘要:操作數棧虛擬機的解釋執行引擎被稱為基于棧的執行引擎,其中所指的棧就是指操作數棧。基本數據類型的靜態變量前面提到方法區用來存儲一些共享數據,因此基本數據類型的靜態變量名以及值存儲于方法區的運行時常 本文旨在用最通俗的語言講述最枯燥的基本知識 學過Java基礎的人都知道:值傳遞和引用傳遞是初次接觸Java時的一個難點,有時候記得了語法卻記不得怎么實際運用,有時候會的了運用卻解釋不出原理,而...
摘要:接下了,我們調用方法,來嘗試改變的值以此驗證中的傳值方式。我們將作為實參傳給方法,形參來接受這個實參,在這里就體現出了兩種傳參方式的不同。中只有值傳遞這一種方式,只不過對于引用類型來說,傳遞的參數是對象的引用罷了。 前言 這幾天在整理java基礎知識方面的內容,對于值傳遞還不是特別理解,于是查閱了一些資料和網上相關博客,自己進行了歸納總結,最后將其整理成了一篇博客。 值傳遞 值傳遞是指...
閱讀 1248·2021-11-22 13:54
閱讀 1435·2021-11-22 09:34
閱讀 2712·2021-11-22 09:34
閱讀 4024·2021-10-13 09:39
閱讀 3349·2019-08-26 11:52
閱讀 3370·2019-08-26 11:50
閱讀 1538·2019-08-26 10:56
閱讀 1920·2019-08-26 10:44