java標量替換_JAVA逃逸分析、棧上分配、標量替換、同步消除

一、逃逸分析

逃逸分析是編譯語言中的一種優化分析,而不是一種優化的手段。通過對象的作用范圍的分析,為其他優化手段提供分析數據從而進行優化。

逃逸分析包括:

全局變量賦值逃逸

方法返回值逃逸

實例引用發生逃逸

線程逃逸:賦值給類變量或可以在其他線程中訪問的實例變量. public class EscapeAnalysis {

public static Object object;

public void globalVariableEscape(){//全局變量賦值逃逸

object =new Object();

}

public Object methodEscape(){ //方法返回值逃逸

return new Object();

}

public void instancePassEscape(){ //實例引用發生逃逸

this.speak(this);

}

public void speak(EscapeAnalysis escapeAnalysis){

System.out.println("Escape Hello");

}

}

使用方法逃逸的案例進行分析: public StringBuffer createString(String ... values){

StringBuffer stringBuffer = new StringBuffer();

for (String string : values) {

stringBuffer.append(string+",");

}

return stringBuffer;

}

public static void main(String[] args) {

StringBuffer sb = new EscapeAnalysis().createString("Escape","Hello");

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

}

從上面的案例我們看出stringBuffer是屬于方法返回值逃逸。我們可以通過改變返回值得類型為String限定了StringBuffer的作用域在createString方法中從而不發生逃逸。 public String createString(String ... values){

StringBuffer stringBuffer = new StringBuffer();

for (String string : values) {

stringBuffer.append(string+",");

}

return stringBuffer.toString();

}

public static void main(String[] args) {

String string = new EscapeAnalysis().createString("Escape","Hello");

System.out.println(string);

}

一、標量替換 1.標量和聚合量

標量即不可被進一步分解的量,而JAVA的基本數據類型就是標量(如:int,long等基本數據類型以及reference類型等),標量的對立就是可以被進一步分解的量,而這種量稱之為聚合量。而在JAVA中對象就是可以被進一步分解的聚合量。 2.替換過程

通過逃逸分析確定該對象不會被外部訪問,并且對象可以被進一步分解時,JVM不會創建該對象,而會將該對象成員變量分解若干個被這個方法使用的成員變量所代替。這些代替的成員變量在棧幀或寄存器上分配空間。

二 、棧上分配

我們通過JVM內存分配可以知道JAVA中的對象都是在堆上進行分配,當對象沒有被引用的時候,需要依靠GC進行回收內存,如果對象數量較多的時候,會給GC帶來較大壓力,也間接影響了應用的性能。為了減少臨時對象在堆內分配的數量,JVM通過逃逸分析確定該對象不會被外部訪問。那就通過標量替換將該對象分解在棧上分配內存,這樣該對象所占用的內存空間就可以隨棧幀出棧而銷毀,就減輕了垃圾回收的壓力。

測試逃逸分析后堆內存對比: private int count = 1000000;

public static void main(String[] args) throws InterruptedException, IOException {

EscapeAnalysis escapeAnalysis = new EscapeAnalysis();

for (int i = 0; i < escapeAnalysis.count ; i++) {

escapeAnalysis.getAge();

}

Thread.sleep(500);

for (int i = 0; i < escapeAnalysis.count ; i++) {

escapeAnalysis.getAge();

}

System.in.read();

}

public int getAge(){

Person person = new Person("小明",18,28.1);

return person.getAge();

}

class Person {

private String name;

private int age;

private double weight;

public Person(String name, int age, double weight) {

super();

this.name = name;

this.age = age;

this.weight = weight;

}

public String getName() {

return name;

}

public void setName(String name) {

this.name = name;

}

public int getAge() {

return age;

}

public void setAge(int age) {

this.age = age;

}

public double getWeight() {

return weight;

}

public void setWeight(double weight) {

this.weight = weight;

}

}

1.通過jps查看Class的Main進程的PID: 關閉逃逸分析的數據 C:\Users\li>jps

13216

16592 EscapeAnalysis

5640

17308

17916 Jps

9308 RemoteMavenServer 啟用逃逸分析的數據,在JDK1.8是默認開啟逃逸分析 C:\Users\li>jps

13216

13412 Jps

5640

17308

6460 EscapeAnalysis

7100 EscapeAnalysis

9308 RemoteMavenServer

2.通過jmap -histo [pid]查看java堆上的對象分布情況: 關閉逃逸分析的數據 C:\Users\li>jmap -histo 16592

num #instances #bytes class name

----------------------------------------------

1: 980000 31360000 test.EscapeAnalysis$Person

2: 154 785000 [I

3: 2158 288504 [C

4: 489 55832 java.lang.Class

5: 2017 48408 java.lang.String

6: 839 33560 java.util.TreeMap$Entry 啟用逃逸分析的數據 C:\Users\li>jmap -histo 7100

num #instances #bytes class name

----------------------------------------------

1: 229881 7356192 test.EscapeAnalysis$Person

2: 446 756944 [I

3: 3105 442024 [C

4: 2408 57792 java.lang.String

通過上面數據可以看出沒有開啟逃逸分析時,Person在堆的內存是31360000 ,而開啟逃逸分析時,Person在堆中的內存為7356192。 兩者之間相差24003808。證明了啟用了逃逸分析,可以減少堆內存的使用和減少GC。

三、同步消除

同步消除是java虛擬機提供的一種優化技術。通過逃逸分析,可以確定一個對象是否會被其他線程進行訪問

如果對象沒有出現線程逃逸,那該對象的讀寫就不會存在資源的競爭,不存在資源的競爭,則可以消除對該對象的同步鎖。 public String createString(String ... values){

StringBuffer stringBuffer = new StringBuffer();

for (String string : values) {

stringBuffer.append(string+" ");

}

return stringBuffer.toString();

}

public static void main(String[] args) {

long start = System.currentTimeMillis();

EscapeAnalysis escapeAnalysis = new EscapeAnalysis();

for (int i = 0; i < 1000000; i++) {

escapeAnalysis.createString("Escape", "Hello");

}

long bufferCost = System.currentTimeMillis() - start;

System.out.println("craeteString: " + bufferCost + " ms");

} -server -XX:+DoEscapeAnalysis -XX:-EliminateLocks

craeteString: 202 ms

-server -XX:+DoEscapeAnalysis -XX:+EliminateLocks

craeteString: 173 ms

我們可以通過測試結果看出,如果開啟了同步消除,在開啟同步消除的執行效率比沒有開啟同步消除的高。

Reference:

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/528950.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/528950.shtml
英文地址,請注明出處:http://en.pswp.cn/news/528950.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

python圖像人類檢測_OpenCV人類行為識別(3D卷積神經網絡)

1. 3D卷積神經網絡相比于2D 卷積神經網絡&#xff0c;3D卷積神經網絡更能很好的利用視頻中的時序信息。因此&#xff0c;其主要應用視頻、行為識別等領域居多。3D卷積神經網絡是將時間維度看成了第三維。人類行為識別的實際應用&#xff1a;安防監控。(檢測識別異常行為&#x…

Spring Cloud Feign作為HTTP客戶端調用遠程HTTP服務

如果你的項目使用了SpringCloud微服務技術,那么你就可以使用Feign來作為http客戶端來調用遠程的http服務。當然,如果你不想使用Feign作為http客戶端,也可以使用比如JDK原生的URLConnection、Apache的Http Client、Netty的異步HTTP Client或者Spring的RestTemplate。 那么,為…

java查看weblogic服務器_java判斷服務器是那種,例如區分tomcat和weblogic | 學步園

package com.geostar.query.util;import org.apache.log4j.Logger;/*** author likehua* 服務器類型探測* Date 2011/04/13* **/public class ServerUtil {public static final String GERONIMO_CLASS "/org/apache/geronimo/system/main/Daemon.class";public s…

拼接的option會多出空行_Word空格,空行,頁眉橫線等問題,我只花一分鐘就全解決了...

從網上復制下來的資料粘貼到Word文檔上&#xff0c;出現了許多空格、空行、頁眉橫線等問題&#xff0c;這時候我們該如何快速地解決這些問題&#xff0c;看看下面的操作你就知道了。一、刪除空格1、空格案例從網上復制下來的文字到Word文檔中&#xff0c;出現字與字之間有好多空…

基于java高校教師管理系統_基于SSM框架下的JAVA高校教師業務水平綜合管理系統...

每天記錄學習&#xff0c;每天會有好心情。*^_^*今天和一個朋友共同完成了一個高校教師業務水平綜合管理系統項目&#xff0c;我們在開發時選用的框架是SSM(MYECLIPSE)框架。我這個朋友知識有限&#xff0c;只會這個框架&#xff0c;哈哈&#xff0c;都是為了方便他。和往常一樣…

oracle連接工具_扯一扯Tableau軟件配置數據源系列之Oracle

作者&#xff1a;扯蛋君編輯&#xff1a;齊天大圣聲明&#xff1a;本文章僅用于Taleau軟件的應用、學習溝通&#xff0c;不代表Taleau公司&#xff1b;文中所示截圖來源Taleau官方及軟件公開內容&#xff0c;相應著作權歸Tableau所有。 今天給大家介紹Tableau工具如何連接數據庫…

在java中原始時間_Java 日期時間

Java 日期時間java.util包提供了Date類來封裝當前的日期和時間。 Date類提供兩個構造函數來實例化Date對象。第一個構造函數使用當前日期和時間來初始化對象。Date( )第二個構造函數接收一個參數&#xff0c;該參數是從1970年1月1日起的微秒數。Date(long millisec)Date對象創建…

如何和后臺接觸的_后臺產品,不只是做支持

最近在招聘后臺產品經理&#xff0c;面試過程中提到一個問題&#xff0c;這個問題之前也困擾了我很久&#xff1a;你做的后臺產品&#xff0c;價值體現在哪里&#xff1f;只是做業務支持么&#xff1f;今天就來聊聊這個話題。我是做后臺產品出身&#xff0c;最開始入行做的是云…

java手寫的html轉圖片格式_(Java實現)HTML轉JPG,TIFF等圖片格式和TIFF圖片合并功能解決方案。...

上一篇文章說到了HTML轉PDF的實現方式&#xff0c;而就在那個需求的另外一個方面&#xff0c;項目要求要實現頁面轉圖片的需求&#xff0c;主要是JPG&#xff0c;TIFF&#xff0c;PNG等格式。弄得我有點囧&#xff0c;上次一直沒搞定。也沒找到合適的工具進行轉換。前一小段時間…

云計算呼叫中心_干貨|云呼叫中心系統和傳統呼叫中心系統的區別在哪?

隨著社會的發展&#xff0c;呼叫中心由傳統的呼叫中心逐漸發展為云呼叫中心。然而關于這兩者的區別&#xff0c;您知道嗎&#xff1f;跟隨暢遠技術一同來了解一下吧......一、購買、安裝不同傳統呼叫中心軟件在配置方面有幾個特點&#xff1a;一次購買終身使用&#xff1b;安裝…

java從鍵盤為數組賦值,java給數組賦值

java 動態數組賦值,java對象數組詳解,java二維數組賦值,java給數組賦值java數組動態賦值,從零學java筆錄-第24篇 圖解一維數組在內存中,java二維數組賦值,java給數組賦值java 數組動態賦值,從零學java筆錄-第24篇 圖解一維數組在內存中,java二維數組賦值,java給數組賦值數組的基…

隔一段時間查找一次 golang_劍指 offer-04 二維數組中的查找

算法名稱&#xff1a;二維數組中的查找題目內容&#xff1a;在一個二維數組中&#xff0c;每一行都按照從左到右遞增的順序排序&#xff0c;每一列都按照從上到下遞增的順序排序。請完成一個函數&#xff0c;輸入這樣的一個二維數組和一個整數&#xff0c;判斷數組中是否含有該…

decorator php,php設計模式 Decorator(裝飾模式)

/*** 裝飾模式** 動態的給一個對象添加一些額外的職責,就擴展功能而言比生成子類方式更為靈活*/header("Content-type:text/html;charsetutf-8");abstract class MessageBoardHandler{public function __construct(){}abstract public function filter($msg);}class …

python中format函數用法簡書_增強的格式化字符串format函數

自python2.6開始&#xff0c;新增了一種格式化字符串的函數str.format()&#xff0c;可謂威力十足。那么&#xff0c;他跟之前的%型格式化字符串相比&#xff0c;有什么優越的存在呢&#xff1f;讓我們來揭開它羞答答的面紗。它通過{}和:來代替%。“映射”示例通過位置In [1]: …

在線電腦配置PHP源碼,域名授權系統PHP源碼 V2.7.0 支持盜版追蹤

最新漂亮簡潔大氣的域名授權系統PHP源碼&#xff0c;域名授權系統PHP版&#xff0c;功能強大帶有后臺&#xff0c;經過版本升級&#xff0c;全新美觀大氣的UI潔面&#xff01;支持盜版追蹤&#xff0c;與卡密系統對接購買卡密對域名進行授權&#xff0c;支持授權代碼、到期時間…

python分詞代碼_中文分詞--最大正向匹配算法python實現

最大匹配法&#xff1a;最大匹配是指以詞典為依據&#xff0c;取詞典中最長單詞為第一個次取字數量的掃描串&#xff0c;在詞典中進行掃描(為提升掃描效率&#xff0c;還可以跟據字數多少設計多個字典&#xff0c;然后根據字數分別從不同字典中進行掃描)。例如&#xff1a;詞典…

python輸出所有組合數_python – GridSearchCV是否存儲了所有參數組合的所有分數?...

GridSearchCV使用“評分”來選擇最佳估算器.訓練GridSearchCV后,我希望看到每個組合的得分. GridSearchCV是否存儲每個參數組合的所有分數&#xff1f;如果它如何獲得分數&#xff1f;謝謝.這是我在另一篇文章中使用的示例代碼.from sklearn.feature_extraction.text import Co…

php 504網關,504 gateway timeout什么意思

504 gateway time-out(504網關超時錯誤)是HTTP狀態代碼&#xff0c;這意味著一個服務器在嘗試加載網頁或填寫瀏覽器的另一個請求時未從其訪問的另一臺服務器收到及時響應。換句話說&#xff0c;504錯誤通常表明不同的計算機&#xff0c;即您正在獲取504消息的網站無法控制但依賴…

python 二維數組長度_劍指offer二維數組中的查找【Java+Python】

點擊上方"藍字"&#xff0c;關注了解更多二維數組中的查找1. 題目描述在一個二維數組中(每個一維數組的長度相同)&#xff0c;每一行都按照從左到右遞增的順序排序&#xff0c;每一列都按照從上到下遞增的順序排序。請完成一個函數&#xff0c;輸入這樣的一個二維數組…

php 靜態變量 引用,PHP的返回引用(方法名前加)和局部靜態變量(static)

先閱讀手冊從函數返回一個引用&#xff0c;必須在函數聲明和指派返回值給一個變量時都使用引用操作符 & &#xff1a;例子 17-13. 由函數返回一個引用有關引用的更多信息, 請查看引用的解釋。在來看一段很多開源代碼喜歡用的單例注冊模式 class a{} class b{} function &am…