当前位置: 首页 > article >正文

[Java基础] JVM常量池介绍(BeanUtils.copyProperties(source, target)中的属性值引用的是同一个对象吗)

文章目录

  • 1. JVM内存模型
  • 2. 常量池中有什么类型?
  • 3. 常量池中真正存储的内容是什么
  • 4. 判断一个字符串(引用)是否在常量池中
  • 5. BeanUtils.copyProperties(source, target)中的属性值引用的是同一个对象吗?
  • 6. 获取堆内存使用情况、非堆内存使用情况

1. JVM内存模型

在这里插入图片描述

2. 常量池中有什么类型?

常量池中主要包含以下几种类型的常量:

  • 字面量

  • 文本字符串:程序中直接书写的字符串,如"hello"

  • 基本数据类型常量:整数、浮点数、字符等,如1233.14'a'

  • 被声明为final的常量值:在编译时已知的常量表达式结果。

  • 符号引用

  • 类和接口的全限定名:如java.lang.String

  • 字段的名称和描述符:包括字段的类型信息,如int age中的ageint

  • 方法的名称和描述符:包括方法的参数类型和返回类型,如public String getName()中的getName(Ljava/lang/String;)V

  • 其他类型

  • 方法句柄和动态调用点:用于支持Java 7引入的invokedynamic指令,实现动态语言特性。

常量池是JVM的重要组成部分,它存储了程序运行所需的各种常量信息,有助于优化内存使用和提升程序性能。

3. 常量池中真正存储的内容是什么

以字符串举例:

  • 字符串常量池存储的是字符串对象的引用,而不是对象本身,字符串本身还是在堆中。
  • 当使用字面量形式创建字符串(如String s = "hello";)时,该字符串会被放入常量池,实际存储的是对堆中字符串对象的引用。
  • 可以认为字符串常量池的作用就是使堆中的字符串被重复引用,而不用在堆中创建新的字符串对象

4. 判断一个字符串(引用)是否在常量池中

String str = "abc"; // 这种写法字符串在常量池中
String str1 = new String("abc"); // 这种写法字符串在堆内存中
System.out.println(str == str1); // false,可以判断两个对象是否是相同的内存地址,都在堆中或都在常量池(引用)中

// String的intern()方法会返回字符串在常量池中的引用。如果常量池中已存在该字符串,则直接返回;
// 否则,将该字符串加入常量池并返回引用。
String str2 = str1.intern();
System.out.println(str == str2); // true
System.out.println(str == str1); // false

// 返回常量池中的引用,虽然将str1加入常量池,但str2和str1不是同一个对象,
// 因为"abc"已经存在于常量池中,所以str1.intern()返回的其实是str的引用地址
System.out.println(str2 == str1); // false

// 创建新对象
String str3 = new String("abc"); // 堆内存
System.out.println(str == str3); // false,证明str3不在常量池中,而在堆中
System.out.println(str1 == str3); // false,证明str3是堆中不同于str1的另一个对象

5. BeanUtils.copyProperties(source, target)中的属性值引用的是同一个对象吗?

Person person = new Person(); // 堆内存
person.setName("xiaohua");
Person person1 = new Person();
BeanUtils.copyProperties(person, person1);
System.out.println(person == person1); // false,person和person1不是同一个对象
System.out.println("person.name == person1.name: " + (person.getName() == person1.getName())); // person.name == person1.name: true
person1.setName("xiaohuahua");
System.out.println("person: " + JSONObject.toJSONString(person)); // person: {"name":"xiaohua"}
System.out.println("person1: " + JSONObject.toJSONString(person1)); // person1: {"name":"xiaohuahua"}
System.out.println("person.name == person1.name: " + (person.getName() == person1.getName())); // person.name == person1.name: false
String str4 = "xiaohua";
System.out.println(str4 == person.getName()); // true,证明person中name的值是常量池中的引用
String str5 = "xiaohuahua";
System.out.println(str5 == person1.getName()); // true,证明person1中name的值是常量池中的引用

总结:
BeanUtils.copyProperties(source, target),source和target的属性name的值是在常量池中,
所以target虽然是一个新对象,但里面属性name其实引用相同,常量池中存的是堆中对象的地址;
当target重新给name赋值时,值是新的也是属于常量池。

6. 获取堆内存使用情况、非堆内存使用情况

MemoryMXBean memoryMXBean = ManagementFactory.getMemoryMXBean();
// getHeapMemoryUsage() 方法返回一个 MemoryUsage 对象,表示当前 JVM 堆内存的使用情况。
// MemoryUsage 包含以下属性:
// init: 初始分配的内存量(以字节为单位)。
// used: 当前已使用的内存量(以字节为单位)。
// committed: 已提交给 JVM 的内存量(以字节为单位),这部分内存可以立即使用。
// max: 最大可用内存量(以字节为单位),如果未设置最大值,则为 -1。
MemoryUsage heapUsage = memoryMXBean.getHeapMemoryUsage();
// heapUsage:init = 117440512(114688K) used = 18740896(18301K) committed = 112721920(110080K) max = 1648361472(1609728K)
System.out.println("heapUsage:" + heapUsage);

// getNonHeapMemoryUsage() 方法返回一个 MemoryUsage 对象,表示当前 JVM 非堆内存的使用情况。
MemoryUsage nonHeapUsage = memoryMXBean.getNonHeapMemoryUsage();
// nonHeapUsage:init = 2555904(2496K) used = 16119416(15741K) committed = 16842752(16448K) max = -1(-1K)
System.out.println("nonHeapUsage:" + nonHeapUsage); 


http://www.kler.cn/a/565049.html

相关文章:

  • JavaScript 简单类型与复杂类型-复杂类型的内存分配
  • Redis中SDS的数据结构
  • SpringBoot 日志文件相关 门面模式
  • ONNX转RKNN的环境搭建
  • 【Day48 LeetCode】图论问题 Ⅵ
  • Vue-Flow绘制流程图(Vue3+ElementPlus+TS)简单案例
  • 【C++教程】布尔类型
  • python量化交易——金融数据管理最佳实践——qteasy创建本地数据源
  • 8.Dashboard的导入导出
  • 打破关节动力桎梏!杭州宇树科技如何用“一体化设计”重塑四足机器人性能?
  • MFC获取所有硬件厂商和序列号
  • 如何搭建和管理 FTP 服务器
  • 【精】使用 Apktool 反编译 APK 并重新签名的详细教程
  • es 写入数据的工作原理是什么啊?es 查询数据的工作原理是什么啊?底层的 lucene 介绍一下呗?倒排索引了解吗?
  • JVM 面试
  • GEO数据结构
  • DeepSeek 开源狂欢周(一)FlashMLA:高效推理加速新时代
  • vue从入门到精通(十六):自定义指令
  • 神经网络中感受野的概念和作用
  • 浅谈C++/C命名冲突