String常量池

看《深入理解Java虚拟机》时发现一段有意思的代码:

//jdk 1.8
    String str1=new StringBuilder("计算机").append("软件").toString();
    System.out.println(str1.intern()==str1);//true
    String str2=new StringBuilder("ja").append("va").toString();
    System.out.println(str2.intern()==str2);//false

这段代码在jdk1.6中运行得到两个false,因为jdk1.6的intern()方法将首次出现的字符串实例复制到常量池中并返回该实例的引用,而该引用与原字符串的引用(堆中的引用)不同所以得到false,jdk1.7之后,对于第一次出现的字符,intern()将位于堆中的引用复制到常量池中,之后返回的是这个引用,所以上文str1.intern()==str1得到的是true。而str2.intern()==str2得到的却是false,这是因为”java”字符串已经在StringBuilder.toString()前存入常量池了。换成其他字符得到的就是true了。

String str3=new StringBuilder("jaaaaaa").append("va").toString();
        System.out.println(str3.intern()==str3);//true

关于常量池的详细理解参见链接
http://blog.youkuaiyun.com/xsf50717/article/details/47339415

http://book.51cto.com/art/201202/317488.htm

将其中的总结摘抄如下,并做一些补充:
1. 单独使用”“引号创建的字符串都是常量,编译期就已经确定存储到常量中。
2. 使用new String(“”)创建的对象会存储到heap中,是运行期新创建的,如果常量池中没有,常量池也存放一份。所以new String()会创建两个或一个对象。
3. 使用只包含常量的字符串连接符如”aa” + “aa”创建的也是常量,编译期就能确定,已经确定存储到常量池中。
4. 使用包含变量的字符串连接符如”aa” + s1创建的对象是运行期才创建的,存储在heap中,但常量池不会存放该对象。

### ### Java String 常量池详解 在 Java 中,String 是一个特殊的类,它不仅被广泛使用,还通过常量池机制进行了深度优化。为了提升性能和减少内存开销,JVM 引入了 **字符串常量池StringTable)** 来管理字符串字面量。 #### 字符串常量池的定义与作用 字符串常量池是一种特殊的内存区域,用于存储字符串字面量(literal)和通过 `intern()` 方法添加的字符串对象。当使用双引号定义字符串时,JVM 会优先检查常量池中是否已存在该字符串,若存在则直接返回其引用,否则新建一个并放入池中[^2]。 例如: ```java String a = "hello"; String b = "hello"; ``` 变量 `a` 和 `b` 指向的是同一个字符串对象,因为它们都指向常量池中的同一个实例。 #### 字符串常量池的运行机制 在程序运行期间,字符串常量池会动态管理字符串实例。当使用 `new String("hello")` 创建字符串时,JVM 会先检查常量池中是否存在 `"hello"`,若不存在则将其加入常量池。随后,会在堆中创建一个新的 `String` 对象,并指向常量池中的字符串[^4]。 ```java String c = new String("hello"); ``` 上述代码中,`c` 是一个堆中的新对象,但它内部的字符数据仍然指向常量池中的 `"hello"`。 #### intern() 方法与字符串池的扩展 Java 提供了 `String.intern()` 方法用于手动将字符串加入常量池。当调用该方法时,JVM 会查找常量池中是否存在相同 Unicode 编码的字符串,如果存在则返回其引用,否则将当前字符串加入常量池并返回其引用[^3]。 ```java String d = new String("world").intern(); String e = "world"; ``` 此时,`d` 和 `e` 指向的是同一个常量池中的字符串对象。 #### 常量池的分类与关系 字符串常量池是 JVM 中多种常量池机制的一部分。在编译时,类的常量信息(如字符串字面量)会被存储在 **静态常量池** 中。类加载后,这些信息会被加载到 **运行时常量池** 中,而字符串常量池则是运行时常量池的一部分,专门用于管理字符串字面量和通过 `intern()` 方法加入的字符串[^1]。 例如,在编译后的 `.class` 文件中,以下代码: ```java String a = "a"; String b = "b"; String ab = "ab"; ``` 会被编译为常量池中的符号信息,在类加载后成为运行时常量池的一部分,并在运行时由 JVM 管理其在字符串常量池中的存在[^5]。 #### 总结 字符串常量池通过复用字符串对象,减少了内存的使用并提升了程序性能。它与静态常量池、运行时常量池共同构成了 Java 的常量管理机制。开发者可以通过 `intern()` 方法显式控制字符串在常量池中的存在状态,从而优化内存使用。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值