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

server/2025/3/3 5:40:19/

文章目录

  • 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. 判断一个字符串(引用)是否在常量池中

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); 

http://www.ppmy.cn/server/171698.html

相关文章

React vs Vue3深度对比与使用场景分析

在前端开发领域,React 和 Vue3 是两个备受瞩目的框架。它们都提供了强大的功能和灵活的开发方式,但各自的设计理念、使用方式和适用场景有所不同。本文将深入探讨 React 和 Vue3 的区别,通过代码示例和具体的使用场景,帮助开发者更…

FastAPI系列:如何响应txt和json文件

这篇简洁实用的文章向您展示了如何在FastAPI中返回TXT或JSON文件。我假设你已经对web框架有了一定的了解,所以我不会浪费你的时间来解释什么是FastAPI,它是如何工作的,或者漫谈它的历史。让我们直入主题吧。 返回TXT文件 TXT文件是一种纯文本…

linux 后台执行并输出日志

在Linux系统中,后台执行程序并输出日志通常有多种方法,这里列出几种常见的方法: 1. 使用&将命令放入后台 可以在命令的末尾加上&符号,将命令放入后台执行。例如: your_command > output.log 2>&1…

《Zookeeper 分布式过程协同技术详解》读书笔记-2

目录 zk的一些内部原理和应用请求,事务和标识读写操作事务标识(zxid) 群首选举Zab协议(ZooKeeper Atomic Broadcast protocol)文件系统和监听通知机制分布式配置中心, 简单Demojava code 集群管理code 分布式锁 zk的一…

基于SpringBoot的校园消费点评管理系统

作者:计算机学姐 开发技术:SpringBoot、SSM、Vue、MySQL、JSP、ElementUI、Python、小程序等,“文末源码”。 专栏推荐:前后端分离项目源码、SpringBoot项目源码、Vue项目源码、SSM项目源码、微信小程序源码 精品专栏:…

vue3 keep-alive 页面切换不触发onActivated和onDeactivated方法周期

<script setup lang"ts"> import { onActivated, onDeactivated, shallowRef } from vue import CompA from ../components/CompA.vue import CompB from ../components/CompB.vue const current shallowRef(CompA) onActivated(() > {console.log(组件被激…

springBean介绍

Spring Bean核心机制解析&#xff1a;从生命周期到工厂实践 引言 Spring框架的基石——Bean对象&#xff0c;承载着应用的核心业务逻辑。理解Bean的创建机理与管理策略&#xff0c;是构建高质量Spring应用的关键。本文将深入剖析Bean的全生命周期管理、容器核心架构设计&…

信号与系统笔记——第一章 信号与系统概述(二)

1.2 基本信号 全课程有三个关键问题 基本信号及基本响应任意信号的分解LTI系统分析 Z1.05 阶跃函数 1.定义 2.性质 &#xff08;1&#xff09;表示分段常量信号 f(t) 2ε(t) - 3ε(t-1) ε(t-2) &#xff08;2&#xff09;表示信号的作用区间 &#xff08;3&#xff09;积分…