String作为java常用的类,这里深入解析一下这个类的一下操作情况。
这里做的比较都是内存里的地址,因此我们认识几个感念。
栈(stack)存放基本数据类型的值与引用数据数据类型的引用地址。
堆(heap)存放引用数据类型的对象,即new的对象都放在这里。
方法区(Method Area)我们这里讲的stringpool(常量池)放在这里
equals方法和==的区别:
==用来比较基本数据类型的值是否相等,若是引用类型则判断引用的堆(heap)存储的地址是否相同。
equals通常用来比较基本数据类型的值或者引用类型的内容是否相同,比较还是基于他们在内存中的存放位置的地址值。
先来说说equals方法:
超类Object中有这个equals()方法,该方法主要用于比较两个对象是否相等。
public boolean equals(Object paramObject)
{
return this == paramObject;
}
由此可见Object比较的还是内存引用地址是否相同。
String类与其他类对equals进行了覆写
public boolean equals(Object anObject) {
if (this == anObject) {
return true;
}
if (anObject instanceof String) {
String anotherString = (String) anObject;
int n = value.length;
if (n == anotherString.value.length) {
char v1[] = value;
char v2[] = anotherString.value;
int i = 0;
while (n-- != 0) {
if (v1[i] != v2[i])
return false;
i++;
}
return true;
}
}
return false;
}
如果引用地址相同返回true,否则判断字符串是否相同。
下面主要说说==方法
这里就要用到上面说的内存里面的地址了,先看几个例子
String a = new String("张三");//堆内存中会产生一个“张三”的对象,栈中是a的引用对象
String b = new String("张三");
String n = "张三";//先判断常量池中是否有“张三”这对象,没有就new一个,栈中放的是n的对象
String m = "张三";
System.out.println(a==b);//false
System.out.println(n==m);//true
System.out.println(a==n);//false
String的+操作
当+的两边都是字面值常量(引用常量池的对象不算)时,+完后会先判断常量池中是否有这个字符串,没有再new。
如果+两边有一个不是字符串,则直接是在堆内存中new一个对象。
String s = "hello world";
String x = "hello "+"world";
String world ="world";
String y = "hello "+world;
System.out.println(s==x);//true 在常量池中找
System.out.println(s==y);//false 直接在堆中new
String的concat方法拼接字符串
public String concat(String str) {
int otherLen = str.length();
if (otherLen == 0) {
return this;
}
int len = value.length;
char buf[] = Arrays.copyOf(value, len + otherLen);
str.getChars(buf, len);
return new String(buf, true);
}
这里是将拼接的字符串在堆中new了一个出来。
String对象的intern()方法返回的是常量池的引用地址,没有则创建一个返回
String s = new String("abc");
String s1 = "abc";
String s2 = new String("abc");
System.out.println(s == s.intern());//false s.intern()在常量池中创建一个abc并返回
System.out.println(s1 == s1.intern());//true s1引用的本身就是常量地址
System.out.println(s.intern() == s2.intern());//true s与s2引用的常量地址相同
System.out.println(s2 == s2.intern());//false s2.intern()在常量池中创建一个abc并返回