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.mzph.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…