【J2SE快速进阶】——Java内存分析

简介: 要在Java中分析内存,我们先来了解一下程序的执行过程:

       程序的执行过程


       要在Java中分析内存,我们先来了解一下程序的执行过程:


58.png

        正如上图所示,大致分为3个步骤:


        1、最开始,我们的程序是存在于硬盘中的,当启动运行时,程序会被加载(load)到内存中去,这里的内存可以看做我们的内存条;


        2、此时,内存中除了存在刚加载的程序的代码,还存在操作系统本身的代码(好吧,此句可以当做废话→_→),操作系统会找到程序中的Main方法开始执行程序;


        3、第三步就是本文的重点,系统在程序执行过程中对内存的管理。在Java中,内存大致会被分为四块——heap(栈)、stack(堆)、data segment(数据段)、code segment(代码段),分别用来存放程序中的局部变量、new出来的对象或数组等、静态变量、程序代码。


        这里主要讨论heap(栈)和stack(堆)。


        Java的数据类型


        先来复习一下Java的数据类型,Java中数据类型分为两种,基本数据类型和引用数据类型,如下图:


59.png


       基本数据类型就这八种,引用数据类型包括类、数组、接口等。


     (初学者可能会把String、Integer等类型和char、int等基本数据类型混淆,这里说明一下,Integer相当于int的“包装类”,String可以看做是char[]类型的数组,此外,Byte、Float等类似。所以这些类型应当当做引用类型去对待。)



      内存分析

      正如第一张图所示,程序运行时,我们定义的局部变量一般都存放于栈内存中,这些局部变量既可以是基本数据类型的变量(基本数据类型的变量在栈中直接保存它的值),也可以是引用类型的变量(引用类型的变量在栈中保存的是它所指向的堆内存中对象的地址)。


      堆内存中存放的就是引用类型变量的地址所指向的对象。        




      实践出真知,下面具体在代码中分析一下,希望您能抽出10秒钟来仔细看一下这段代码再继续:      


public class MemoryAnalysis {
  public static void main(String[] args) {        
        Person person=new Person("小明",15);
    String newName="小红";
    int newAge=18;
    person.SetName(newName);
    person.SetAge(newAge);
    person.SayHello();
  }
}
public class Person {
  public String name;
  public int age;
  public Person(String name,int age){
    this.name =name;
    this.age=age;
  }
  public void SetName(String name){
    this.name=name;
  }
  public void SetAge(int age){
    this.age=age;
  }
  public void SayHello(){
    System.out.println("我的名字叫"+name+",我"+age+"岁了");
  }
}

      下面对Main方法中的代码逐句分析  :


Person person=new Person("小明",15);


      实例化了Person类后:


      堆内存:堆内存中分配一块内存用来存放Person实例person中的数据,因为person的name属性为String类型,所以在堆内存中会另外分配一块内存用来存放String类型的“小明”,person中的name中存放的只是一个地址(地址3),这个地址指向存放“小明”的内存块;person中age属性为int类型,所以直接在age的内存单元存放int类型的“15”。


       栈内存:因为person为引用类型,所以在栈内存中分配的内存单元person中存放的是一个指向堆内存中person实例的地址(地址1)。如果上面的代码中只是定义了person,而没有new,那么只会在栈内存中分配一个person的内存单元,内才能中的值为空。


60.png

String newName="小红";


61.png


      因为newName类型为String类型,所以newName的实际内容也会存放于堆内存中,栈内存分配的内存单元newName中只是存放指向堆内存中“小红”的地址。

int newAge=18;


62.png

     newAge的类型为int类型,所以直接将值(18)存放在栈内存分配的单元newAge中。

person.SetName(newName);

 

      到了函数这块,会有一点点复杂,因为SetName(String name)函数有一个类型为引用类型参数name,而且传入的实参为newName,这时newName中存储的值(地址2)会赋值给这个name,所以这时newName和name存储的地址相同(即同时指向“小红”)。同理,当执行了SetName(String name)函数中的this.name=name时,会把栈内存中name中存储的值(地址2)赋值给堆内存中person的name,此时person中的name里存储的也是指向“小红”的地址。


63.png


       之后,person原来的name值“小明”会在某个时刻被java的垃圾回收机制所回收。


64.png


       方法执行完毕后,栈内存中变量name所占的内存被回收。

65.png

person.SetAge(newAge);


         跟SetName(String name)一样,执行时也会在栈内存中为形参age分配内存单元,只不过SetAge(int age)函数中的形参age为int类型,所以直接在栈内存分配的单元中直接存储实参的值(18)即可。

66.png


          方法执行完毕后,栈内存中变量age所占的内存被回收。

68.png


        以上就是这几句代码的执行过程。多少懂一些java基础的您肯定已经猜出运行结果了:


        运行结果:            

69.png


相关文章
|
8天前
|
存储 Java
【编程基础知识】 分析学生成绩:用Java二维数组存储与输出
本文介绍如何使用Java二维数组存储和处理多个学生的各科成绩,包括成绩的输入、存储及格式化输出,适合初学者实践Java基础知识。
36 1
|
11天前
|
缓存 算法 Java
Java中的内存管理:理解与优化
【10月更文挑战第6天】 在Java编程中,内存管理是一个至关重要的主题。本文将深入探讨Java内存模型及其垃圾回收机制,并分享一些优化内存使用的策略和最佳实践。通过掌握这些知识,您可以提高Java应用的性能和稳定性。
35 4
|
11天前
|
存储 监控 算法
Java中的内存管理:理解Garbage Collection机制
本文将深入探讨Java编程语言中的内存管理,着重介绍垃圾回收(Garbage Collection, GC)机制。通过阐述GC的工作原理、常见算法及其在Java中的应用,帮助读者提高程序的性能和稳定性。我们将从基本原理出发,逐步深入到调优实践,为开发者提供一套系统的理解和优化Java应用中内存管理的方法。
|
11天前
|
编译器 C语言
动态内存分配与管理详解(附加笔试题分析)(上)
动态内存分配与管理详解(附加笔试题分析)
34 1
|
4天前
|
存储 监控 算法
Java中的内存管理与垃圾回收机制解析
本文深入探讨了Java编程语言中的内存管理方式,特别是垃圾回收机制。我们将了解Java的自动内存管理是如何工作的,它如何帮助开发者避免常见的内存泄漏问题。通过分析不同垃圾回收算法(如标记-清除、复制和标记-整理)以及JVM如何选择合适的垃圾回收策略,本文旨在帮助Java开发者更好地理解和优化应用程序的性能。
|
5天前
|
存储 Java
Java内存模型
【10月更文挑战第11天】Java 内存模型(JMM)是 Java 虚拟机规范中定义的多线程内存访问机制,解决内存可见性、原子性和有序性问题。它定义了主内存和工作内存的概念,以及可见性、原子性和有序性的规则,确保多线程环境下的数据一致性和操作正确性。使用 `synchronized` 和 `volatile` 等同步机制可有效避免数据竞争和不一致问题。
16 3
|
5天前
|
缓存 安全 Java
使用 Java 内存模型解决多线程中的数据竞争问题
【10月更文挑战第11天】在 Java 多线程编程中,数据竞争是一个常见问题。通过使用 `synchronized` 关键字、`volatile` 关键字、原子类、显式锁、避免共享可变数据、合理设计数据结构、遵循线程安全原则和使用线程池等方法,可以有效解决数据竞争问题,确保程序的正确性和稳定性。
13 2
|
8天前
|
Java
让星星⭐月亮告诉你,Java synchronized(*.class) synchronized 方法 synchronized(this)分析
本文通过Java代码示例,介绍了`synchronized`关键字在类和实例方法上的使用。总结了三种情况:1) 类级别的锁,多个实例对象在同一时刻只能有一个获取锁;2) 实例方法级别的锁,多个实例对象可以同时执行;3) 同一实例对象的多个线程,同一时刻只能有一个线程执行同步方法。
9 1
|
9天前
|
Java 数据挖掘 数据库连接
Java使用直接内存的好处
综上所述,Java直接内存的使用为开发者提供了一种绕过JVM堆限制、直接高效操作内存资源的途径,特别适用于高吞吐量、低延迟和大规模数据处理的场景。虽然直接内存的使用需要更精细的管理以避免内存泄漏和过度消耗系统资源,但恰当的利用能够显著提升应用的性能表现,是现代高性能Java应用不可或缺的工具之一。
10 2
|
8天前
|
小程序 Oracle Java
JVM知识体系学习一:JVM了解基础、java编译后class文件的类结构详解,class分析工具 javap 和 jclasslib 的使用
这篇文章是关于JVM基础知识的介绍,包括JVM的跨平台和跨语言特性、Class文件格式的详细解析,以及如何使用javap和jclasslib工具来分析Class文件。
21 0
JVM知识体系学习一:JVM了解基础、java编译后class文件的类结构详解,class分析工具 javap 和 jclasslib 的使用