常量池也是JVM中的一块内存区域,在JDK1.6及以前,这时的方法区也被成为永久代;常量池是存储在方法区的,在JDK1.7之后,常量池被划分到了堆内存。
创建字符串的方式有很多种,不同的方式创建的字符串在内存中的表现形式是不一样的;因此我们在使用字符串做==比较时需要格外注意;因为==比较的是两个对象的内存地址值;
- package com.dfbz.demo01;
- /**
- * @author lscl
- * @version 1.0
- * @intro:
- */
- public class Demo06 {
- public static void main(String[] args) {
- // 存储在常量池
- String s1 = "abc";
- // new的东西都存储在堆内存
- String s2 = new String("abc");
- // new的东西都存储在堆内存
- String s3 = new String("abc");
- System.out.println(s1); // abc
- System.out.println(s2); // abc
- System.out.println(s3); // abc
- System.out.println(s1 == s2); // false
- System.out.println(s2 == s3); // false
- }
- }
-
内存图解:
我们之前就介绍过,String是一个不可变的字符序列,即字符串一旦创建就是不可变的;但我们在操作字符串时,字符串好像是可以拼接改变的,这究竟是怎么回事呢?
下列代码产生了几个字符串?
- String s1="1";
- s1="2";
-
答:2个,“1”、“2”;
- String s1 = "10" + "1";
-
答:3个,“10”、“1”、“101”;并且3个都存储在常量池
- String s1 = new String("abc");
-
答:2个,一个"abc"存储在常量池,另一个"abc"存储在堆内存;s1保存着堆内存中"abc"的引用;
- String s1 = "10";
- s1 += "1"; // s1=s1+"1";
-
tips:任何与变量相加的字符串都会产生一个新字符串,存储在堆内存;
总结:字符串一旦创建了就不能改变,如果改变了那也是创建了一个新的字符串将其返回,再也不是原来那个字符串了;
在字符串拼接时,只要是和变量相加,其结果都会产生一个新的字符串;
观察如下代码:
- int i = 1;
- String s1 = "10" + i;
- String s2 = "10" + i;
- System.out.println(s1 == s2); // false
-
将变量换成常量:
- String s1 = "10" + 1;
- String s2 = "10" + 1;
- System.out.println(s1 == s2); // true
-
- package com.dfbz.demo01;
- /**
- * @author lscl
- * @version 1.0
- * @intro:
- */
- public class Demo08 {
- public static void main(String[] args) {
- String s1 = "abc";
- String s2 = "abc";
- String s3 = "ab" + "c";
- String s4 = "ab" + new String("c");
- String s5 = new String("abc");
- String s6 = "a";
- String s7 = "bc";
- String s8 = s6 + s7;
- System.out.println(s1 == s2); // true(都存储在常量池)
- System.out.println(s1 == s3); // true(常量相加,生成的值也是在常量池,保留常量池中的引用)
- System.out.println(s1 == s4); // false(只要是和变量相加,都会new出一份新的)
- System.out.println(s1 == s8); // false(只要是和变量相加,都会new出一份新的)
- System.out.println(s4 == s5); // false
- }
- }
-
示例代码:
- package com.dfbz.demo01;
- /**
- * @author lscl
- * @version 1.0
- * @intro:
- */
- public class Demo09 {
- public static void main(String[] args) {
- String s1 = "1" + new String("1");
- // 查看常量池是否有"11"这个字符串,如果没有,则将"11"存入常量池,并返回常量池的地址
- String s2 = s1.intern();
- String s3 = "11";
- System.out.println(s2 == s3); // true
- }
- }
-
图解: