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

文章目录

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

java 复制代码
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)中的属性值引用的是同一个对象吗?

java 复制代码
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. 获取堆内存使用情况、非堆内存使用情况

java 复制代码
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); 
相关推荐
web136885658715 分钟前
PHP For 循环
android·java·php
loyd319 分钟前
【数据分析】5 设计不同业务分析框架
java·网络·数据分析
m0_7482451726 分钟前
Spring Boot项目开发常见问题及解决方案(上)
java·spring boot·后端
今天的接口写完了吗?26 分钟前
Spring Boot操作MaxComputer(保姆级教程)
java·spring boot·后端
Jumbo星38 分钟前
ms-swift 3.x和2.x中参数不一致的暗坑
开发语言·ios·swift
金州小铁匠39 分钟前
基于EasyExcel封装的Excel工具类,支持高效导出和读取操作
java·spring·excel
B.-42 分钟前
在已有的原生 App 里嵌入 Flutter 页面的方法
开发语言·flutter·macos·cocoa
IIIIIIlllii42 分钟前
java练习(43)
java·开发语言
Neo Evolution1 小时前
每天一个Flutter开发小项目 (6) : 表单与验证的专业实践 - 构建预约应用
android·开发语言·前端·javascript·flutter
xxxxxmy1 小时前
Spring MVC 程序开发(1)
java·spring·mvc