日韩成人免费在线_国产成人一二_精品国产免费人成电影在线观..._日本一区二区三区久久久久久久久不

當前位置:首頁 > 科技  > 軟件

微博二面:所有對象都一定被分配在堆中么?

來源: 責編: 時間:2023-10-27 09:14:45 294觀看
導讀什么是逃逸分析所謂逃逸,包括方法逃逸和線程逃逸,線程逃逸的逃逸程度高于方法逃逸(線程逃逸 > 方法逃逸):當一個對象在方法里面被定義后,它如果被外部方法所引用(例如作為調用參數傳遞到其他方法中),這種稱為方法逃逸;可能被外

什么是逃逸分析

所謂逃逸,包括方法逃逸和線程逃逸,線程逃逸的逃逸程度高于方法逃逸(線程逃逸 > 方法逃逸):65S28資訊網——每日最新資訊28at.com

當一個對象在方法里面被定義后,它如果被外部方法所引用(例如作為調用參數傳遞到其他方法中),這種稱為方法逃逸;65S28資訊網——每日最新資訊28at.com

可能被外部其他線程訪問到,譬如賦值給可以在其他線程中訪問的實例變量,這種稱為線程逃逸;65S28資訊網——每日最新資訊28at.com

this 引用逃逸就是一種線程逃逸:在構造器構造還未徹底完成前(即實例初始化階段還未完成),將自身 this 引用向外拋出并被其他線程復制(訪問)了該引用,那么其他線程就可能會訪問到該還未被初始化的變量。65S28資訊網——每日最新資訊28at.com

舉個例子:65S28資訊網——每日最新資訊28at.com

public class FinalReferenceEscapeTest {     final int i;     static FinalReferenceEscapeTest obj;     public FinalReferenceEscapeTest () {          i = 1;                   // 1. 寫 final 域  obj = this;              // 2. this 引用在此 "逸出" } // 線程 A public static void writer() {   new FinalReferenceEscapeExample();  } // 線程  B public static void reader() {   if (obj != null) {      // 3    int temp = obj.i;   // 4   }  }}

假設一個線程 A 執行 writer() 方法,另一個線程 B 執行 reader() 方法。這里的操作 2 將自身 this 引用向外拋出,使得 FinalReferenceEscapeTest 對象還未完成構造前就為其他線程可見。65S28資訊網——每日最新資訊28at.com

有的同學可能會問,這個操作 2 不是在構造函數的最后一步嗎,它執行完構造函數也執行完了,對象不就已經完成構造了嗎?65S28資訊網——每日最新資訊28at.com

But 這里的操作 1 和操作 2 之間可能被重排序。如下圖所示,線程 B 不能正確地讀到 i = 1,而是未初始化的 i = 0:65S28資訊網——每日最新資訊28at.com

所以,我們可以得出這樣的結論:在構造函數返回前,被構造對象的引用不能為其他線程所見,因為此時的各個字段(域)可能還沒有被初始化。65S28資訊網——每日最新資訊28at.com

如果虛擬機能夠確定一個對象不會發生方法逃逸和線程逃逸,或者逃逸程度比較低(只發生方法逃逸,不發生線程逃逸),則(JIT 即時編譯器)可以為這個對象實例采取不同程度的優化,比如鎖消除 Lock Elimination(也稱為 “同步消除 Synchronization Elimination”)、還有 棧上分配(Stack Allocations) 和 標量替換(Scalar Replacement)等65S28資訊網——每日最新資訊28at.com

棧上分配

棧上分配(Stack Allocations)是 JIT 即時編譯器的一項優化技術:如果確定一個對象不會逃逸出線程之外(不發生逃逸或逃逸程度較低 - 方法逃逸),那讓這個對象在棧(線程私有)上分配內存將會是一個很不錯的主意,對象所占用的內存空間就可以隨棧幀出棧而銷毀。65S28資訊網——每日最新資訊28at.com

在一般應用中,完全不會逃逸的局部對象和不會逃逸出線程的對象所占的比例是很大的,如果能使用棧上分配,那大量的對象就會隨著方法的結束而自動銷毀了,垃圾收集子系統的壓力將會下降很多65S28資訊網——每日最新資訊28at.com

示例代碼:65S28資訊網——每日最新資訊28at.com

public class StackAllocationExample {    private static final int MAX = 10000000;        public static void main(String[] args) {        long start = System.currentTimeMillis();        for (int i = 0; i < MAX; i++) {            allocateOnStack();        }        long end = System.currentTimeMillis();        System.out.println("Time taken: " + (end - start) + "ms");    }        private static void allocateOnStack() {        Point p = new Point();        p.x = 1;        p.y = 2;    }        private static class Point {        int x;        int y;    }}

在這個示例代碼中,我們定義了一個私有的靜態內部類 Point,它包含兩個 int 類型的成員變量 x 和 y。在 main 方法中,我們循環調用 allocateOnStack 方法,該方法內部創建一個 Point 對象并將其成員變量賦值為 1 和 2。由于 allocateOnStack 方法沒有返回 Point 對象,換言之 Point 對象是不會被暴露給其他線程的,即不會發生線程逃逸,因此編譯器可以將該對象分配在棧上而不是堆上。65S28資訊網——每日最新資訊28at.com

本文鏈接:http://www.www897cc.com/showinfo-26-15315-0.html微博二面:所有對象都一定被分配在堆中么?

聲明:本網頁內容旨在傳播知識,若有侵權等問題請及時與本網聯系,我們將在第一時間刪除處理。郵件:2376512515@qq.com

上一篇: DFA算法,高效實現敏感詞檢測與替換!

下一篇: Python字典全解析:從基礎到高級應用

標簽:
  • 熱門焦點
Top 主站蜘蛛池模板: 青海省| 岳阳市| 雅江县| 准格尔旗| 文昌市| 台江县| 岑巩县| 商洛市| 云南省| 叙永县| 中西区| 鄄城县| 乌什县| 塔河县| 翁源县| 内乡县| 台湾省| 兖州市| 许昌市| 崇左市| 石城县| 龙江县| 宜君县| 钦州市| 青海省| 龙州县| 大方县| 巩义市| 太和县| 涿鹿县| 河池市| 乌拉特前旗| 疏附县| 南安市| 阳新县| 通榆县| 黎川县| 沈丘县| 横山县| 中方县| 石家庄市|