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

java数据结构_再谈String_10

目录

字符串常量池

1. 创建对象的思考

2. 字符串常量池(StringTable)

3. 再谈String对象创建


字符串常量池

1. 创建对象的思考

下面两种创建String对象的方式相同吗?

public static void main(String[] args) {
     String s1 = "hello";
     String s2 = "hello";
     String s3 = new String("hello");
     String s4 = new String("hello");
     System.out.println(s1 == s2);    // true
     System.out.println(s1 == s3);    // false
     System.out.println(s3 == s4);    // false
 }

上述程序创建方式类似,但为什么打印结果s1 == s2的时候是true,s1 == s3打印的时候是false,s3 == s4打印的时候是false呢?

在Java中,对类似于1,2,3,3.14,“Hello”等字面类型的常量经常频繁使用,为了使程序的运行速度更快,更节省内存,Java为8中基本数据类型和String类都提供了常量池。

"池" 是编程中的一种常见的, 重要的提升效率的方式, 我们会在未来的学习中遇到各种 "内存池", "线程池", "数 据库连接池" .... 比如:家里给大家打生活费的方式 1. 家里经济拮据,每月定时打生活费,有时可能会晚,最差情况下可能需要向家里张口要,速度慢 2. 家里有矿,一次性打一年的生活费放到银行卡中,自己随用随取,速度非常快 方式2,就是池化技术的一种示例,钱放在卡上,随用随取,效率非常高。常见的池化技术比如:数据库连接 池、线程池等。

为了节省存储空间以及程序的与运行效率,Java中引入了:

        1.Class文件常量池:每个Java源文件编译后生成.class文件中会保存当前类中的字面常量以及符号信息

        2.运行时常量池:在.class文件被加载时候,.class文件中的常量池被加载到内存中称之为运行时常量池,运行时常量池每个类都有一份

        3.字符常量池

其具体概念会在讲解JVM时候详细解释

2. 字符串常量池(StringTable)

字符串常量池在Java的官方文档中并未详细说明,只是有一个在JVM中的StringTable类,实际上,是一个固定大小是HashTable(一种高效用来进行查找的数据结构),不同JDK版本下字符串常量池的位置以及默认大小是不同的:

3. 再谈String对象创建

如下图例子

打印结果为true

JVM中的大致过程如下:

在JVM中存在栈和堆

当执行代码String s1 = "hello"的时候,堆中有一个StringTable类,其中是一个HashTable,在HashTable中存储的是一些链表的地址

当我们String s1 = "hello"的时候,常量池中会有一个空间来指向一个链表,这个链表会再指向一个String对象,这个String对象的value会指向一个字符数组,从而存储起来hello

同时,栈中开辟一块空间存放0x11

这就是一次将常量放入常量池的过程。

存储字符串常量的时候,会先检查当前的常量池中 是否存在所要存储的常量,如果有的话,就不会再放入常量池中!

我们刚刚定义了String s2 = "hello",则先会再常量池中,寻找一下,是否有“hello”的常量,检查发现有,则直接再在栈中开辟一块空间给s2,但因为s2所要存储的常量以及在常量池中了,所以s2的地址直接指向常量池中是String对象即可

因此在打印s1 == s2的时候结果是true。

但如果是String s3 = new String("hello") 如下图,new出一个String对象

打印s3 == s4的结果是false

过程如下:

最开始创建s3的过程和s1是相同的,此处直接以s1的图为例:

但是,执行代码String s4 = new Sting("hello")的时候,因为只要是new出来的对象,就都是唯一的,所以会有如下过程:

即只要有new一个对象的发生,就一定会在堆中创建一个对象,则s4指向的地址为0x999的对象的地址,虽然也可以找到地址为0x88的hello,但s3和s4所存的地址就不一样了,所以在打印的时候,会出现s3 == s4的结果为false。

通过上面的例子可以看出:使用常量串创建String类型对象的效率更高,而且更节省空间。

用户也可以通过intern方法,将创建的字符串对象添加进字符串常量池中。

举例:有如下代码:

通过上面的讲解,可以知道 s1 == s2打印的结果一定是false

过程如下:

char[] ch = new char[]{'a', 'b', 'c'};之后,会在堆里有一个字符数组0x87,然后ch中存储地址0x87

然后String s1 = new String(ch); 和上面一样,new出了新是对象String

执行String s2 = "abc"的时候,过程如下,将 a b c放入常量池中

但此时s1 和 s2指向的地址并不相同,索引打印结果自然是false

但如果在String s1 = new String(ch)的代码之后,添加一行代码:s1.intern(),就可以实现将s1对象的引用放入到常量池

代码如下:

如果有了这行代码,过程就会是如下:

这样s1 和 s2指向的位置就是相同的了,s1 == s2打印的结果就是true

补充intern方法:

1.会先检查调用该方法所指的对象是否在常量池中存在

2.如果有则不入池

3.如果没有则入池

相关文章:

  • 索引(MySQL)
  • C# iText 抽取PDF页特定区域文本内容
  • MySQL:MySQL的数据类型
  • Autojs无线连接vscode方法
  • 【JAVA架构师成长之路】【持久层】第2集:SQL常用优化手段
  • 高精算法的用法及其优势
  • PHP之数组
  • Java 多线程
  • 初识Qt · 信号与槽 · 基础知识
  • 计算机视觉算法实战——图像分割(主页有源码)
  • 【FFmpeg之如何新增一个硬件解码器】
  • LeetCode 双指针章节
  • 【Spring AOP】_切点类的切点表达式
  • 安装IK分词器;IK分词器配置扩展词库:配置扩展字典-扩展词,配置扩展停止词字典-停用词
  • 【工具】COME对比映射学习用于scRNA-seq数据的空间重构
  • 通过HTML有序列表(ol/li)实现自动递增编号的完整解决方案
  • 基于遗传算法的无人机三维路径规划仿真步骤详解
  • GStreamer —— 2.3、Windows下Qt加载GStreamer库后运行 - “教程3:动态管道“(附:完整源码)
  • Redis7——进阶篇(三)
  • LLM实践——DeepSeek技术报告学习(含实现逻辑梳理)
  • jsp如何做动态网站/海外网站
  • wordpress唯美破解主题/seo外链发布技巧
  • 成都哪家做网站建设比较好/软文推广发布平台
  • 那个网站做的好/百度站长资源平台
  • 古镇企业网站建设定制/seo推广软件
  • wordpress 快速发布/福州搜索引擎优化公司