对象“变形记”——初识引用与GC | 带你学《Java面向对象编程》之三

简介: 作为Java世界的核心内容,类与对象凭借其引用数据类型的内在特质,实现了引用传递的能力,为整个Java世界添上了浓墨重彩的一笔。

上一篇:带你“找对象”--Java内存分析 | 带你学《Java面向对象编程》之二
【本节目标】
通过阅读本章,你将通过多组实例从内存上深度了解通过对象声明、方法调用等方法进行引用传递的原理,并深刻理解引用为何会产生内存垃圾以及GC机制的相关内容。

3.1 引用传递分析

类本身属于引用数据类型,既然是引用数据类型,那么就牵扯到内存的引用传递,所谓的引用传递的本质:同一块堆内存的空间可以被不同的栈内存所指向,也可以更换指向。
范例:定义一个引用传递的分析程序

public class JavaDemo {
    public static void main(String args[]) {
      Person per1 = new Person() ;     //声明并实例化对象
      per1.name = “张三” ;
      per1.age = 18 ;
      Person per2 = per1 ; //引用传递
      per2.age =80 ;
      per1.tell() ;            //进行方法的调用
   }
}

image.png
图一 内存分析结果一

image.png
图二 运行结果一

这个时候的引用传递是直接在主方法之中定义的,也可以通过方法实现引用传递
范例:利用方法实现引用传递处理

public class JavaDemo {
    public static void main(String args[]) {
      Person per = new Person() ;     //声明并实例化对象
      per.name = “张三” ;
      per.age = 18 ;
      change(per) ;     //等价于:Person temp = per ;
      per.tell() ;            //进行方法的调用
}
public static void change(Person  temp){
    temp.age = 80 ;
   }
}

image.png
图三 运行结果二

image.png
图四 内存分析结果二

与之前的差别最大的地方在于,此时的程序是将Person类的实例化对象(内存地址、数值)传递到了change方法之中,由于传递的是一个Person类型,那么change()方法接收的也是Person类型。
引用传递可以发生在方法上,这个时候一定要观察方法的参数类型,同时也要观察方法的执行过程。

3.2 引用与垃圾产生分析

经过一系列分析后,所有的引用传递的本质就是一场堆内存的调戏游戏。但是对于引用传递,如果处理不当,那么也会造成垃圾的产生,那么本次将针对于垃圾产生的原因进行简单分析。
范例:定义一个要分析的程序

public class JavaDemo {
    public static void main(String args[]) {
      Person per1 = new Person() ;     //声明并实例化对象
      Person per2 = new Person() ;
      per1.name = “张三” ;
      per1.age = 18 ;
      per2.name = “李四” ; 
      per2.age =19 ;
      per2 = per1 ;         //引用传递
      per2.age =80 ;
      per1.tell() ;            //进行方法的调用
   }
}

image.png
图五 运行结果三

此时已经明确发生了引用传递,并且也成功的完成了引用传递的处理操作,但是下面来观察一下其内存的分配与处理操作。
一个栈内存只能够保存有一个一堆内存的地址数据,如果发生更改,则之前的地址数据将从此栈内存中彻底消失。下面来看看它的内存分析结果。

image.png
图六 内存分析结果三

所谓的垃圾空间指的就是没有任何栈内存所指向的堆内存空间,所有的垃圾将被GC(Garbage Collector、垃圾收集器)不定期进行回收并且释放无用内存空间,但是如果垃圾过多,一定将影响到GC的处理性能,从而降低整体的程序性能。那么在实际开发之中,对于垃圾的产生应该越少越好。
想学习更多的Java的课程吗?从小白到大神,从入门到精通,更多精彩不容错过!免费为您提供更多的学习资源。
本内容视频来源于阿里云大学

下一篇:保守VS开放?搞懂封装对象属性 | 带你学《Java面向对象编程》之四
更多Java面向对象编程文章查看此处

相关文章
|
1月前
|
安全 Java 编译器
Java对象一定分配在堆上吗?
本文探讨了Java对象的内存分配问题,重点介绍了JVM的逃逸分析技术及其优化策略。逃逸分析能判断对象是否会在作用域外被访问,从而决定对象是否需要分配到堆上。文章详细讲解了栈上分配、标量替换和同步消除三种优化策略,并通过示例代码说明了这些技术的应用场景。
Java对象一定分配在堆上吗?
|
19天前
|
Java 开发者
Java 面向对象编程
总之,Java 的面向对象编程为开发者提供了一种有效的编程范式,帮助他们构建出高质量、可维护的软件系统。理解和掌握面向对象的概念和原则是成为优秀 Java 开发者的重要基础。
34 3
|
2月前
|
Java API
Java 对象释放与 finalize 方法
关于 Java 对象释放的疑惑解答,以及 finalize 方法的相关知识。
50 17
|
1月前
|
存储 安全 Java
Java编程中的对象序列化与反序列化
【10月更文挑战第22天】在Java的世界里,对象序列化和反序列化是数据持久化和网络传输的关键技术。本文将带你了解如何在Java中实现对象的序列化与反序列化,并探讨其背后的原理。通过实际代码示例,我们将一步步展示如何将复杂数据结构转换为字节流,以及如何将这些字节流还原为Java对象。文章还将讨论在使用序列化时应注意的安全性问题,以确保你的应用程序既高效又安全。
|
2月前
|
存储 Java 数据管理
Java零基础-Java对象详解
【10月更文挑战第7天】Java零基础教学篇,手把手实践教学!
30 6
|
2月前
|
Oracle Java 关系型数据库
重新定义 Java 对象相等性
本文探讨了Java中的对象相等性问题,包括自反性、对称性、传递性和一致性等原则,并通过LaptopCharger类的例子展示了引用相等与内容相等的区别。文章还介绍了如何通过重写`equals`方法和使用`Comparator`接口来实现更复杂的相等度量,以满足特定的业务需求。
28 3
|
2月前
|
存储 Java
Java编程中的对象序列化与反序列化
【10月更文挑战第9天】在Java的世界里,对象序列化是连接数据持久化与网络通信的桥梁。本文将深入探讨Java对象序列化的机制、实践方法及反序列化过程,通过代码示例揭示其背后的原理。从基础概念到高级应用,我们将一步步揭开序列化技术的神秘面纱,让读者能够掌握这一强大工具,以应对数据存储和传输的挑战。
|
2月前
|
存储 Java 数据管理
Java零基础-Java对象详解
【10月更文挑战第3天】Java零基础教学篇,手把手实践教学!
26 1
|
2月前
|
Java 数据安全/隐私保护
java类和对象
java类和对象
26 5
|
1月前
|
存储 缓存 NoSQL
一篇搞懂!Java对象序列化与反序列化的底层逻辑
本文介绍了Java中的序列化与反序列化,包括基本概念、应用场景、实现方式及注意事项。序列化是将对象转换为字节流,便于存储和传输;反序列化则是将字节流还原为对象。文中详细讲解了实现序列化的步骤,以及常见的反序列化失败原因和最佳实践。通过实例和代码示例,帮助读者更好地理解和应用这一重要技术。
35 0