昨天看了一篇关于《Java后端程序员1年工作经验总结》的文章,其中有一段关于String和StringBuffer的描述,对于执行结果仍然把握不准,趁此机会也总结了下JVM内存模型。

1、JVM运行时数据区域

关于JVM内存模型之前也了解过一些,也是看过就忘,好记性比如烂笔头,记下来吧。参考此文章http://chenzhou123520.iteye.com/blog/1585224

图1 JVM运行时数据区域

(1)、程序计数器(Program Counter Register):
程序计数器是一块较小的内存空间,可以看做是当前线程所执行的字节码的行号指示器。
由于java虚拟机的多线程是通过线程轮流切换并分配处理器执行时间的方式来实现的,在任何一个确定的时刻,一个处理器(对于多核处理器来说是一个内核)都只会执行一条线程中的指令。因此为了线程切换后能恢复到正确的执行位置,每个线程都需要有一个独立的程序计数器,各条线程之间计数器互不影响,独立存储,是‘线程私有’的内存。
(2)、JAVA虚拟机栈(Java Virtual Machine Stack):
     与程序计数器一样,java虚拟机栈也是线程私有的,虚拟机栈描述的是Java方法执行的内存模型:在执行的同时会创建一个栈帧用于存储局部变量表、操作数栈、动态链接、方法出口等信息。每一个方法从调用直至执行完成的过程,就对应着一个栈帧在虚拟机栈中入栈到出栈的过程。
局部变量表存放了编译期可知的各种基本数据类型()、对象引用和returnAddress类型(指向了一条字节码指令的地址)
局部变量表所需的内存空间在编译期间完成分配,当进入一个方法时,这个方法需要在帧中分配多大的局部变量空间是完全确定的,在方法运行期间不会改变局部变量表的大小。

JVM Stack 异常情况:

StackOverflowError:当线程请求分配的栈容量超过JVM允许的最大容量时抛出

OutOfMemoryError:如果JVM Stack可以动态扩展,但是在尝试扩展时无法申请到足够的内存去完成扩展,或者在建立新的线程时没有足够的内存去创建对应的虚拟机栈时抛出
(3)、本地方法栈(Native Method Stack):
      本地方法栈与虚拟机栈所发挥的作用是非常相似,区别不过是虚拟机栈为虚拟机执行java方法(也就是字节码)服务,而本地方法栈则为虚拟机使用到的Native方法(使用Java语言以外的其它语言编写的方法)服务。
本地方法栈也可以抛出StackOverflowError和OutOfMemoryError异常
(4)、JAVA堆(Java Heap):
虚拟机管理的内存中最大的一块,同时也是被所有线程所共享的,它在虚拟机启动时创建,此内存区域存在的唯一目的就是存放对象实例,几乎所有的对象实例以及数组都要在这里分配内存。这里面的对象被自动管理,也就是俗称的GC(Garbage Collector)所管理。用就是了,有GC扛着呢,不用操心销毁回收的事儿。
Java堆的容量可以是固定大小,也可以随着需求动态扩展(-Xms和-Xmx),并在不需要过多空间时自动收缩。
Java堆所使用的内存不需要保证是物理连续的,只要逻辑上是连续的即可。
JVM实现应当提供给程序员调节Java 堆初始容量的手段,对于可动态扩展和收缩的堆来说,则应当提供调节其最大和最小容量的手段。
如果堆中没有内存完成实例分配并且堆也无法扩展,就会抛OutOfMemoryError。
(5)、方法区(Method Area):
跟堆一样是被各个线程共享的内存区域,用于存储以被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。虽然这个区域被虚拟机规范把方法区描述为堆的一个逻辑部分,但是它的别名叫非堆,用来与堆做一下区别。
(6)、运行时常量池(Runtime Constant Pool):
     运行时常量池是方法区一部分。Class文件中除了有类的版本、字段、方法、接口等描述信息外,还有一项信息是常量池,用于存放编译期生成的各种字面量和符号引用,这部分内容将在类加载后进入方法区的运行时常量池中存放。
2、String对象内存分配分析
先看以下代码,运行后,结果如代码1,2,3,4,5,6所示
package com.xtli.controller;public class StringTest {public static void main(String[] args) {String s1 = "hello";String s2 = "world";System.out.println(s1+"---"+s2);//1:hello---worldchange(s1,s2);System.out.println(s1+"---"+s2);//3:hello---worldStringBuffer sb1 = new StringBuffer("hello");StringBuffer sb2 = new StringBuffer("world");System.out.println(sb1+"---"+sb2);//4:hello---worldchange(sb1,sb2);System.out.println(sb1+"---"+sb2);//6:hello---worldworld}public static void change(String s1, String s2) {s1 = s2;s2 = s1+s2;System.out.println("change(s1,s2)---"+s1+"---"+s2);//2:change(s1,s2)---world---worldworld}public static void change(StringBuffer sb1, StringBuffer sb2) {sb1 = sb2;sb2.append(sb1);System.out.println("change(sb1,sb2)---"+sb1+"---"+sb2);//5:change(sb1,sb2)---worldworld---worldworld}
}

对以上代码进行分析说明,如下

public class StringTest {public static void main(String[] args) {//在main方法的栈中创建引用s1和引用s2,此引用s1和引用s2存放在栈(main方法的栈)中;编译时,在常量池中创建两个常量"hello"和"world",s1和s2分别//指向两个常量String s1 = "hello";String s2 = "world";System.out.println(s1+"---"+s2);//1:hello---worldchange(s1,s2);//引用s1和s2作为参数传递到change方法中//change方法中的引用s1,s2和main方法中的引用s1,s2存放地址并不同,以下输出的是main方法栈中的s1和s2,并没有发生变化,故代码3有以下输出System.out.println(s1+"---"+s2);//3:hello---world//以下两行代码将会在main方法栈中创建引用sb1和sb2,并在堆内存中创建两个对象"hello"和"world",sb1和sb2分别指向两个对象StringBuffer sb1 = new StringBuffer("hello");StringBuffer sb2 = new StringBuffer("world");System.out.println(sb1+"---"+sb2);//4:hello---worldchange(sb1,sb2);//引用sb1和sb2作为参数传递到change方法中//main方法中的sb1所指向的堆内存地址未发生变化,故仍为"hello",而change(sb1,sb2)方法改变了main方法中sb2所指向的堆内存地址的内容,故代码6有以下输出System.out.println(sb1+"---"+sb2);//6:hello---worldworld}public static void change(String s1, String s2) {//在change方法的栈中创建引用s1和s2,并指向常量池中的常量s1 = s2;//将引用s1指向s2的常量池中的"world"s2 = s1+s2;//在堆内存中创建"worldworld"对象,并将s2指向此堆内存地址System.out.println("change(s1,s2)---"+s1+"---"+s2);//2:change(s1,s2)---world---worldworld}public static void change(StringBuffer sb1, StringBuffer sb2) {//在change方法的栈(和上面的change方法栈不同)中创建引用sb1和sb2,并指向main方法栈中sb1和sb2所指向的对象sb1 = sb2;//将引用sb1指向sb2所引用的对象"world"sb2.append(sb1);//引用sb2所指向的对象发生变化,变为"worldworld",注意此时外部main方法中的sb2和此方法中的sb1均指向此堆内存地址,//此地址内容发生变化后,外部main方法中的sb2指向的内容也跟着变化System.out.println("change(sb1,sb2)---"+sb1+"---"+sb2);//5:change(sb1,sb2)---worldworld---worldworld}
}

为了进一步说明change(String s1, String s2)中的结果,可以进行以下验证。

public static void change(String s1, String s2) {String s= "world";String ss= "worldworld";s1 = s2;System.out.println(s==s1);//输出trues2 = s1+s2;System.out.println(ss==s2);//输出falseSystem.out.println("change(s1,s2)---"+s1+"---"+s2);//2:change(s1,s2)---world---worldworld}
故在change(String s1, String s2)方法中s1=s2后,s1所指向的是常量池中的"world",s2=s1+s2代码执行后,会在堆内存中重新创建对象,并将s2指向此堆内存地址。
以上均为个人总结,如有不正确之处,请指出,大家共同进步。转载请注明出处。
本文链接:https://blog.csdn.net/lixingtao0520/article/details/77978333

JVM内存模型及String对象内存分配相关推荐

  1. java内存模型 创建类_JVM内存模型及String对象内存分配

    昨天看了一篇关于<Java后端程序员1年工作经验总结>的文章,其中有一段关于String和StringBuffer的描述,对于执行结果仍然把握不准,趁此机会也总结了下JVM内存模型. 1. ...

  2. String 对象内存分配策略

    这个问题可以说是一个高频的面试题目,以前把这个问题弄懂了,最近突然想到这个问题,一时间竟然没有太好的思路了.所以花些时间整理一下其中的知识点. 一.内存分配策略 我们先来看一个题目(这个问题都快看吐了 ...

  3. 快速带你分清java内存结构,java内存模型,java对象模型和jvm内存结构!

    现如今你是否有这样的感觉,无论生活还是学习,节奏都是非常的快,每天面对海量的知识信息,自己感觉都要hold不住了,每天打开微信公众号,是不是发现有几十条未读,无论是技术文章还是其他类型的文章,我们大多 ...

  4. Java内存模型和常见的内存溢出类型及解决方案

    Java内存模型与常见的内存溢出 一. Java 运行过程和内存分配 1 Java的平台无关性 2 Java内存模型 2.1Java内存模型图 3 内存结构详解 3.1 元空间(MetaSpace) ...

  5. (五)JVM成神路之对象内存布局、分配过程、从生至死历程、强弱软虚引用全面剖析

    引言 在上篇文章中曾详细谈到了JVM的内存区域,其中也曾提及了:Java程序运行过程中,绝大部分创建的对象都会被分配在堆空间内.而本篇文章则会站在对象实例的角度,阐述一个Java对象从生到死的历程.J ...

  6. java jvm内存模型_Java(JVM)内存模型– Java中的内存管理

    java jvm内存模型 Understanding JVM Memory Model, Java Memory Management are very important if you want t ...

  7. jvm内存模型和java内存模型_JVM运行时内存模型综述

    JVM内存模型 JVM分为五个区域:虚拟机栈.本地方法栈.方法区.堆.程序计数器. JVM不同区域的占用内存大小不同,一般情况下堆最大,程序计数器较小. JVM五个区中虚拟机栈.本地方法栈.程序计数器 ...

  8. jvm内存模型_JVM基础:内存模型

    本篇主要从JVM的内存模型和堆内存详解两个角度整理. 内存模型 JVM中的根据应用使用数据的特性把内存划分为若干个区,根据线程使用的方式可以粗略的分为堆区和栈区,堆区即线程共享的部分(这里的堆区一般指 ...

  9. java中数组的内存模型_Java如何在内存有限的情况下实现一个超大数组?jvm性能调优+内存模型+虚拟机原理能解决什么样问题...

    在上一篇文章中,小编为您详细介绍了关于<变频器调速问题?三星R458更换CPU>相关知识.本篇中小编将再为您讲解标题Java如何在内存有限的情况下实现一个超大数组?jvm性能调优+内存模型 ...

  10. 我出题,你来算。根据Flink TaskManager内存模型,各部分内存分配?

    点击上方蓝色字体,选择"设为星标" 回复"面试"获取更多惊喜 大数据面试提升私教训练营 Hi,我是王知无,一个大数据领域的原创作者. 放心关注我,获取更多行业的 ...

最新文章

  1. java 下载文件功能代码例子
  2. 算法-------二分法查找
  3. scanf函数详解与缓冲区
  4. Linux运行可执行文件
  5. Kafka中副本机制的设计和原理
  6. 使用spring jdbc的batchUpdate功能提高性能
  7. 台式计算机睡眠了怎么唤醒,台式电脑睡眠了怎么唤醒
  8. 使用Chrome谷歌浏览器打开Axure原型图
  9. yolo系列之yolo v3【深度解析】
  10. 十一条Python学习路线,推荐收藏
  11. linux搭建pptpd服务器,最简单的Linux系统上的pptpd服务器安装
  12. Uboot11之主Makefile分析2
  13. 网络天才网页中文版_【在线猜谜类网页游戏】网络天才网页版在线玩
  14. 计算机网络实验一、验证性实验
  15. Excel日期修改为统一月份
  16. 【C进阶】qsort函数详解
  17. 【软件工程实践】Hive研究-Blog10
  18. Multisim仿真—整流滤波电路(二)
  19. 计算机毕业设计之SpringBoot+Vue.js工会管理系统 工会管理平台
  20. NLP中的关系抽取方法归纳

热门文章

  1. Matlab趋势线公式引用,批量提取excel中图表趋势线的公式中的斜率 - 程序语言 - 小木虫 - 学术 科研 互动社区...
  2. Python_随机生成11位手机号
  3. 不登录复制 CSDN代码
  4. 教育部取消计算机语言,《永远不要说你已经尽力了》--清华生的讲座报告【转】...
  5. 新版FMEA软件:特殊特性自动分类 免费申请试用(FMEAHunter)
  6. 使用Spring Boot开发邮件系统
  7. 2017年编程语言排名
  8. spoon在linux上运行,kettle在linux上运行
  9. windows通过注册表修改IP
  10. geotools绘制地图导出图片