还有人不了解String?

2019-05-18  本文已影响0人  有没有口罩给我一个

大家去面试的时候,有没有被问到下面关于string的题?我相信都有。

String s3 = new String("1") + new String("1");

内存有几个对象?两个?三个?四个?答案:四个。

来了解两个概念方法区和运行时常量,抄自深入理解Java虚拟机:

上面都是些概念性,举个栗子?

 //首先在堆内存中创建了一个对象,然后在常量池中查找equals "1"的对象,如果没有则创建一个对象保存在常量池中,有的话则不管
   String s1 = new String("aaa");
    // 在常量池中
    String s2 = "aaa";
   Log.e("tag","1"+ String.valueOf(s1 == s2));   // false


    // 首先在堆内存中创建了一个对象,然后在常量池中查找equals "1"的对象,如果没有则创建一个对象保存在常量池中,有的话则不管,之后调用intern()直接返回常量池中的引用
    s1 = new String("bbb").intern();
    s2 = "bbb";
   Log.e("tag","2"+ String.valueOf(s1 == s2));   // true

    //都是返回常量池中的引用
    s1 = "ccc";
    s2 = "ccc";
   Log.e("tag","3"+ String.valueOf(s1 == s2));   // true


    //调用intern()直接返回常量池中的引用
    s1 = new String("ddd").intern();
    s2 = new String("ddd").intern();
   Log.e("tag","4"+ String.valueOf(s1 == s2));   // true


    // 直接返回常量池中的引用,只要有任何一个不是字符串字面常量形式,都不会在常量池生成"aabb"
    s1 = "ab" + "cd";
    s2 = "abcd";
   Log.e("tag","5"+ String.valueOf(s1 == s2));   // true


    // 
    String temp = "hh";
    s1 = "a" + temp;//只要有任何一个不是字符串字面常量形式,都不会在常量池生成"ahh",且此时jvm做了优化,不会同时生成"a"和"hh"在字符串常量池中
    // 如果调用s1.intern 则最终返回true
    s2 = "ahh";//常量池中
   Log.e("tag","6"+ String.valueOf(s1 == s2));   // false

    temp = "hh".intern();//"hh" 被加入常量池中
    s1 = "a" + temp;//
    s2 = "ahh";
   Log.e("tag","7"+ String.valueOf(s1 == s2));   // false

    temp = "hh".intern();
    s1 = ("a" + temp).intern();
    s2 = "ahh";
   Log.e("tag","8"+ String.valueOf(s1 == s2));   // true

    s1 = new String("1");    // 同时会生成堆中的对象 以及常量池中1的对象,但是此时s1是指向堆中的对象的
    s1.intern();            // 常量池中的已经存在
    s2 = "1";
   Log.e("tag","9"+ String.valueOf(s1 == s2));   // false

    String s3 = new String("1") + new String("1");    // 此时生成了四个对象 常量池中的"1" + 2个堆中的"1" + s3指向的堆中的对象(注此时常量池不会生成"11")
    s3.intern();    // jdk1.7之后,常量池不仅仅可以存储对象,还可以存储对象的引用,会直接将s3的地址存储在常量池
    String s4 = "11";    // jdk1.7之后,常量池中的地址其实就是s3的地址
    Log.e("tag","10"+ String.valueOf(s3 == s4)); // jdk1.7之前false, jdk1.7之后 true

    s3 = new String("2") + new String("2");
    s4 = "22";        // 常量池中不存在22,所以会新开辟一个存储22对象的常量池地址
    s3.intern();    // 常量池22的地址和s3的地址不同
    Log.e("tag","11"+ String.valueOf(s3 == s4)); // false

对于什么时候会在常量池存储字符串对象,结论:

  1. 显示调用String的intern方法的时候;
  2. 直接声明字符串字面常量的时候,例如: String a = "aaa";
  3. 字符串直接常量相加的时候,例如: String c = "aa" + "bb";
    其中的aa或bb只要有任何一个不是字符串字面常量形式,都不会在常量池生成"aabb"且此时jvm做了优化,不会同时生成"aa"和"bb"在字符串常量池中
上一篇下一篇

猜你喜欢

热点阅读