String类的分析

String中intern的方法

首先查看官方API那个的解释:

———————————————————————————————————————

intern

public String intern()

返回字符串对象的规范化表示形式。

一个初始时为空的字符串池,它由类 String 私有地维护。

当调用 intern 方法时,如果池已经包含一个等于此 String 对象的字符串(该对象由 equals(Object) 方法确定),则返回池中的字符串。否则,将此 String 对象添加到池中,并且返回此 String 对象的引用。

它遵循对于任何两个字符串 s 和 t,当且仅当 s.equals(t) 为 true 时,s.intern() == t.intern() 才为 true。

所有字面值字符串和字符串赋值常量表达式都是内部的。

返回:

一个字符串,内容与此字符串相同,但它保证来自字符串池中。

 

———————————————————————————————————————

尽管在输出中调用intern方法并没有什么效果,但是实际上后台这个方法会做一系列的动作和操作。在调用”ab”.intern()方法的时候会返回”ab”,但是这个方法会首先检查字符串池中是否有”ab”这个字符串,如果存在则返回这个字符串的引用,否则就将这个字符串添加到字符串池中,然会返回这个字符串的引用。

可以看下面一个范例:

 
   
<!-- Code highlighting produced by Actipro CodeHighlighter (freeware) http://www.CodeHighlighter.com/ --> 1 String str1 = " a " ; 2 String str2 = " b " ; 3 String str3 = " ab " ; 4 String str4 = str1 + str2; 5 String str5 = new String( " ab " ); 6 7 System.out.println(str5.equals(str3)); 8 System.out.println(str5 == str3); 9 System.out.println(str5.intern() == str3); 10 System.out.println(str5.intern() == str4);

得到的结果:

true

false

true

false

 

为什么会得到这样的一个结果呢?我们一步一步的分析。

第一、str5.equals(str3)这个结果为true,不用太多的解释,因为字符串的值的内容相同。

第二、str5 == str3对比的是引用的地址是否相同,由于str5采用new String方式定义的,所以地址引用一定不相等。所以结果为false。

第三、当str5调用intern的时候,会检查字符串池中是否含有该字符串。由于之前定义的str3已经进入字符串池中,所以会得到相同的引用。

第四,当str4 = str1 + str2后,str4的值也为”ab”,但是为什么这个结果会是false呢?先看下面代码:

 
   
<!-- Code highlighting produced by Actipro CodeHighlighter (freeware) http://www.CodeHighlighter.com/ --> 1 String a = new String( " ab " ); 2 String b = new String( " ab " ); 3 String c = " ab " ; 4 String d = " a " + " b " ; 5 String e = " b " ; 6 String f = " a " + e; 7 8 System.out.println(b.intern() == a); 9 System.out.println(b.intern() == c); 10 System.out.println(b.intern() == d); 11 System.out.println(b.intern() == f); 12 System.out.println(b.intern() == a.intern());

运行结果:

false

true

true

false

true

 

由运行结果可以看出来,b.intern() == a和b.intern() == c可知,采用new 创建的字符串对象不进入字符串池,并且通过b.intern() == d和b.intern() == f可知,字符串相加的时候,都是静态字符串的结果会添加到字符串池,如果其中含有变量(如f中的e)则不会进入字符串池中。但是字符串一旦进入字符串池中,就会先查找池中有无此对象。如果有此对象,则让对象引用指向此对象。如果无此对象,则先创建此对象,再让对象引用指向此对象。

    当研究到这个地方的时候,突然想起来经常遇到的一个比较经典的Java问题,就是对比equal和==的区别,当时记得老师只是说“==”判断的是“地址”,但是并没说清楚什么时候会有地址相等的情况。现在看来,在定义变量的时候赋值,如果赋值的是静态的字符串,就会执行进入字符串池的操作,如果池中含有该字符串,则返回引用。

执行下面的代码:

 
   
<!-- Code highlighting produced by Actipro CodeHighlighter (freeware) http://www.CodeHighlighter.com/ --> 1 String a = " abc " ; 2 String b = " abc " ; 3 String c = " a " + " b " + " c " ; 4 String d = " a " + " bc " ; 5 String e = " ab " + " c " ; 6 7 System.out.println(a == b); 8 System.out.println(a == c); 9 System.out.println(a == d); 10 System.out.println(a == e); 11 System.out.println(c == d); 12 System.out.println(c == e);

运行的结果:

true

true

true

true

true

true

运行的结果刚好验证了我刚才的猜想。

 

***************************************************************

String a = new String("xyz");到底是几个对象?

哈哈,要理解这个,就要知道string类的工作原理。

你知道在java中除了8中基本类型外,其他的都是类对象以及其引用。所以 "xyz "在java中它是一个String对象.对于string类对象来说他的对象值是不能修改的,也就是具有不变性。

看:

String s= "Hello ";

s= "Java ";

String s1= "Hello ";

String s2=new String( "Hello ");

啊,s所引用的string对象不是被修改了吗?之前所说的不变性,去那里了啊?

你别着急,让我告诉你说发生了什么事情:

在jvm的工作过程中,会创建一片的内存空间专门存入string对象。我们把这片内存空间叫做string池。

String s= "Hello ";当jvm看到 "Hello ",在string池创建string对象存储它,并将他的引用返回给s。

s= "Java ",当jvm看到 "Java ",在string池创建新的string对象存储它,再把新建的string对象的引用返回给s。而原先的 "Hello "仍然在string池内。没有消失,他是不能被修改的。

所以我们仅仅是改变了s的引用,而没有改变他所引用的对象,因为string对象的值是不能被修改的。

String s1= "Hello ";jvm首先在string池内里面看找不找到字符串 "Hello ",找到,返回他的引用给s1,否则,创建新的string对象,放到string池里。这里由于s= "Hello "了,对象已经被引用,所以依据规则s和s1都是引用同一个对象。所以 s==s1将返回true。(==,对于非基本类型,是比较两引用是否引用内存中的同一个对象)

String s2=String( "Hello ");jvm首先在string池内里面看找不找到字符串 "Hello ",找到,不做任何事情,否则,创建新的string对象,放到string池里面。由于遇到了new,还会在内存上(不是string池里面)创建string对象存储 "Hello ",并将内存上的(不是string池内的)string对象返回给s2。所以s==s2将返回false,不是引用同一个对象。

好现在我们看题目:

String s = new String( "xyz ");

首先在string池内找,找到?不创建string对象,否则创建, 这样就一个string对象

遇到new运算符号了,在内存上创建string对象,并将其返回给s,又一个对象

所以总共是2个对象

例如::

String a=new String("haha");
              String b=new String("haha");
              String aa="haha";
              String aaa="ha"+"ha";
              String c="ha";
             
             
              System.out.println(a==b);
              System.out.println(a==aa);
              System.out.println(a==aaa);
              System.out.println(a==c+"ha");
         System.out.println(aa==aaa);

 

输出结果:false
                       false
                       false
                       false
                       true

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值