Jvm(一)之栈、堆、方法区

2024-05-31 20:12
文章标签 java jvm 方法 之栈

本文主要是介绍Jvm(一)之栈、堆、方法区,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言-与正文无关

        生活远不止眼前的苦劳与奔波,它还充满了无数值得我们去体验和珍惜的美好事物。在这个快节奏的世界中,我们往往容易陷入工作的漩涡,忘记了停下脚步,感受周围的世界。让我们一起提醒自己,要适时放慢脚步,欣赏生活中的每一道风景,享受与家人朋友的温馨时光,发现那些平凡日子里隐藏的幸福时刻。因为,这些点点滴滴汇聚起来的,才是构成我们丰富多彩生活的本质。希望每个人都能在繁忙的生活中找到自己的快乐之源,不仅仅为了生存而工作,更为了更好的生活而生活.

        送你张美图!希望你开心!

目录

栈、堆、方法区

栈和堆

堆和方法区

栈、堆、方法区和线程

存储内容

栈的特性

栈与请求

图解数据创建


栈、堆、方法区

栈内存中放哪些东西?
  ①基本类型的变量,例如int a=3中的a;
  ②上图对象的引用变量,例如Person person=new Person();中的person


堆内存中存放哪些东西?
  ① 存放由new创建的对象和数组。如上图的new Person()后的对象:
  在堆中存放的内存,由Java虚拟机垃圾回收器来管理。在堆中产生了一个数组或者对象后,还可以在栈中定义一个特殊的变量,这个变量持有的内容等于数组或者对象在堆内存中的首地址。在栈中的这个特殊的变量,就成了数组或者对象的引用变量,以后就可以在程序中使用栈内存中的引用变量来访问堆中的数组或者对象,引用变量相当于为数组或者对象起的一个别名,或者代号。


静态区/方法区(也叫元空间)存放哪些东西?:
  方法区(method)也叫做静态区,跟堆一样,被所有的线程共享。方法区包含所有的class和static变量,还有string的直接赋值的数据值。
  方法区中包含的都是在整个程序中永远唯一的元素,例如class,static变量。
  全局变量和静态变量的存储是放在一块的,初始化的全局变量和静态变量在一块区域,未初始化的全局变量和未初始化的静态变量放在相邻的另一块区域。

栈和堆

        通俗来讲,堆是用来存放对象的(在细说一下代码执行的new,包括new后set的内容,也就是对象所持内容都是堆存放。),而栈是用来执行程序还有对象引用的,方法区存放类,静态变量的。同一个栈中有3个部分是共享的:基本类型变量区,执行环境上下文,操作指令区。

        就速度来说,有如下关系: 寄存器 > 栈 >堆 >其他这是栈的优势。但缺点是,存在栈中的数据大小和生存期是必须确定的,缺乏灵活性。栈有一个很重要的特性,就是存在栈中的数据可以共享。假设我们同时定义:

        int a = 3;

        int b = 3;

        编译器先处理int a = 3;首先它会在栈中创建一个变量为a的引用,然后查找栈中是否有3这个值,如果没找到,就将3存放进来,然后将a指向3。接着处理int b = 3;在创建完b的引用变量后,因为在栈中已经有3这个值,便将b直接指向3。这样,就出现了a与b同时均指向3的情况。这时,如果再令a=4;那么编译器会重新搜索栈中是否有4值,如果没有,则将4存放进来,并令a指向4;如果已经有了,则直接将a指向这个地址。因此a值的改变不会影响到b的值。要注意这种数据的共享与两个对象的引用时

堆和方法区

        注意一下String s3 = "china" 中china也存在方法区;String ss1 = new String("china"); 才存在堆中。对于通过 new 产生一个字符串(假设为 ”china” )时,会先去常量池中查找是否已经有了 ”china” 对象,如果没有则在常量池中创建一个此字符串对象,然后堆中再创建一个常量池中此 ”china” 对象的拷贝对象。并把ss1指向堆中的地址。(在常量池中创建目的是为了提升字符串的访问效率

栈、堆、方法区和线程

栈是私有,堆和元空间是公有

栈也叫栈内存,是在线程创建时创建,用于管理该线程的局部变量、方法调用信息、操作数栈等。它的生命期是跟随线程的生命期,线程结束栈内存也就释放,对于栈来说不存在垃圾回收问题,只要线程一结束该栈就Over,这些栈在物理内存中是独立的,互不干扰。8种基本类型的变量+对象的引用变量+实例方法都是在函数的栈内存中分配 。每一个栈对应一个线程,但是栈的数量实在太多了,导致jvm无法容纳这么多的栈。虚拟机将抛出一个OutOfMemory 异常

每个线程在创建时都会分配一个独立的栈,

存储内容
  • 局部变量表:包含方法参数和局部变量。局部变量表的大小在编译时确定。

  • 操作数栈:用于执行字节码指令时存储操作数和中间结果。

  • 栈帧:每个方法调用都会创建一个栈帧,栈帧包含局部变量表、操作数栈、动态链接和方法返回地址。

  • 动态链接:用于支持方法调用的动态链接。

  • 方法返回地址:方法返回时需要跳转的地址。

栈的特性
  • 独立性:每个线程都有一个独立的栈,栈内数据不会与其他线程共享。

  • 内存大小:栈的大小可以通过 JVM 参数 -Xss 配置,通常在 512 KB 到 1 MB 之间。1.5以后默认1M.

  • 栈帧:每次方法调用都会创建一个新的栈帧,保存方法的局部变量、操作数栈、动态链接和返回地址。

  • 生命周期:栈的生命周期与线程相同,从线程创建到线程销毁。

栈与请求
  1. 请求到达服务器

    • 当一个新的 HTTP 请求到达服务器时,服务器会从线程池中获取一个可用线程(如果没有可用线程,可能会阻塞或拒绝请求,视线程池配置而定)。

  2. 线程处理请求

    • 获取到线程后,该线程会开始处理请求,包括解析请求头、处理业务逻辑、访问数据库、生成响应等。

    • 在这个过程中,线程会使用它的栈来保存方法调用的信息、局部变量和中间计算结果。

  3. 请求处理完成

    • 当请求处理完成后,线程会生成 HTTP 响应并返回给客户端。

    • 处理完成后,线程会返回到线程池中,等待下一个请求。

图解数据创建

package test01;public class StackHeadMethod {public static void main(String[] args) {int a = 1;String b = "testBName";Test test = new Test();test.testMethod(a, b);}
}class Test {private int testA;private String testB;private static final int testC = 2;public void testMethod(int testA, String testB){this.testA = testA;this.testB = testB;}
}

 首先,JVM将StackHeadMethod.class、Test.class装载到方法区(JVM会执行启动类装载器、扩展类装载器和类路径装载器,在此只详细讲解针对本测试代码的字节码文件的执行)。其中方法区中的2是Test.java中的静态变量,在类加载的时候就在方法区中的静态存储空间分配内存。

 然后,在StackHeadMethod.class文件中,找到main方法,开始执行main方法。将main方法在栈中开辟一个空间,称为栈帧。执行下面07、08、09、17、18、19行代码。   再然后,执行test.testMethod(a, b);这一行,在栈中新分配一个栈帧,调用test中的testMethod方法。

  最后,testMethod方法执行完之后,testMethod栈帧从栈中释放空间,然后main方法执行完之后,main栈帧也释放空间,最后堆中的对象和方法区中的静态变量、字符串和字节码指令都没被使用时,根据java虚拟机的垃圾回收机制,进行对垃圾的回收。
    以上,是执行一段代码,对内存进行开辟和释放的整个过程。

------------------------------------------与正文内容无关------------------------------------
 如果觉的文章写对各位读者老爷们有帮助的话,麻烦点赞加关注呗!作者在这拜谢了!

混口饭吃了!如果你需要Java 、Python毕设、商务合作、技术交流、就业指导、技术支持度过试用期。请在关注私信我,本人看到一定马上回复!

这是我全部文章所在目录,看看是否有你需要的,如果遇到觉得不对地方请留言,看到后我会查阅进行改正。

A乐神-CSDN博客

关注在文章左上角,作者信息处。

这篇关于Jvm(一)之栈、堆、方法区的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1018927

相关文章

Java中流式并行操作parallelStream的原理和使用方法

《Java中流式并行操作parallelStream的原理和使用方法》本文详细介绍了Java中的并行流(parallelStream)的原理、正确使用方法以及在实际业务中的应用案例,并指出在使用并行流... 目录Java中流式并行操作parallelStream0. 问题的产生1. 什么是parallelS

MySQL数据库双机热备的配置方法详解

《MySQL数据库双机热备的配置方法详解》在企业级应用中,数据库的高可用性和数据的安全性是至关重要的,MySQL作为最流行的开源关系型数据库管理系统之一,提供了多种方式来实现高可用性,其中双机热备(M... 目录1. 环境准备1.1 安装mysql1.2 配置MySQL1.2.1 主服务器配置1.2.2 从

Java中Redisson 的原理深度解析

《Java中Redisson的原理深度解析》Redisson是一个高性能的Redis客户端,它通过将Redis数据结构映射为Java对象和分布式对象,实现了在Java应用中方便地使用Redis,本文... 目录前言一、核心设计理念二、核心架构与通信层1. 基于 Netty 的异步非阻塞通信2. 编解码器三、

SpringBoot基于注解实现数据库字段回填的完整方案

《SpringBoot基于注解实现数据库字段回填的完整方案》这篇文章主要为大家详细介绍了SpringBoot如何基于注解实现数据库字段回填的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以了解... 目录数据库表pom.XMLRelationFieldRelationFieldMapping基础的一些代

一篇文章彻底搞懂macOS如何决定java环境

《一篇文章彻底搞懂macOS如何决定java环境》MacOS作为一个功能强大的操作系统,为开发者提供了丰富的开发工具和框架,下面:本文主要介绍macOS如何决定java环境的相关资料,文中通过代码... 目录方法一:使用 which命令方法二:使用 Java_home工具(Apple 官方推荐)那问题来了,

Java HashMap的底层实现原理深度解析

《JavaHashMap的底层实现原理深度解析》HashMap基于数组+链表+红黑树结构,通过哈希算法和扩容机制优化性能,负载因子与树化阈值平衡效率,是Java开发必备的高效数据结构,本文给大家介绍... 目录一、概述:HashMap的宏观结构二、核心数据结构解析1. 数组(桶数组)2. 链表节点(Node

Java AOP面向切面编程的概念和实现方式

《JavaAOP面向切面编程的概念和实现方式》AOP是面向切面编程,通过动态代理将横切关注点(如日志、事务)与核心业务逻辑分离,提升代码复用性和可维护性,本文给大家介绍JavaAOP面向切面编程的概... 目录一、AOP 是什么?二、AOP 的核心概念与实现方式核心概念实现方式三、Spring AOP 的关

详解SpringBoot+Ehcache使用示例

《详解SpringBoot+Ehcache使用示例》本文介绍了SpringBoot中配置Ehcache、自定义get/set方式,并实际使用缓存的过程,文中通过示例代码介绍的非常详细,对大家的学习或者... 目录摘要概念内存与磁盘持久化存储:配置灵活性:编码示例引入依赖:配置ehcache.XML文件:配置

Java 虚拟线程的创建与使用深度解析

《Java虚拟线程的创建与使用深度解析》虚拟线程是Java19中以预览特性形式引入,Java21起正式发布的轻量级线程,本文给大家介绍Java虚拟线程的创建与使用,感兴趣的朋友一起看看吧... 目录一、虚拟线程简介1.1 什么是虚拟线程?1.2 为什么需要虚拟线程?二、虚拟线程与平台线程对比代码对比示例:三

Python版本信息获取方法详解与实战

《Python版本信息获取方法详解与实战》在Python开发中,获取Python版本号是调试、兼容性检查和版本控制的重要基础操作,本文详细介绍了如何使用sys和platform模块获取Python的主... 目录1. python版本号获取基础2. 使用sys模块获取版本信息2.1 sys模块概述2.1.1