我的编程空间,编程开发者的网络收藏夹
学习永远不晚

正确理解和使用Java中的字符串常量池

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

正确理解和使用Java中的字符串常量池

理解字符串常量池

当您从在类中写一个字符串字面量时,JVM将首先检查该字符串是否已存在于字符串常量池中,如果存在,JVM 将返回对现有字符串对象的引用,而不是创建新对象。我们通过一个例子更好的来理解。

比如下面的代码:

String s1 = "Harry Potter";
String s2 = "The Lord of the Rings";
String s3 = "Harry Potter";

在这段代码中,JVM 将创建一个值为“Harry Potter”的字符串对象,并将其存储在字符串常量池中。s1和s3都将是对该单个字符串对象的引用。

如果s2的字符串内容“The Lord of the Rings”不存在于池中,则在字符串池中生成一个新的字符串对象。

两种创建字符串方式

在 Java​ 编程语言中有两种创建 String​ 的方法。第一种方式是使用String Literal​字符串字面量的方式,另一种方式是使用new关键字。他们创建的字符串对象是都在常量池中吗?

字符串字面量的方式创建

String s1 = "Harry Potter";
String s2 = "The Lord of the Rings";
String s3 = "Harry Potter";

new关键字创建

String s4 = new String("Harry Potter");
String s5 = new String("The Lord of the Rings");

我们来比较下他们引用的是否是同一个对象:

s1==s3 //
s1==s4 //
s2==s5 //

使用 == 运算符比较两个对象时,它会比较内存中的地址。

正如您在上面的图片和示例中看到的,每当我们使用new运算符创建字符串时,它都会在 Java 堆中创建一个新的字符串对象,并且不会检查该对象是否在字符串常量池中。

那么我现在有个问题,如果是字符串拼接的情况,又是怎么样的呢?

字符串拼接方式

前面讲清楚了通过直接用字面量的方式,也就是引号的方式和用new关键字创建字符串,他们创建出的字符串对象在堆中存储在不同的地方,那么我们现在来看看用+这个运算符拼接会怎么样。

例子1

public static void test1() {
// 都是常量,前端编译期会进行代码优化
// 通过idea直接看对应的反编译的class文件,会显示 String s1 = "abc"; 说明做了代码优化
String s1 = "a" + "b" + "c";
String s2 = "abc";

// true,有上述可知,s1和s2实际上指向字符串常量池中的同一个值
System.out.println(s1 == s2);
}

常量与常量的拼接结果在常量池,原理是编译期优化。

例子2

public static void test5() {
String s1 = "javaEE";
String s2 = "hadoop";

String s3 = "javaEEhadoop";
String s4 = "javaEE" + "hadoop";
String s5 = s1 + "hadoop";
String s6 = "javaEE" + s2;
String s7 = s1 + s2;

System.out.println(s3 == s4); // true 编译期优化
System.out.println(s3 == s5); // false s1是变量,不能编译期优化
System.out.println(s3 == s6); // false s2是变量,不能编译期优化
System.out.println(s3 == s7); // false s1、s2都是变量
System.out.println(s5 == s6); // false s5、s6 不同的对象实例
System.out.println(s5 == s7); // false s5、s7 不同的对象实例
System.out.println(s6 == s7); // false s6、s7 不同的对象实例
}

只要其中有一个是变量,结果就在堆中, 变量拼接的底层原理其实是StringBuilder。

例子3:

public void test6(){
String s0 = "beijing";
String s1 = "bei";
String s2 = "jing";
String s3 = s1 + s2;
System.out.println(s0 == s3); // false s3指向对象实例,s0指向字符串常量池中的"beijing"
String s7 = "shanxi";
final String s4 = "shan";
final String s5 = "xi";
String s6 = s4 + s5;
System.out.println(s6 == s7); // true s4和s5是final修饰的,编译期就能确定s6的值了
}

不使用final修饰,即为变量。如s3行的s1和s2,会通过new StringBuilder进行拼接

使用final修饰,即为常量。会在编译器进行代码优化。

妙用String.intern() 方法

前面提到new关键字创建出来的字符串对象以及某些和变量进行拼接不会在字符串常量池中,而是直接在堆中新建了一个对象。这样不大好,做不到复用,节约不了空间。那有什么好办法呢?intern()就派上用场了,这个非常有用。

intern()方法的作用可以理解为主动将常量池中还没有的字符串对象放入池中,并返回此对象地址。

String s6 = new String("The Lord of the Rings").intern();

s2==s6 //真
s2==s5 //假

字符串常量池有多大?

关于字符串常量池究竟有多大,我也说不上来,但是讲清楚它底层的数据结构,也许你就明白了。

字符串常量池是一个固定大小的HashTable​,哈希表,默认值大小长度是1009​。如果放进String Pool的String​非常多,就会造成Hash​冲突严重,从而导致链表会很长,而链表长了后直接会造成的影响就是当调用String.intern时性能会大幅下降。

使用-XX:StringTablesize​可设置StringTable的长度

在jdk6中StringTable​是固定的,就是1009​的长度,所以如果常量池中的字符串过多就会导致效率下降很快。StringTable Size设置没有要求

在jdk7中,StringTable的长度默认值是60013,StringTable Size设置没有要求

  • 在jdk8中,设置StringTable​长度的话,1009是可以设置的最小值

字符串常量池的优缺点

字符串池的优点

  • 提高性能。由于 JVM 可以返回对现有字符串对象的引用而不是创建新对象,因此使用字符串池时字符串操作更快。
  • 共享字符串,节省内存。字符串池允许您在不同的变量和对象之间共享字符串,通过避免创建不必要的字符串对象来帮助节省内存。

字符串池的缺点

它有可能导致性能下降。从池中检索字符串需要搜索池中的所有字符串,这可能比简单地创建一个新的字符串对象要慢。如果程序创建和丢弃大量字符串,则尤其如此,因为每次使用字符串时都需要搜索字符串池。

总结

其实在 Java 7 之前,JVM将 Java String Pool​放置在PermGen​空间中,它具有固定大小——它不能在运行时扩展,也不符合垃圾回收的条件。在PermGen​(而不是堆)中驻留字符串的风险是,如果我们驻留太多字符串,我们可能会从 JVM 得到一个OutOfMemory​错误。从 Java 7 开始,Java String Pool​存放在Heap空间​,由 JVM进行垃圾回收。这种方法的优点是降低了OutOfMemory错误的风险,因为未引用的字符串将从池中删除,从而释放内存。

现在通过本文的学习,你该知道如何更好的创建字符串对象了吧。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

正确理解和使用Java中的字符串常量池

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

正确理解和使用Java中的字符串常量池

当您从在类中写一个字符串字面量时,JVM将首先检查该字符串是否已存在于字符串常量池中,如果存在,JVM 将返回对现有字符串对象的引用,而不是创建新对象。我们通过一个例子更好的来理解。

一文带你了解如何正确使用Java中的字符串常量池

研究表明,Java堆中对象占据最大比重的就是字符串对象,所以弄清楚字符串知识对学习Java很重要。本文主要重点聊聊字符串常量池,希望对大家有所帮助
2022-12-26

深入理解Java字符串常量池

Java字符串常量池是Java语言中一个重要而又神秘的概念。许多Java程序员对它有一定的了解,但深入理解它的工作原理和优化技巧对于编写高效的Java代码至关重要。本文将带你深入探索Java字符串常量池,从基础知识到高级优化策略,逐步揭开它
JavaString2024-11-30

JAVA中的字符串常量池使用操作代码

Java中的字符串常量池是Java堆中的一块特殊存储区域,用于存储字符串。它的实现是为了提高字符串操作的性能并节省内存,这篇文章主要介绍了JAVA中的字符串常量池,需要的朋友可以参考下
2022-12-27

JAVA中字符串常量池和缓冲池的作用是什么

这篇“JAVA中字符串常量池和缓冲池的作用是什么”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“JAVA中字符串常量池和缓冲池
2023-06-29

Java怎么清除常量池中的字符串

在Java中,无法直接清除常量池中的字符串。因为字符串常量池是一个特殊的内存区域,用于存储编译时期的字符串常量,它的内容是不可变的。一旦一个字符串被创建并放入常量池中,就无法从常量池中删除它。但是,可以通过让字符串变量的引用置为null来
Java怎么清除常量池中的字符串
2024-02-29

Java字符串常量池的作用是什么

这篇文章将为大家详细讲解有关Java字符串常量池的作用是什么,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。为什么会有常量池的概念?不知道小伙伴们是否有思考过这个问题? 没有思考也无所谓,小编
2023-06-20

.NET字符串内存管理:常量字符串、动态创建和字符串池的巧妙结合

字符串在内存中的存储方式主要有两种:字符串常量和动态创建的字符串。字符串常量可以共享内存,而动态创建的字符串存储在堆上。字符串池用于存储字符串常量,以便在需要时重用相同的字符串对象,减少内存开销。

Java中的堆、栈和常量池怎么理解

这篇文章主要讲解了“Java中的堆、栈和常量池怎么理解”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Java中的堆、栈和常量池怎么理解”吧!1.寄存器最快的存储区, 由编译器根据需求进行分配
2023-06-17

Java 关键字 volatile 的理解与正确使用

概述Java语言中关键字 volatile 被称作轻量级的 synchronized,与synchronized相比,volatile编码相对简单且运行的时的开销较少,但能够正确合理的应用好 volatile 并不是那么的容易,因为它比使用
2023-05-31

如何理解Java 虚拟机中的String 类和常量池

本篇文章为大家展示了如何理解Java 虚拟机中的String 类和常量池,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。一、String 对象的两种创建方式String str1 = "abcd";S
2023-06-05

Java 中如何使用正则表达式匹配字符串?(java正则表达式匹配字符串的方法是什么)

在Java编程中,正则表达式是一种强大的工具,用于在字符串中搜索、匹配和操作特定的模式。正则表达式提供了一种简洁而灵活的方式来处理文本数据。本文将详细介绍Java中使用正则表达式匹配字符串的方法。一、正则表达式的基本概念
Java 中如何使用正则表达式匹配字符串?(java正则表达式匹配字符串的方法是什么)
Java2024-12-19

Java 中变量的访问修饰符有哪些?如何正确使用?(java中变量的访问修饰符)

在Java编程中,变量的访问修饰符起着至关重要的作用,它决定了变量在不同的类和包中的可见性和可访问性。本文将详细介绍Java中变量的访问修饰符,包括它们的作用、使用场景以及相关的注意事项。一、访问修饰符的种类Java中
Java 中变量的访问修饰符有哪些?如何正确使用?(java中变量的访问修饰符)
Java2024-12-20

Java 中的 parseInt 方法如何妥善处理字符串解析异常情况?(Java parseInt方法如何处理字符串解析异常情况)

在Java编程中,parseInt方法是一个常用的用于将字符串转换为整数的方法。然而,在使用parseInt方法时,我们需要特别注意字符串解析异常情况,因为如果字符串不能被正确解析为整数,parseInt方法将会
Java 中的 parseInt 方法如何妥善处理字符串解析异常情况?(Java parseInt方法如何处理字符串解析异常情况)
Java2024-12-15

Java 中 charAt 函数在字符串处理方面的常见用法有哪些?(charat函数在Java字符串处理中的常见用法有哪些)

在Java编程中,字符串是一种非常重要的数据类型,而charAt函数是处理字符串的常用工具之一。它允许我们获取字符串中特定位置的字符,为字符串的操作和处理提供了便利。以下是charAt函数在Java字符串处理中的常见用法:一、获取字符串中的单个字
Java 中 charAt 函数在字符串处理方面的常见用法有哪些?(charat函数在Java字符串处理中的常见用法有哪些)
Java2024-12-18

使用Java怎么实现统计字符串中汉字与英文的数量

这期内容当中小编将会给大家带来有关使用Java怎么实现统计字符串中汉字与英文的数量,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。具体如下:package org.zhy.demo.algorithm;/*
2023-05-31

Java中字符串String的+和+=及循环操作String原理详解

Java编译器在编译时对String的+和+=操作会创建StringBuilder对象来进行字符串的拼接,下面这篇文章主要给大家介绍了关于Java中字符串String的+和+=及循环操作String原理的相关资料,需要的朋友可以参考下
2023-01-30

Java 新手如何使用Spring MVC 中的查询字符串和查询参数

目录前言什么是查询字符串和查询参数?Spring MVC中的查询参数处理可选参数处理多个值处理查询参数的默认值处理查询字符串示例:
Java 新手如何使用Spring MVC 中的查询字符串和查询参数
2024-01-21

详解Java中static关键字的使用和原理

static可以用来修饰的成员变量和成员方法,被修饰的成员是属于类的,而不是单单是属于某个对象的。本文就来详细说说他的使用和原理,需要的可以参考一下
2022-11-13

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录