【虚拟机】走进Java、自动内存管理

简介: 【虚拟机】走进Java、自动内存管理

好书分享:

在这里插入图片描述

一、走进Java

==世界上并没有完美的程序,但我们并不因此而沮丧,因为写程序本来就是一个不断追求完美的过程。==

1.Java的优点:

a. 摆脱了硬件平台的束缚,“一次编写,到处运行”(这个思想个人感觉忒牛了,想想鸿蒙的理念也是如此!)
Write Once,Run Anywhere
b.相对安全的==内存管理==和==访问机制==,避免了绝大部分内存泄漏和指针越界问题
c.实现了热点代码检测和运行时编译及优化
d.完善的应用程序接口
在这里插入图片描述

2.Java技术体系

a.组成:

b.服务领域:
在这里插入图片描述
c.具体:
在这里插入图片描述

3.Java的发展

==Java之父:詹姆斯·高斯林==

  • 最早语言为Oak,用于嵌入式系统,没有成功;
  • 1995年互联网发展,改名为Java,开始火爆,提出Write once run anywhere的原则;
  • 1996年1月 发布JDK1.0,jvm为Sun Classic VM;
  • 1996年5月 首届JavaOne大会;
  • 1997年2月 JDK1.1(内部类、反射、jdbc、javabean、rmi);
  • 1998年 JDK1.2 发布J2Se J2EE J2ME swing jit Hotspot VM;
  • 2000年5月 JDK1.3 Timer Java2d;
  • 2002年2月 JDK1.4 Struts Hibernate Spring 正则表达式 NIO 日志 Xml解析器;
  • 2004年9月 JDK1.5(tiger) 自动装箱拆箱 泛型 注解 枚举 增强for 可变参数 Spring2.X;
  • 2006年 JDK6 JavaSe JavaEE JavaME 提供脚本语言支持 支持http服务器api;
  • 2009年 Java7 Jigsaw模块化 Orical74亿收购Sun;
  • 2014年 Java8 Lambda表达式 函数式接口 方法引用 默认方法 Stream;
  • 2017年 Java9 模块化
  • 2018年 Java10(内部重构)、Java11 (免费and收费)
  • 2019年 Java12加入由RedHat领导开发的Shenandoah垃圾收集器

二、自动内存管理

2.1 Java内存区域与内存溢出异常

2.11运行时数据区域

在这里插入图片描述

程序计数器

程序计数器是一块较小的内存空间,它可以看成是当前线程所执行的字节码的==行号指示器==。每条线程都需要一个独立的程序计数器,和Java虚拟栈一样 ==“线程私有”==,生命周期与线程相同。此区域是==唯一==一个在《Java虚拟机规范》中没有规定任何OutOfMemoryError情况的区域。如果线程执行的是==java方法==,这个计数器记录的是==正在执行的虚拟字节码指令的地址==。如果正在执行的是==本地(native)方法==,那么这个计数器的值为==空(undefined)==。

Java虚拟机栈

虚拟机栈描述的是Java方法执行的动态内存模型。

  • 栈帧: 每个方法执行都会==同步创建一个栈帧==方法执行完毕,栈帧销毁。用于存储局部变量表,操作数栈,动态链接,方法出口等。
  • 局部变量表:==存放==编译期可知的各种基本数据类型,引用类型,局部变量表的大小在==编译期便已经完全确定==,在==运行时期不会发生改变==。不过这里大小指的是==变量槽==的数量(slot),这些数据类型在局部变量表中的存储空间以局部变量槽来表示。
  • 栈的大小:如果栈满了,StackOverFlowError,递归调用很常见。
public class Main {
    public static void main(String[] args) {
        test();
    }   
    public static void test() {
        System.out.println("start......");
        test();
    }
}
// 报错Exception in thread "main" java.lang.StackOverflowError

本地方法栈

本地方法栈为虚拟机执行native方法服务

Java堆

  • java虚拟机==最大==的内存区域,==存放对象实例==,也是==垃圾收集==器管理的主要区域,GC堆(Garbage Collected Heap)不是垃圾堆哦哈哈
  • Java堆可以处于物理内存不连续,逻辑上应视为连续。
  • 可固定、可拓展
不过哈,如今的及时编译技术日益进步,比如逃逸分析技术进步,栈上分配,标量替换优化手段的出现,说对象实例都分配到Java堆感觉有点过于绝对了。
十年前,主流的HotSpot虚拟机中的垃圾收集器全部基于“经典分代”设计,但今天,垃圾收集器技术已不同昨日,HotSpot里面也出现了不采用分代设计,说Java虚拟机都划分新生代、老年代也显得绝对了。

在这里插入图片描述

方法区

存储虚拟机加载的==类信息==(类的版本、字段、方法、接口),==常量==,==静态常量==,==即时编译后的代码==等数据,也可能会抛出OutOfMemoryError异常。
==方法区≠永久代==,对于HotSpot中才有永久代的概念。

运行时常量池是方法区的一部分

public class Changliang {
    public static void main(String[] args) {
        // s1与s2是相等的,为字节码常亮
        String s1 = "abc";
        String s2 = "abc";
        
        // s3创建在堆内存中
        String s3 = new String("abc");
        
        // intern方法可以将对象变为运行时常量
        // intern是一个native方法
        System.out.println(s1 == s3.intern()); // true
    }
}

直接内存:jdk1.4中增加了NIO,可以分配堆外内存(系统内存替代用户内存),提高了性能。

对象的创建:
在这里插入图片描述

2.12在堆中给对象分配内存

两种方式:==指针碰撞和空闲列表==。我们具体使用的哪一种,由==Java堆是否规整决定==,而这又由所采用的==垃圾收集器是否带有空间压缩整理能力==,如果有压缩整理,可以使用指针碰撞的分配方式。

  • 指针碰撞(Bump and Pointer):假设Java堆中内存是绝对规整的,所有用过的内存度放一边,空闲的内存放另一边,中间放着一个指针作为分界点的指示器,所分配内存就仅仅是把哪个指针向空闲空间那边挪动一段与对象大小相等的举例,这种分配方案就叫指针碰撞
  • 空闲列表(Free List):有一个列表,其中记录中哪些内存块有用,在分配的时候从列表中找到一块足够大的空间划分给对象实例,然后更新列表中的记录,这就叫做空闲列表。

2.13线程安全性问题

在两个线程同时创建对象时,可能会造成空间分配的冲突,解决方案有:==线程同步==(但执行效率过低)或==给每一个线程单独分配一个堆区域==TLAB Thread Local Allocation Buffer(本地线程分配缓冲)。

2.14对象的内存布局

在HotSpot虚拟机中,对象在内存中存储的布局可用分为3块区域:==对象头(Header)、实例数据(Instance Data)和对齐填充(Padding)==

  • 对象头:包括两部分信息,第一部分信息用于存储==对象自身的运行时数据(32位~64位 MarkWord)==,如哈希码(HashCode)、GC分代年龄、锁状态标志、线程持有的锁、偏向线程ID、偏向时间戳等。第二部分是==类型指针==,即对象指向它的类元数据的指针,虚拟机通过这个指针来确定这个对象是哪个类的实例。另外,==如果对象是一个Java数组,那在对象头中还必须有一块用于记录数组长度的数据==,因为虚拟机可以通过普通Java对象的元数据信息确定Java对象的大小,但是从数组的元数据中却无法确定数组的大小。
    在这里插入图片描述
  • 实例数据:是对象真正存储的有效信息,也是在程序代码中所定义的各种类型的字段内容。==相同宽度(如long和double)的字段被分配在一起,父类属性在子类属性之前。==
  • 对齐填充:==占位符填充作用==。(起始地址必须是8字节的整数倍)

2.15对象的访问定位

访问方式:

  • 句柄访问:Java堆中可能会划分出一块内存来作为句柄池,引用变量中存储的就是对象的句柄地址,而句柄中包含了对象实例数据和类型数据各自的具体地址信息。==一个句柄又包含了两个地址,一个对象实例数据,一个是对象类型数据(这个在方法区中,因为类字节码文件就放在方法区中)。==

在这里插入图片描述

  • 直接指针访问:引用变量中存储的就直接是对象地址了,在堆中不会分句柄池,直接指向了对象的地址,对象中包含了对象类型数据的地址。==HotSpot主要采用直接定位==(Shenandoah收集器会有一次额外的转发)

2.16OutOfMemoryError异常

在Java虚拟机规范的描述中,除了程序计数器外,虚拟机内存的其他几个运行时区域都有发生OutOfMemoryError异常的可能,如下所示:

  • Java堆溢出:Java堆用于存储对象实例,只要不断地创建对象,并且保证GCRoots到对象之间有可达路径来避免垃圾回收机制清除这些对象,那么在对象数量到达最大堆的容量限制后就会产生内存溢出异常。
  • 虚拟机栈和本地方法栈溢出:在Java虚拟机规范中,描述了两种异常:

      
      如果线程请求的栈深度大于虚拟机所允许的最大深度,将抛出StackOverflowError异常。
      
      如果虚拟机在扩展栈时无法申请到足够的内存空间,则抛出OutOfMemoryError异常。
  • 方法区和运行时常量池溢出:运行时产生大量的类去填满方法区,就会溢出。
  • 本机直接内存溢出:若向操作系统中申请分配过多内存,就会溢出。
相关文章
|
15天前
|
算法 安全 Java
Java内存管理:深入理解垃圾收集器
在Java的世界里,内存管理是一块基石,它支撑着应用程序的稳定运行。本文将带你走进Java的垃圾收集器(GC),探索它是如何默默守护着我们的内存安全。我们将从垃圾收集的基本概念出发,逐步深入到不同垃圾收集器的工作机制,并通过实例分析它们在实际应用中的表现。文章不仅旨在提升你对Java内存管理的认识,更希望你能通过这些知识优化你的代码,让程序运行更加高效。
34 3
|
6天前
|
监控 算法 Java
Java中的内存管理:理解垃圾回收机制的深度剖析
在Java编程语言中,内存管理是一个核心概念。本文将深入探讨Java的垃圾回收(GC)机制,解析其工作原理、重要性以及优化方法。通过本文,您不仅会了解到基础的GC知识,还将掌握如何在实际开发中高效利用这一机制。
|
6天前
|
存储 监控 算法
Java中的内存管理与垃圾回收机制解析
本文深入探讨了Java编程语言中的内存管理策略和垃圾回收机制。首先介绍了Java内存模型的基本概念,包括堆、栈以及方法区的划分和各自的功能。进一步详细阐述了垃圾回收的基本原理、常见算法(如标记-清除、复制、标记-整理等),以及如何通过JVM参数调优垃圾回收器的性能。此外,还讨论了Java 9引入的接口变化对垃圾回收的影响,以及如何通过Shenandoah等现代垃圾回收器提升应用性能。最后,提供了一些编写高效Java代码的实践建议,帮助开发者更好地理解和管理Java应用的内存使用。
|
13天前
|
安全 Java API
【性能与安全的双重飞跃】JDK 22外部函数与内存API:JNI的继任者,引领Java新潮流!
【9月更文挑战第7天】JDK 22外部函数与内存API的发布,标志着Java在性能与安全性方面实现了双重飞跃。作为JNI的继任者,这一新特性不仅简化了Java与本地代码的交互过程,还提升了程序的性能和安全性。我们有理由相信,在外部函数与内存API的引领下,Java将开启一个全新的编程时代,为开发者们带来更加高效、更加安全的编程体验。让我们共同期待Java在未来的辉煌成就!
43 11
|
15天前
|
安全 Java API
【本地与Java无缝对接】JDK 22外部函数和内存API:JNI终结者,性能与安全双提升!
【9月更文挑战第6天】JDK 22的外部函数和内存API无疑是Java编程语言发展史上的一个重要里程碑。它不仅解决了JNI的诸多局限和挑战,还为Java与本地代码的互操作提供了更加高效、安全和简洁的解决方案。随着FFM API的逐渐成熟和完善,我们有理由相信,Java将在更多领域展现出其强大的生命力和竞争力。让我们共同期待Java编程新纪元的到来!
38 11
|
12天前
|
监控 Java 大数据
【Java内存管理新突破】JDK 22:细粒度内存管理API,精准控制每一块内存!
【9月更文挑战第9天】虽然目前JDK 22的确切内容尚未公布,但我们可以根据Java语言的发展趋势和社区的需求,预测细粒度内存管理API可能成为未来Java内存管理领域的新突破。这套API将为开发者提供前所未有的内存控制能力,助力Java应用在更多领域发挥更大作用。我们期待JDK 22的发布,期待Java语言在内存管理领域的持续创新和发展。
|
8天前
|
存储 缓存 算法
Java中的内存管理:理解垃圾回收机制
本文将深入探讨Java中的内存管理,特别是垃圾回收机制。我们将从基本的内存分配开始,逐步解析垃圾回收的原理和过程,以及它对Java应用程序性能的影响。通过实例演示,我们会展示如何在Java中有效地管理和优化内存使用。最后,我们将讨论一些常见的内存泄漏问题及其解决方案。
|
1天前
|
监控 算法 Java
Java中的内存管理:理解Garbage Collection机制
本文将深入探讨Java编程语言中的内存管理,特别是垃圾回收(Garbage Collection, GC)机制。我们将从基础概念开始,逐步解析垃圾回收的工作原理、不同类型的垃圾回收器以及它们在实际项目中的应用。通过实际案例,读者将能更好地理解Java应用的性能调优技巧及最佳实践。
10 0
|
2月前
|
算法 Java 开发者
Java面试题:Java内存探秘与多线程并发实战,Java内存模型及分区:理解Java堆、栈、方法区等内存区域的作用,垃圾收集机制:掌握常见的垃圾收集算法及其优缺点
Java面试题:Java内存探秘与多线程并发实战,Java内存模型及分区:理解Java堆、栈、方法区等内存区域的作用,垃圾收集机制:掌握常见的垃圾收集算法及其优缺点
28 0
|
算法 Java
【Java 虚拟机原理】垃圾回收算法 ( Java 虚拟机内存分区 | 垃圾回收机制 | 引用计数器算法 | 引用计数循环引用弊端 )
【Java 虚拟机原理】垃圾回收算法 ( Java 虚拟机内存分区 | 垃圾回收机制 | 引用计数器算法 | 引用计数循环引用弊端 )
128 0