⑤. intern()的使用
前言:
(1). 如果不是用双引号声明的String对象,可以使用String提供的intern方法: intern方法会从字符串常量池中查询当前字符串是否存在,若不存在就会将当前字符串放入常量池中
(2). 比如: String myInfo = new String(“I love u”).intern();
也就是说,如果在任意字符串上调用String. intern方法,那么其返回结果所指向的那个类实例,必须和直接以常量形式出现的字符串实例完全相同。因此,下 列表达式的值必定是true:
(“a” + “b” + “c”).intern()== “abc”;
(3). 通俗点讲,Interned String就是确保字符串在内存里只有一份拷贝,这样可以节约内存空间,加快字符串操作任务的执行速度。注意,这个值会被存放在字符串内部池
(String Intern Pool)
①. new String(“ab”)会创建几个对象
- ①. new String(“ab”)会创建几个对象
(new String(“ab”)会创建几个对象?看字节码,就知道是两个)
- 一个对象是:new关键字在堆空间创建的
- 另一个对象是:字符串常量池中的对象"ab"。 字节码指令:ldc
- 如下图:(常量池中已经有了该对象)
②. new String(“a”) + new String(“b”)创建几个对象呢?
- ②. new String(“a”) + new String(“b”)呢?
(在常量池中是没有ab存在的 [很关键] )
- 对象1:new StringBuilder()
- 对象2: new String(“a”)
- 对象3: 常量池中的"a"
- 对象4: new String(“b”)
- 对象5: 常量池中的"b"
- 深入剖析: StringBuilder的toString(): 对象6 :new String(“ab”)
- 注意:强调一下,toString()的调用,在字符串常量池中,没有生成"ab" 没有ldc指令
③. 关于String.intern( )的面试题
- ③. 关于String.intern( )的面试题
/** * 如何保证变量s指向的是字符串常量池中的数据呢? * 有两种方式: * 方式一: String s = "shkstart";//字面量定义的方式 * 方式二: 调用intern() * String s = new String("shkstart").intern(); * String s = new StringBuilder("shkstart").toString().intern(); * */ public class StringIntern { public static void main(String[] args) { String s = new String("1"); String s1 = s.intern();//调用此方法之前,字符串常量池中已经存在了"1" String s2 = "1"; //s 指向堆空间"1"的内存地址 //s1 指向字符串常量池中"1"的内存地址 //s2 指向字符串常量池已存在的"1"的内存地址 所以 s1==s2 System.out.println(s == s2);//jdk6:false jdk7/8:false System.out.println(s1 == s2);//jdk6: true jdk7/8:true System.out.println(System.identityHashCode(s));//491044090 System.out.println(System.identityHashCode(s1));//644117698 System.out.println(System.identityHashCode(s2));//644117698 //s3变量记录的地址为:new String("11") String s3 = new String("1") + new String("1"); //执行完上一行代码以后,字符串常量池中,是否存在"11"呢?答案:不存在!! //在字符串常量池中生成"11"。如何理解:jdk6:创建了一个新的对象"11",也就有新的地址。 //jdk7:此时常量中并没有创建"11",而是创建一个指向堆空间中new String("11")的地址 s3.intern(); //s4变量记录的地址:使用的是上一行代码代码执行时,在常量池中生成的"11"的地址 String s4 = "11"; System.out.println(s3 == s4);//jdk6:false jdk7/8:true } }
public class StringIntern1 { public static void main(String[] args) { //StringIntern.java中练习的拓展: String s3 = new String("1") + new String("1");//new String("11") //执行完上一行代码以后,字符串常量池中,是否存在"11"呢?答案:不存在!! String s4 = "11";//在字符串常量池中生成对象"11" String s5 = s3.intern(); System.out.println(s3 == s4);//false System.out.println(s5 == s4);//true } }
④. 总结String的intern()的使用
- ④. 如果拼接的结果调用intern()方法分为两种情况:
JDK1.6,将这个字符串对象尝试放入串池 ①. 如果字符串常量池中有,则并不会放入。返回已有的串池中的对象的地址②. 如果没有,它会在常量池中创建一个对象放入串池中,并返回串池中的对象地址
JDK1.7,将这个字符串对象尝试放入串池 ①. 如果字符串常量池中有,则并不会放入。返回已有的串池中的对象的地址②. 如果没有,它不会创建一个对象,如果堆中已经这个字符串,那么会将堆中的引用地址赋给它
public class StringExer1 { public static void main(String[] args) { //String x = "ab"; String s = new String("a") + new String("b");//new String("ab") //在上一行代码执行完以后,字符串常量池中并没有"ab" String s2 = s.intern();//jdk6中:在串池中创建一个字符串"ab" //jdk8中:串池中没有创建字符串"ab",而是创建一个引用,指向new String("ab"),将此引用返回 System.out.println(s2 == "ab");//jdk6:true jdk8:true System.out.println(s == "ab");//jdk6:false jdk8:true } }
intern题目: