最近看javaSe基础,被String类的equals方法和“==”搞懵逼了,类似的题目总是搞不到,觉得好好花点时间把这个问题搞清楚,写下自己的心得。如有错误,还望各位指正!
一、String类生成对象的内存问题。
字符串在内存通常分配在内存的两个位置,一个是常量区,一个是在堆内存中。
第一种形式: String str ="abc";
这句话是在字符串常量区创建一个“abc”的对象,str为这个对象的引用。如果再来一句String str2 =“abc”,这是JVM并不是重新又创建一个新的对象,而是先在字符串常量池中看有没有这个对象,若有,将str2指向字符串“abc”;若没有才会重新创建。我们可以通过System.out.println(str1==str2)来判断str1和str2是指向同一个引用。
类似String s ="abc"+“edf”也经常碰见。s 也是在字符串常量池中。JVM会先在字符串常量池中查找字符串“abc”和“edf”,没有则重新创建,有则直接引用,然后会在字符串常量池中生成新的字符串“abcedf”;
第二种形式:String s =new String("abc");
这种形式是利用String类的构造器方法在堆内存中创建对象。JVM 首先会在字符串常量池中找“abc",有则直接引用,没有则创建。然后将该对象复制到堆内存中。s指向该对象的内存。由于堆内存中的每个对象内存地址不同,所以String s1 =new String("abc")
和String s2=new String("abc")是两个不同的对象,我们用System.out.println(s1==s2)可以判断它们是两个不同的对象。
二、equals方法和== 的区别。
“==”是直接比较两个对象的地址。我们上面判断字符串是否是同一个对象就是用的“==”方法。equals方法是Object对象的方法,我们可以看下Object类中equals方法的源码:
public boolean equals(Object obj) {
return (this == obj);
}
我们可以看到,它其实就是调用了“==”,Object对象的equals方法就是比较两个对象的内存地址,相同返回true,否则返回false。而String类是Object类的之类,但是String类继承Object类之后重写了它的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;
}
我们可以看到,String类也是先比较两个对象的地址,地址相同则直接返回true,如果不相同,则将传入的对象转换成字符串,再开始进行内容的比较,所以String类的equals方法我们可以认为它是比较字符串的内容的。同理我们可以发现很多类都重写了Object类的equals方法,那么这时候的equals方法可能就不是比较对象的地址了。
总结:String类的对象一般存储在字符串常量池和堆内存中,其中用new关键字的创建的对象在堆内存中。“==”是比较两个对象的内存地址,Object类对象的equals方法就是比较对象的内存地址。由于Object类为一切类的父类,所以如果类没有重写equals方法,那么相当于它是调用的Object类的equals方法,比较的是对象的内存地址。重写了Object类的equals方法的类,该方法比较的内容就要看具体的实现了。