【初学者常见问题】深入理解“==”和equals()

首先感谢老大在这一方面对我的指引。http://v.youku.com/v_show/id_XNjAzNTczMTI4.html

我将把我对于"=="和equals()体会和实例一起结合让菜鸟们能够深入了解(当然我也是菜鸟,嘻嘻)

长久以来,我看淡很多面试题都有下面类似的题目出现,如:

String s=new String(“hello”);
String t =new String(“hello”);
char c [ ] ={„h‟,‟e‟,‟l‟,‟l‟,‟o‟};
下列哪些表达式返回 true ?
A.s.equals(t);
B.t.equals(c);
C.s= =t ;
D.t.equals (new String(“hello”));
E.t= = c;

开始面临这些题目的时候很蛋疼,脑壳转不过来的话很容易出错,但是我们在网上百度“==”和equals()方法的区别时,网上会有这样的解释

“==”测试引用同一地址     而equals()测试值是否相等,除了是比较两个是否为同一对象时,否则都应该用equals()方法 

这些回答是好,可是再次面临这些问题的时候还是会心有余悸,所以今儿就刨根问底的把这个问题彻底解决一遍。

先刨Object类这个“祖坟”,因为他是所有类的一个基类,自然要先干它了。  打开源码:java.lang.Object

  public boolean equals(Object obj) {
        return (this == obj);
    }


例1:猜打印结果

public class Test {
	public static void main(String[] args) {
		Person p1 = new Person("老张", 10);
		Person p2 = new Person("老张",10);
		System.out.println(p1.equals(p2));
	}
}
class Person{
	private String name;
	private int age;
	public Person(String name, int age) {
		this.name = name;
		this.age = age;
	}
}
发现在底层中,Object类是用的是==实现了equals方法,这就解释了上面的代码为什么是false的结果了,因为Person这个类默认继承了Object这个基类,所以有它的equals方法,而比较是根据“==”来返回结果的,p1和p2是两个对象,所以他们返回的是false .

那么怎么比较实现因为姓名相同或者年龄相同返回的是true呢 ?  下面我们重写equals方法,代码如下:

public class Test {
	public static void main(String[] args) {
		Person p1 = new Person("老张", 10);
		Person p2 = new Person("老张", 10);
		System.out.println(p1.equals(p2));
	}
}
class Person {
	private String name;
	private int age;

	public Person(String name, int age) {
		this.name = name;
		this.age = age;
	}
	@Override
	public boolean equals(Object o){
		if(o instanceof Person){
			Person person1 = this;
			Person person2 = (Person)o;
			boolean temp = (person1.getName()==person2.getName())
					&&(person1.getAge()==person2.getAge());
			return temp;
		}
		return false;
	}
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	public int getAge() {
		return age;
	}
	public void setAge(int age) {
		this.age = age;
	}
}

重写了equalls方法后,就根据作者想要的比较两个姓名和年龄都相等的规则进行比较,这也是为什么equals方法会在Object这个基类存在的原因和初衷了吧

对于网上说的equals方法比较内容,这里我也看了一下源码,把String,Integer实现了equals方法的相关源码贴上来(我想其他基本类型的变量应该都重写了equals方法了,大家可以看看源码查询,这里我就不做太多阐述了),供大家参考

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;
    }

Integer类的equals方法重写

    public boolean equals(Object obj) {
        if (obj instanceof Integer) {
            return value == ((Integer)obj).intValue();
        }
        return false;
    }

所以下面的例2大家就知道答案了:

public class Test {
	public static void main(String[] args) {
		String str1 = new String("csdn");
		String str2 = new String("csdn");
		System.out.println(str1.equals(str2));
		Integer num1 = new Integer(1);
		Integer num2 = new Integer(1);
		System.out.println(num1.equals(num2));
	}
}

结果如大家所料,都是返回的是true

另外,下面的代码让我前几天就不得其解:

public class Test {
	public static void main(String[] args) {
		String str1 = "csdn";
		String str2 = "csdn";
		System.out.println(str1==str2);
 	}
}

返回的结果是true,我在这里做一个简单的叙述,==是测试引用的是否是同一地址,这里我们用eclipse的dubug功能调试,结果发现我们创建的两个变量str1和str2拥有的是同一个ID,就说明他们是指向同一个地址,所以返回的是true .....详细的解释请参见: http://blog.csdn.net/withiter/article/details/6699110

希望对初学者有所帮助,多看源码,多实践。




 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
String a="hello world"; //在java中有一个常量池,当创建String 类型的引用变量给它赋值时,java会到它的常量池中找"hello world"是不是在常量池中已存在。如果已经存在则返回这个常量池中的"hello world"的地址(在java中叫引用)给变量a 。注意a并不是一个对象,而是一个引用类型的变量。它里面存的实际上是一个地址值,而这个值是指向一个字符串对象的。在程序中凡是以"hello world"这种常量似的形式给出的都被放在常量池中。 String b=new String("hello world"); //这种用new关键字定义的字符串,是在堆中分配空间的。而分配空间就是由new去完成的,由new去决定分配多大空间,并对空间初始化为字符串"hello world" 返回其在堆上的地址。 通过上面的原理,可以做如下实验: String a="hello world"; String b="hello world"; String c=new String("hello world"); String d=new String("hello world"); if(a==b) System.out.println("a==b"); else System.out.println("a!=b"); if(c==d) System.out.println("c==d"); else System.out.println("c!=d"); //输出结果: a==b c!=d 为什么会出现上面的情况呢? String a="hello world"; String b="hello world"; 通过上面的讲解可以知道,a和b都是指向常量池的同一个常量字符串"hello world"的,因此它们返回的地址是相同的。a和b都是引用类型,相当于c语言里面的指针。java里面没有指针的概念,但是实际上引用变量里面放的确实是地址值,只是java为了安全不允许我们对想c语言中的那样对指针进行操作(如++ 、--)等。这样就有效的防止了指针在内存中的游离。 而对于 String c=new String("hello world"); String d=new String("hello world"); 来说是不相等的,他们是有new在堆中开辟了两块内存空间,返回的地址当然是不相等的了。如果我们要比较这两个字符串的内容怎么办呢?可以用下面的语句: if(c.equals(d)) System.out.println("c==d"); else System.out.println("c!=d"); //输出 c==d

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值