java 字符串缓冲池 String缓冲池 == 和equals

来源:互联网 发布:淘宝代付关闭怎么回事 编辑:程序博客网 时间:2024/06/03 19:36
java中的数据类型,可分为两类: 
1.基本数据类型,也称原始数据类型。byte,short,char,int,long,float,double,boolean 
  他们之间的比较,应用双等号(==),比较的是他们的值。 
2.复合数据类型(类) 
  当他们用(==)进行比较的时候,比较的是他们在内存中的存放地址,所以,除非是同一个new出来的对象,他们的比较后的结果为true,否则比较后结果为false。 JAVA当中所有的类都是继承于Object这个基类的,在Object中的基类中定义了一个equals的方法,这个方法的初始行为是比较对象的内存地 址,但在一些类库当中这个方法被覆盖掉了,如String,Integer,Date在这些类当中equals有其自身的实现,而不再是比较类在堆内存中的存放地址了。

  对于复合数据类型之间进行equals比较,在没有覆写equals方法的情况下,他们之间的比较还是基于他们在内存中的存放位置的地址值的,因为Object的equals方法也是用双等号(==)进行比较的,所以比较后的结果跟双等号(==)的结果相同。


public class TestString { public static void main(String[] args) {String s1 = "Monday";String s2 = "Monday";if (s1 == s2){System.out.println("s1 == s2");}else{System.out.println("s1 != s2");}}}



编译并运行程序,输出:s1 == s2说明:s1 与 s2 引用同一个 String 对象 -- "Monday"!



public class TestString {public static void main(String[] args) {String s1 = "Monday";String s2 = new String("Monday");if (s1 == s2){System.out.println("s1 == s2");}else{System.out.println("s1 != s2");}if (s1.equals(s2)) {System.out.println("s1 equals s2");}else{System.out.println("s1 not equals s2");}}}

我们将s2用new操作符创建
程序输出:
s1 != s2
s1 equals s2
说明:s1 s2分别引用了两个"Monday"String对象


3. 字符串缓冲池
原来,程序在运行的时候会创建一个字符串缓冲池当使用 s2 = "Monday" 这样的表达是创建字符串的时候,程序首先会在这个String缓冲池中寻找相同值的对象,在第一个程序中,s1先被放到了池中,所以在s2被创建的时候,程序找到了具有相同值的 s1
将s2引用s1所引用的对象"Monday"
第二段程序中,使用了 new 操作符,他明白的告诉程序:"我要一个新的!不要旧的!"于是一个新的"Monday"Sting对象被创建在内存中。他们的值相同,但是位置不同,一个在池中游泳一个在岸边休息。哎呀,真是资源浪费,明明是一样的非要分开做什么呢?


</pre><pre name="code" class="html">public class TestString {public static void main(String[] args) {String s1 = "Monday";String s2 = new String("Monday");s2 = s2.intern();if (s1 == s2){System.out.println("s1 == s2");}else{System.out.println("s1 != s2");}if (s1.equals(s2)) {System.out.println("s1 equals s2");}else{System.out.println("s1 not equals s2");}}}


这次加入:s2 = s2.intern();
程序输出:
s1 == s2
s1 equals s2
原 来,(java.lang.String的intern()方法"abc".intern()方法的返回值还是字符串"abc",表面上看起来好像这个方 法没什么用处。但实际上,它做了个小动作:检查字符串池里是否存在"abc"这么一个字符串,如果存在,就返回池里的字符串;如果不存在,该方法会 把"abc"添加到字符串池中,然后再返回它的引用。




String缓冲池 

java 中 String 创建详细解释
字符串对象是一种特殊的对象.String 类是一个不可变的类..也就说,String 对象
一旦创建就不允许修改
String 类有一个对应的 String 池,也就是 String pool.每一个内容相同的字符串对
象都对应于一个 pool 里的对象.
1 看下面一段代码.

[java] view plaincopy在CODE上查看代码片派生到我的代码片
  1. String s = new String("abc");  
  2. String s1 = "abc";  
  3. String s2 = new String("abc");  
  4. System.out.println(s == s1);  
  5. System.out.println(s == s2);  
  6. System.out.println(s1 == s2);  
请问 前面三条语句分别创建了几个对象,分别是什么.后面的输出分别是什么
(1)String s = new String("abc"); 这句,创建了两个对象..其内容都是"abc".注意,
s 不是对象,只是引用.只有 new 生成的才是对象.
创建的流程是,首先括号里的"abc"先到 String pool 里看有没"abc"这个对象,没有
则在 pool 里创建这个对象..所以这里就在 pool 创建了一个"abc"对象.然后 通过
new 语句又创建了一个"abc"对象..而这个对象是放在内存的堆里. .这里的 s 指
向堆里的对象.
(2) String s1 = "abc"; 这条语句,s1 当然还是引用 .后面的"abc".其实就是上面括
号里的"abc".执行的是相同的操作.即 在 pool 里查找有没"abc"这个对象.没有则
创建一个...很显然,第一条语句在 pool 里已经创建了一个"abc".所以这条语句没
有创建对象,s1 指向的是 pool 中的"abc"
(3)String s2 = new String("abc"); 这条语句,其实和第一条是一样的。因为第一
条已经在 pool 中创建了"abc"这个对象,所以这条语句由于在 pool 中先找到了
"abc",所以不用在 pool 中再次创建"abc"了,而只是在堆里创建了一个新的
"abc"对象.s2 指向的是堆里的"abc".注意,虽然内容都是"abc",s 与 s2 表示的是
不同的对象
(4)接下来就很好说了.下面的三个==判断.(注意,==永远是判断内存地址是否相
等) s 与 s1,一个指向堆里的对象,一个指向 pool 里的.很明显是不同的对象.s 与
s2.上面说了,虽然都是指向堆里的对象,内容也是"abc",但是也不是相同的对象.
s1 与 s2.一个指向 pool,一个指向堆.也不是相同的对象.所以三个都返回 false.
2 第二个问题
[java] view plaincopy在CODE上查看代码片派生到我的代码片
  1. String s = new String("abc");  
  2. String s1 = "abc";  
  3. String s2 = new String("abc");  
  4. System.out.println(s == s1.intern());  
  5. System.out.println(s == s2.intern());  
  6. System.out.println(s1 == s2.intern());  
求最后输出是什么
解答.最后的答案是 false false true
intern()方法.按照 jdk 的帮助文档来说,是返回字符串对象的规范化表示形式。通
俗一点说,就是返回对应这个字符串内容的那个 pool 里的对象.这样说也许还看
不太明白,那可以拿具体例子来说
s1.intern().他的执行流程是,在 pool 里去查找 s1 对应的内容(也就是"abc").如果
找到,则返回 pool 里的对象.如果没有(老实说,我没想到有哪种情况是没有的),则
在 Pool 创建这个对象,并返回...
这样就很容易理解了.s1.intern 返回的是 pool 里的"abc"对象.与 s 这个堆里的对
象肯定不同,返回 false.同理,s 与 s2.intern()也肯定不同,返回 false.第三个,s1 与
s2.intern().其中 s2.intern()返回的是 pool 中的"abc"对象,而 s1 也是指向 pool 中
的"abc"对象.所以返回的是 true:
3 第三个问题
[java] view plaincopy在CODE上查看代码片派生到我的代码片
  1. String hello = "hello";  
  2. String hel = "hel";  
  3. String lo = "lo";  
  4. System.out.println(hello == "hel" + "lo");  
  5. System.out.println(hello == "hel" + lo);  
求输出的结果
解答 true false
首先,上面已经说明了,hello hel lo 这三个都是指向 pool 中的对象..
现在我们考虑"hel" + "lo" 按照内容来说,两个相加也就是"hello".这个时候,这个会
返回 pool 中的"hello"对象.所以,hello == "hel" + "lo" 返回的是 true .
而"hel" + lo 虽然内容也是"hello",但是它将在堆里面生成一个"hello"对象,并返回
这个对象...所以这里的结果是 false
总结一下就是,如果加号两边的是字面值(字面值就是直接在""里写的值,比如上面
的"hel"与"lo"),那么将在 pool 里查找有没对应内容的对象(这里的内容就是
"hello"),并返回 pool 里的对象.这和 hello 是一样的....
如果加号两边不满足上面的条件(比如,两边的值是引用值或者堆里的字符串对
象).那么将不会再 pool 里查找"hello",而是直接在堆里生成一个新的对象...





0 0
原创粉丝点击