如何才能学好JVM?——零基础入门篇

1. JVM是什么?

JVM是Java Virtual Machine的简称,它是一个虚拟的计算机,专门为执行Java程序而设计。
你可以想象它是一个能够运行Java字节码的平台,无论你的程序在Windows、Mac还是Linux上,它们都能通过JVM在这些系统中平稳运行。

因此,JVM提供了一个抽象层,让Java程序摆脱了具体硬件和操作系统的限制。这就是为什么你可以在任何安装了JVM的设备上运行同一个Java程序。

举一个生活中的例子:

想象一下,你写了一封信,希望全世界的人都能读懂,不论他们说什么语言。
JVM就相当于一个翻译机器,能够把你的信(也就是Java程序)翻译成任何地方的“本地语言”(也就是机器码),让你的程序无论在中国的Windows、美国的Mac还是英国的Linux系统上运行得都很好。

&nbsp

2. JVM的工作原理

要理解JVM是如何工作的,我们先得知道,JVM主要有三个重要的组成部分:类加载器(Class Loaders)、执行引擎(Execution Engine)和内存区(Memory Areas)。

类加载器(Class Loaders)

类加载器的作用就像是一位图书管理员,负责将书籍(也就是你的Java类)按照一定的顺序放到图书馆的书架上(即加载到JVM内存中)。

这个过程分为三个阶段:

1、加载(Loading)
在这个阶段,类加载器读取.class文件,将这些静态数据转换成方法区的运行时数据结构,然后在堆中生成一个代表这个类的java.lang.Class对象,作为方法区这些数据的入口。

2、链接(Linking)
链接阶段验证类的正确性,为静态字段分配存储空间,并且如果必要的话,将原始码转换成机器码。

3、初始化(Initialization)
最后,在初始化阶段,JVM负责对类的静态变量赋予正确的初始值,以及执行静态代码块。

&nbsp

执行引擎(Execution Engine)

执行引擎就像一个翻译官,它读取字节码,将其转换成机器能理解和执行的指令。

它主要包括解释器和即时编译器(JIT Compiler):

1、解释器(Interpreter)
当Java程序运行时,解释器逐条读取字节码,并且将它们一条一条地翻译成机器码。这种方式简单但效率不高,因为每次执行都需要重新翻译。

2、即时编译器(JIT Compiler)
为了提高效率,JIT编译器会将热点代码(频繁执行的代码)编译成本地代码,以便直接执行,极大地提高了程序的性能。

内存区(Memory Areas)

JVM内存区是执行Java程序过程中存储各种数据的地方。

它主要包括以下几个部分:

1、堆(Heap)
这是JVM的工作重地,所有的对象实例和数组都在这里分配内存。堆是在虚拟机启动时创建,是垃圾收集器管理的主要区域,因此也被称为GC堆(Garbage-Collected Heap)。

2、方法区(Method Area)
方法区与堆一样,是各个线程共享的内存区域。它用于存储已被虚拟机加载的类信息、常量、静态变量,以及即时编译器编译后的代码等。

3、程序计数器(Program Counter Register)
这是一小块内存空间,可以看作是当前线程所执行的字节码的行号指示器。每个线程都有一个程序计数器,是线程私有的。

4、虚拟机栈(VM Stack)
每个方法在执行的同时都会创建一个栈帧,用于存储局部变量表、操作数栈、动态链接、方法出口等信息。每个方法从调用直至执行完成的过程,就是一个栈帧在虚拟机栈中入栈到出栈的过程。

5、本地方法栈(Native Method Stack)
本地方法栈与虚拟机栈发挥的作用非常相似,差别在于虚拟机栈为虚拟机执行Java方法(即字节码)服务,而本地方法栈则为虚拟机使用到的Native方法服务。

3. 探秘JVM内存模型

了解JVM内存模型,就像是学会了如何给电脑升级内存一样,你会了解Java程序是如何使用内存的,以及如何高效利用内存。

  • 堆和栈:堆是Java内存中最大的一块,用于存储对象实例,而栈用于存储局部变量和方法调用。堆是所有线程共享的,栈是每个线程私有的。
  • 方法区:这里存储了每个类的结构信息,如运行时常量池、字段和方法数据、构造函数和普通方法的字节码内容。
  • 直接内存:直接内存并不是JVM运行时数据区的一部分,但它经常被用于NIO操作,以提高性能。

打两个比喻来解释复杂的内存概念:

  • 堆和栈:假设内存是一个仓库,堆就像仓库中的大仓库,存放着所有的货物(对象实例)。栈则像仓库里的货架,用于放置我们日常工作中需要用到的工具箱(局部变量)。
  • 方法区和运行时常量池:这里可以想象成仓库的档案室,存放着产品说明书和制造标准(类信息和常量)。

&nbsp

理解JVM内存模型的重要性在于,它能帮助我们更好地管理系统资源,避免内存溢出和泄露,同时也是性能调优的基础。
例如,合理配置新生代和老年代的大小,可以优化垃圾收集器的性能,从而提高应用程序的响应速度和吞吐量。

在实际开发中,我们会使用JVM提供的监控和分析工具(如JConsole、VisualVM等)来观察和调整内存使用情况。
通过这些工具,可以了解到当前的堆内存使用状况、监控垃圾收集过程、查看方法区中类的加载情况等,这对于发现内存泄漏、调试以及提高应用性能非常有帮助。

4. 垃圾收集是什么?

垃圾收集器(Garbage Collector, GC)的任务是自动监控和回收JVM内存中不再使用的对象。
垃圾收集就像是JVM内部的清洁工,它帮助程序回收不再使用的内存空间,防止内存泄漏,确保程序的健康运行。

  • 垃圾收集的基础:当程序创建对象后,如果这些对象不再被使用,就会成为垃圾收集器的回收目标。
  • 垃圾收集器的种类:有多种垃圾收集器,例如Serial GC、Parallel GC、CMS GC、G1 GC等,不同的收集器适用于不同的场景和需求。
  • 如何监控和调优:可以使用JDK的工具如jVisualVM、jConsole等来监控垃圾收集的情况,并根据情况进行调优。

举两个例子来说明垃圾回收的必要性和原理:

  • 垃圾收集的基础:就像我们家中会定期扔掉不再使用的物品,JVM也需要定期清理不再使用的对象,以节省空间和资源。
  • 垃圾收集器的种类:不同的房子(应用场景)可能需要不同类型的清洁工具(垃圾收集器),这部分将介绍最常见的几种垃圾收集器,和它们各自的优缺点。

虽然垃圾收集器减轻了内存管理的负担,但它们也会对应用性能产生影响。
因此,在选择垃圾收集器时,需要考虑应用程序的需求,如响应时间、吞吐量或内存大小等。
使用JVM提供的参数,可以调整垃圾收集器的行为,以达到最佳的应用性能。

理解垃圾收集的原理和不同垃圾收集器的特性,有助于在开发过程中做出更明智的决策,以及在性能调优中找到合适的平衡点。
同时,搭配适当的监控工具,定期分析垃圾收集日志,可以让我们及时发现并解决内存相关的问题。

5. 实战技巧:学习和使用JVM
  • 安装和配置JVM:这就像是为你的Java程序搭建一个舞台。我将介绍如何设置环境变量,以及如何确定JVM参数。
  • 使用JDK内置工具:这些工具就像是观察和诊断工具,可以帮助你了解JVM的内部工作情况。

1、安装和配置JVM

安装JVM通常是通过安装Java Development Kit(JDK)来完成的,因为JDK中包含了JVM。
配置JVM实际上就是在系统中设置一些环境变量,这些环境变量会告诉你的计算机如何找到Java编译器和运行时环境。

  • 设置环境变量
    你需要设置JAVA_HOME环境变量,它指向你安装JDK的目录。另外,还需要把%JAVA_HOME%\bin(Windows)或$JAVA_HOME/bin(Linux/Mac)添加到系统的PATH变量中,这样你才能在命令行中方便地运行javajavac命令。

  • 确定JVM参数
    JVM的行为可以通过传递参数来调整,例如堆大小(-Xms-Xmx),选择垃圾收集器(-XX:+UseG1GC),生成堆转储文件(-XX:+HeapDumpOnOutOfMemoryError)等。这些参数可以在启动Java程序时传递给JVM,也可以在系统环境变量中设置。

2、使用JDK内置工具

JDK提供了许多强大的监控和故障排查工具,它们可以帮助开发者深入了解JVM的运行状况。

  • jconsole
    这是一个Java监视和管理控制台,可以用来监控Java虚拟机的性能和资源消耗。

  • jvisualvm
    它是一个多合一故障排除工具,包括了对JVM的实时监控、应用程序快照、堆转储分析、内存泄漏检测等功能。

  • jstat
    用于收集并显示虚拟机运行时的性能数据,非常适合监控垃圾收集过程。

  • jmap
    生成堆内存映射,对于分析堆内存使用和查找内存泄露非常有用。

  • jstack
    这个命令可以生成虚拟机当前时刻的线程快照,通常用来分析线程的堆栈信息,对于定位线程阻塞和死锁等问题非常有帮助。

  • javap
    Java反汇编命令,可以用来查看类文件中的字节码。

掌握这些工具,能帮助我们更好地理解JVM的内部运作机制,以及在开发、测试和生产环境中对Java应用程序进行有效的监控和故障排查。

实际运用中,最好结合日志信息及时调整JVM参数,以达到最优性能。

6. JVM的实战案例

通过真实的案例,我们可以看到JVM知识在实际开发中的应用,理论结合实践,帮助我们更好地理解和掌握JVM。

  • 内存泄漏案例:我们将通过一个简单的购物车例子来解释内存泄漏,并展示如何定位和修复这些问题。
  • 性能调优经历:我们将分享一个电商网站在“双十一”大促期间的性能调优案例,包括如何监控、定位瓶颈和调优策略。

1、内存泄漏案例

内存泄漏是指已分配的内存由于某种原因未被释放或未能回收,随着程序的运行,这些不再使用的内存占用会逐渐积累,最终可能导致内存溢出错误(OutOfMemoryError)。

  • 案例描述
    一个在线购物平台,用户在添加商品到购物车时,系统会创建一个购物车对象并保持会话。

  • 但是,由于某些对象引用没有被正确清除,在用户会话结束后,购物车对象没有被垃圾收集器回收,导致内存泄漏。

  • 定位和修复
    使用JDK内置的工具,如jvisualvm,我们可以监控堆内存的使用情况。通过分析堆转储(Heap Dump),我们可以看到哪些对象占用了最多的内存,并检查这些对象的引用链。一旦找到问题点,我们可以检查相关的代码,比如session管理,确保在用户会话结束后,移除购物车对象的引用,允许垃圾收集器进行回收。

2、性能调优经历

性能调优是保证应用稳定运行、响应快速的关键环节,特别是在流量高峰期间,比如“双十一”。

  • 案例描述
    电商网站在“双十一”大促期间,面临巨大的访问压力。系统出现了响应延迟,分析后发现是由于垃圾收集器频繁执行Full GC造成的。

  • 监控与定位瓶颈
    通过JVM监控工具(如jconsole)可以实时监控JVM的各项性能指标。观察到垃圾收集的时间和频率异常上升。通过进一步分析GC日志,发现老年代(Old Generation)内存不足。

  • 调优策略
    调整了JVM的启动参数,增加了老年代的大小,并根据系统的实际运行情况调整了堆内存的初始大小(-Xms)和最大大小(-Xmx)。同时,进行了代码层面的优化,比如重用对象、减少大对象的创建等,减少了内存占用和GC压力。

通过实战经历,你会慢慢积累起一套属于自己的性能调优和问题定位的方法论。这不是一蹴而就的,而是需要在日常开发和维护中不断实践和学习。

7. JVM学习资源和社区

推荐一些个人认为还不错的资源和社区:

1、书籍推荐

  • 《深入理解Java虚拟机》
    这本书由周志明著作,是学习JVM的经典之作,书中不仅详细介绍了JVM的工作原理,还涉及了性能监控与调优,垃圾收集器与内存分配策略等实用内容。

  • 《Java性能权威指南》
    由Scott Oaks所著,这本书全面介绍了Java的性能。从代码层面到JVM,再到操作系统和硬件层面,都有深入的探讨和建议。

  • 《Java并发编程实战》
    虽然这本书重点在于并发,但良好的并发性能离不开对JVM内存模型的理解。书中对JVM内存模型有深入的讲解,是高级Java开发者必读的书籍。

2、在线资源

  • Oracle官方文档
    这是获取最权威JVM信息的途径之一。Oracle的官方文档详细描述了JVM规范和JDK工具的使用。

  • InfoQ
    InfoQ上有很多关于JVM的高质量文章、演讲以及新闻,是了解行业动态的好地方。

  • Stack Overflow
    在这个问题和答案网站上,你可以找到关于JVM的大量技术问题和解决方案,也可以提出自己的疑问。

3、社区和论坛

  • Reddit上的 r/java
    这里是一个活跃的Java社区,可以在这里找到很多关于JVM的讨论。

  • OpenJDK邮件列表
    如果你对JVM的开发和未来方向感兴趣,可以关注OpenJDK邮件列表,那里有很多核心开发者参与讨论。

  • GitHub
    关注一些开源JVM项目,比如OpenJDK、GraalVM等,可以了解到最前沿的开发动态,并且可以贡献代码或者提出问题。

通过这些社区的学习与交流,不仅能系统地学习JVM的理论知识,还能深入了解实际开发中的问题和解决方案,甚至参与到相关项目的贡献中去。
同时,与其他开发者的交流也能帮助你开阔视野,获得灵感,学习路上,遇到任何问题,提出来,社区里总会有人愿意帮忙。

8. 结语

JVM的学习是一个长期而有趣的过程,希望你能保持好奇心和实践精神,不断探索,不断学习。

求一键三连:点赞、分享、收藏

点赞对我真的非常重要!在线求赞,加个关注我会非常感激!@小郑说编程

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/257163.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

片上网络NoC(6)——路由算法

目录 一、概述 二、路由算法的类型 三、避免死锁 四、实现 4.1 源路由实现 4.2 基于节点查找表的路由实现 4.3 组合电路实现 五、总结 一、概述 路由算法(routing algorithm),即决定数据包在网络拓扑中从起点到终点路径的算法。路由算…

【医学大模型 知识增强】SMedBERT:结构化语义知识 + 医学大模型 = 显著提升大模型医学文本挖掘性能

SMedBERT:结构化语义知识 医学大模型 显著提升医学文本挖掘任务性能 名词解释结构化语义知识预训练语言模型医学文本挖掘任务 提出背景具体步骤提及-邻居混合注意力机制实体嵌入增强实体描述增强三元组句子增强 提及-邻居上下文建模域内词汇权重学习领域自监督任务…

网络渗透测试:Wireshark抓取qq图片

Wireshark Wireshark Downloadhttps://www.wireshark.org/download.html 简介 WireShark是非常流行的网络封包分析工具,可以截取各种网络数据包,并显示数据包详细信息。常用于开发测试过程中各种问题定位。本文主要内容包括: 1、Wireshar…

安装Centos系统

1.镜像安装 镜像安装:Centos7安装 2.安装过程(直接以图的形式呈现) 选择你已经下载好的镜像 回车即可,等待安装 等待安装即可

单片机学习笔记---串口通信(1)

目录 通信的基本概念 通信的方式 1.按照数据传送的方式,可分为串行通信和并行通信。 1.1串行通信 1.2并行通信 2.按照通信的数据同步方式,又可以分为异步通信和同步通信。 2.1 异步通信 2.2同步通信 3.按照数据的传输方向,又可以分为…

unity 点击事件

目录 点击按钮,显示图片功能教程 第1步添加ui button,添加ui RawImage 第2步 添加脚本: 第3步,把脚本拖拽到button,点击button,设置脚本的变量, GameObject添加 Component组件 点击按钮&am…

Leetcode 452. 用最少数量的箭引爆气球435. 无重叠区间

class Solution {public int findMinArrowShots(int[][] points) {Arrays.sort(points,(o1,o2)->Integer.compare(o1[0], o2[0]));int count1;//箭的数量for(int i1;i<points.length;i) {if(points[i][0]>points[i-1][1]) {count;//边界没重合&#xff0c;又需要一支箭…

高斯伪谱C++封装库开源!

Windows x64/86 C无依赖运行高斯伪谱法求解最优控制问题&#xff0c;你只需要ElegantGP! Author: Y. F. Zhang His Github: https://github.com/ZYunfeii 写在前面 这个库在你下载它的那一时刻起不再依赖任何其他代码&#xff0c;直接可用来构建C的最优控制问题并进行求解。…

jvm垃圾收集器之七种武器

目录 1.回收算法 1.1 标记-清除算法(Mark-Sweep) 1.2 复制算法(Copying) 1.3 标记-整理算法(Mark-Compact) 2.HotSpot虚拟机的垃圾收集器 2.1 新生代的收集器 Serial 收集器&#xff08;复制算法&#xff09; ParNew 收集器 (复制算法) Parallel Scavenge 收集器 (复制…

LeetCode.145. 二叉树的后序遍历

题目 145. 二叉树的后序遍历 分析 上篇文章我们讲了前序遍历&#xff0c;这道题目是后序遍历。 首先要知道二叉树的后序遍历是什么&#xff1f;【左 右 根】 然后利用递归的思想&#xff0c;就可以得到这道题的答案&#xff0c;任何的递归都可以采用 栈 的结构来实现&#…

【Java程序设计】【C00270】基于Springboot的moba类游戏攻略分享平台(有论文)

基于Springboot的moba类游戏攻略分享平台&#xff08;有论文&#xff09; 项目简介项目获取开发环境项目技术运行截图 项目简介 这是一个基于Springboot的游戏攻略分享平台 本系统分为系统功能模块、管理员功能模块、以及用户后台功能模块。 系统功能模块&#xff1a;在平台首…

CVE-2023-22602 漏洞复现

CVE-2023-22602 简述&#xff1a; 由于 1.11.0 及之前版本的 Shiro 只兼容 Spring 的ant-style路径匹配模式&#xff08;pattern matching&#xff09;&#xff0c;且 2.6 及之后版本的 Spring Boot 将 Spring MVC 处理请求的路径匹配模式从AntPathMatcher更改为了PathPatter…

React官网摘抄

https://react.dev/learn 1、组件名称大写 2、变量&#xff0c;用{} vue中用{{}} react中用{}3、遍历 4、state使用

OpenCV基础:用Python生成一幅随机的噪声图像

使用Python&#xff1a;生成一幅随机数值的灰度图像&#xff0c;图像大小为1616像素。借助OpenCV库。输出数值&#xff0c;并显示图像。 # -*- coding: utf-8 -*- """ Created on Wed Feb 14 21:49:09 2024author: 李立宗公众号&#xff1a;计算机视觉之光知识…

【开源图床】使用Typora+PicGo+Gitee搭建个人博客图床

准备工作&#xff1a; 首先电脑得提前完成安装如下&#xff1a; 1. nodejs环境(node ,npm):【安装指南】nodejs下载、安装与配置详细教程 2. Picgo:【安装指南】图床神器之Picgo下载、安装与配置详细教程 3. Typora:【安装指南】markdown神器之Typora下载、安装与无限使用详细教…

docker常用容器命令

首先说下容器&#xff1a; 它是指当docker运行镜像时&#xff0c;创建了一个隔离环境&#xff0c;称之为 容器。 这种方式优点&#xff1a;可以开启多个服务&#xff0c;服务之前是互相隔离的&#xff08;比如&#xff1a;在一台服务器上可以开启多个mysql&#xff0c;可以是…

【Android】使用Android Studio打包APK文件

文章目录 1. 新建项目2. 打包生成APK3. 安装APK 1. 新建项目 打包APK之前&#xff0c;首先需要新建项目&#xff0c;有基础的可以跳过。 无基础的可以参考&#xff1a;使用Android Studio运行Hello World项目 2. 打包生成APK 1.找到Build -> Generate Signed Bundle or …

【Zigbee课程设计系列文章】Zigbee开发环境搭建

【Zigbee课程设计系列文章】Zigbee开发环境搭建 前言IAR 下载安装Z-Stack协议栈安装 &#x1f38a;项目专栏&#xff1a;【Zigbee课程设计系列文章】&#xff08;附详细使用教程完整代码原理图完整课设报告&#xff09; 前言 &#x1f451;由于无线传感器网络&#xff08;也即…

ctfshow——命令执行

文章目录 web 29——通配符*绕过web30——调用其他命令执行函数web 31——参数逃逸web 32-web 36——配合文件包含伪协议web 37-web 39——文件包含web 40—— web 29——通配符*绕过 i不区分大小写&#xff0c;直接?csystem(tac fl*.php); web30——调用其他命令执行函数 调用…

【RL】Bellman Equation (贝尔曼等式)

Lecture2: Bellman Equation State value 考虑grid-world的单步过程&#xff1a; S t → A t R t 1 , S t 1 S_t \xrightarrow[]{A_t} R_{t 1}, S_{t 1} St​At​ ​Rt1​,St1​ t t t, t 1 t 1 t1&#xff1a;时间戳 S t S_t St​&#xff1a;时间 t t t时所处的sta…