字符串和字符串池“==”和“equals()”
上次面试题考到一道String字符串比较的,虽然对String了解比较多,用得也比较多,但感觉还是有很多深入的东西没有彻底理解。比如字符串池。先拿自己运行的代码给大家看看再分析。String a = "ab";String b = "a"+"b";//这里的"a"、"b"不是变量,是常量System.out.println(a==b);//输出结果为true String str1 = "java"; //在编译时就知道对象放在常量池里String str2 = "blog";String str = str1 + str2;//str1、str2是变量在运行时刻才能知道,也就是说str1+str2是在堆里创建的System.out.println(str);//输出结果为javablogSystem.out.println(str=="javablog");//输出结果为falseString a = "Hello";String b = "Hello";String c = new String("Hello");String d = new String("Hello");System.out.println(a == b);System.out.println(b == c);System.out.println(c == d);System.out.println(a.equals(b));System.out.println(b.equals(c));System.out.println(c.equals(d));
结果是true,false,false,true,true,true。这里就涉及到字符串池。“字符串池”,是Java为了提高内存利用率而采用的措施:
当遇到String a = "Hello"; 这样的语句时,Java会先在字符串池中寻找是否已经存在"Hello"这个字符串,如果没有,则建立字符串"Hello"对象,然后把它加入到字符串池中,再将它的引用返回给变量 a ,a指向这个地址;然后遇到语句String b = "Hello",这时字符串池中已经有 "Hello"了,所以直接让变量b也指向这个地址,省去了重新分配的麻烦。而在Java中,操作符“==”对于两个基本型来说,是判断其内容是否相同,对于两个对象来说,则是判断其地址是否相同,所以a == b返回 true。
那么String c = new String("Hello")又如何处理呢?如果是这种写法,则不会去访问字符串池,而是先为变量 c 开辟空间,然后将值写入空间。所以a == c返回false,c == d同样返回false。
至于String的equals方法,因为它比较的不是对象的地址,而是对象的值,所以都返回 true就不奇怪了。
--------------------------------------------------------------------
Java虚拟机有一个字符串池,保存着几乎所有的字符串对象。字符串表达式总是指向字符串池中的一个对象。使用new操作创建的字符串对象不指向字符串池中的对象但是可以使用intern方法(String 类的 intern方法:public native String intern();)使其指向字符串池中的对象.(注:这是一个本地方法。在调用这个方法时,JAVA虚拟机首先检查字符串池中是否已经存在与该对象值相等对象存在——使用equals方法确定,如果有则返回字符串池中对象的引用;如果没有,则先在字符串池中创建一个相同值的String对象,然后再将它的引用返回)。池中两个相等的字符串如果使用“==”来比较将返回真。
-----------------------------------------------------------------------
例5:
String str1="java";
String str2="blog";
String s=str1+str2;
System.out.println(s=="javablog");
结果是false。Jvm确实对型如String str1="java";的String对象放在常量池里,但是它是在编译时那么做的,而String s=str1+str2;是在运行时刻才能知道,也就是说str1+str2是在堆里创建的,所以结果为false了。
比较两个已经存在于字符串池中字符串对象可以用"=="进行,拥有比equals操作符更快的速度
页:
[1]