必威体育Betway必威体育官网
当前位置:首页 > IT技术

String.intern()详细分析,JDK1.8

时间:2019-10-17 16:41:05来源:IT技术作者:seo实验室小编阅读:59次「手机版」
 

intern

前言

  本文的测试环境是jdk1.8

  String.intern()分析在文中偏后

一.创建字符串分析

1.直接使用双引号创建字符串

 判断这个常量是否存在于常量池,

  如果存在,

   判断这个常量是存在的引用还是常量,

    如果是引用,返回引用地址指向的堆空间对象,

    如果是常量,则直接返回常量池常量,

  如果不存在,

    在常量池中创建该常量,并返回此常量

    String a1 = "AA";//在常量池上创建常量AA
    String a2 = "AA";//直接返回已经存在的常量AA
    System.out.println(a1 == a2); //true
    String a3 = new String("AA");    //在堆上创建对象AA
    a3.intern(); //在常量池上创建对象AA的引用
    String a4 = "AA"; //常量池上存在引用AA,直接返回该引用指向的堆空间对象,即a3
    System.out.println(a3 == a4); //false,如果这个例子不理解,请看完整篇文章再回来看这里

2.new String创建字符串

 首先在堆上创建对象(无论堆上是否存在相同字面量的对象),

 然后判断常量池上是否存在字符串的字面量,

  如果不存在

   在常量池上创建常量

  如果存在

   不做任何操作

    String a1 = new String("AA");
    String a2 = new String("AA");
    System.out.println(a1 == a2); //false
    //如果常量池上不存在常量AA,也不存在引用AA,则创建常量AA
    String a1 = new String("AA");
    System.out.println(a1 == a1.intern()); //false

3.双引号相加

 判断这两个常量、相加后的常量在常量池上是否存在

  如果不存在

   则在常量池上创建相应的常量

  如果存在

   判断这个常量是存在的引用还是常量,

    如果是引用,返回引用地址指向的堆空间对象,

    如果是常量,则直接返回常量池常量,

    String a1 = "AA" + "BB";//在常量池上创建常量AA、BB和AABB,并返回AABB
    //常量池上存在常量AABB
    String a2 = "AABB";
    String a3 = "AA" + "BB";
    System.out.println(a2 == a3); //true
    //常量池上存在引用AABB
    String a4 = new String("AA") + new String("BB"); //在堆上创建对象AA、BB和AABB,在常量池上创建常量AA和BB
    a4.intern();
    String a5 = "AA" + "BB";
    System.out.println(a4 == a5); //true

4.两个new String相加

 首先会创建这两个对象以及相加后的对象

 然后判断常量池中是否存在这两个对象的字面量常量

  如果存在

   不做任何操作

  如果不存在

   则在常量池上创建对应常量

    //常量AA不存在,所以第一步在常量池中创建了常量AA
    String a2 = new String("AA") + new String("BB");
    String a3 = new String("A")+new String("A"); //创建对象AA
    System.out.println(a3 == a3.intern()); //false
    //只在堆上创建AABB对象,没有在常量池中创建常量AABB
    String a2 = new String("AA") + new String("BB");
    System.out.println(a2 == a2.intern()); //true

5.双引号字符串与new String字符串

 首先创建两个对象,一个是new String的对象,一个是相加后的对象

 然后判断双引号常量与new String的字面量在常量池是否存在

  如果存在

   不做操作

  如果不存在

   则在常量池上创建对象的常量

    String a1 = "AABB";
    String a2 = "AA" + new String("BB");
    System.out.println(a1 == a2.intern());//true
    System.out.println(a2 == a2.intern()); //false

二.String.intern()分析

 判断这个常量是否存在于常量池。

  如果存在

   判断存在内容是引用还是常量,

    如果是引用,

     返回引用地址指向堆空间对象,

    如果是常量,

     直接返回常量池常量

  如果不存在,

   将当前对象引用复制到常量池,并且返回的是当前对象的引用

    String a1 = "AA";
    System.out.println(a1 == a1.intern()); //true
    String a2 = new String("B") + new String("B");
    a2.intern();
    String a3 = new String("B") + new String("B");
    System.out.println(a2 == a3.intern());//true
    System.out.println(a3 == a3.intern());//false
    String a4 = new String("C") + new String("C");
    System.out.println(a4 == a4.intern()); //true

三.总结

1.只在常量池上创建常量

    String a1 = "AA";

2.只在堆上创建对象

    String a2 = new String("A") + new String("A");

3.在堆上创建对象,在常量池上创建常量

    String a3 = new String("AA");

4.在堆上创建对象,在常量池上创建引用

    String a4 = new String("A") + new String("A");//只在堆上创建对象AA
    a4.intern();//将该对象AA的引用保存到常量池上

5.在堆上创建对象,在常量池上创建常量,在常量池上创建引用(不可能)

    String a5 = new String("A") + new String("A");//只在堆上创建对象
    a5.intern();//在常量池上创建引用
    String a6 = "AA";//此时不会再在常量池上创建常量AA,而是将a5的引用返回给a6
    System.out.println(a5 == a6); //true

6.

常量池上常量 常量池上引用 堆上对象
常量池上常量 X 不共存 共存
常量池上引用 不共存 X 共存
堆上对象 共存 共存 X

四.练习

         String aa = "AA";//设置常量AA到常量池
         String bb = "BB";//设置常量BB到常量池
         String ccdd = "CC"+"DD";//设置常量CCDD到常量池
         String neeff = new String("EE")+new String("FF");//设置EE和FF到常量池。并且添加EE、FF和EEFF对象到堆
         String aabb = aa+bb;//添加AABB对象到堆
         String gghh = "GG"+new String("HH");//设置GG和HH常量到常量池,设置HH和GGHH对象到堆
//         aa.intern();//啥事都不做,返回AA常量
//         ccdd.intern();//啥事都不做,返回CCDD常量
//         neeff.intern();//添加EEFF对象的引用到常量池,并返回EEFF对象
//         aabb.intern();//添加AABB对象的引用到常量池,并返回AABB对象
//         gghh.intern();//添加GGHH对象的引用到常量池,并返回GGHH对象
         System.out.println(aa.intern()==aa); //true
         System.out.println(neeff.intern()=="EEFF");//true
         System.out.println("EEFF"==neeff);//true
         String nccdd = new String("CCDD");
         System.out.println(ccdd==nccdd);//false
         System.out.println(ccdd==nccdd.intern());//true
         System.out.println(aabb.intern()==aabb);//true
         System.out.println(gghh==gghh.intern());//true

相关阅读

js字符串截取函数slice()、substring()、substr()

原文地址为:js字符串截取函数slice()、substring()、substr()摘要在js中字符截取函数有常用的三个slice()、substring()、substr()

StringBuilder重用 清空数据方法

StringBuilder:线程非安全的  StringBuffer:线程安全的  StringBuilder 没有提供clear或empty方法。 清空有3种方法: 1)新生成

String数组的创建

string数组的定义有三种写法: String[] arr = new String[10]; //创建一个长度为10的String 类型数组。 String arr[] = new St

Java开发者如何正确的使用String,StringBuffer,StringBu

通过前面的文章,我们知道String类最大的特点是不可变性,这意味着对String类的任何修改都会新生成一个字符串,比如你执行了String类的

无法解析的外部符号 "class google::protobuf::inter

无法解析的外部符号 "class google::protobuf::internal::ExplicitlyConstructed<class std::basic_string<char,struct std::cha

分享到:

栏目导航

推荐阅读

热门阅读