java==和=_Java语法----Java中equals和==的区别

本文详细解析了Java中基本类型和引用类型下`==`和`equals()`的区别,重点讲解了String类equals方法的实现,并通过实例展示了如何在不同场景下正确使用。了解字符串比较中地址值与内容比较的技巧,避免混淆。

【声明】

欢迎转载,但请保留文章原始出处→_→

【正文】

平时在学Android和Java语言的时候,总是碰到“equals”和“==”这两个字符,老感觉差不多;其实还是有一些区别的,今天干脆把它们彻底弄清楚。

一、java当中的数据类型和“==”的含义:

基本数据类型(也称原始数据类型) :byte,short,char,int,long,float,double,boolean。他们之间的比较,应用双等号(==),比较的是他们的值。

引用数据类型:当他们用(==)进行比较的时候,比较的是他们在内存中的存放地址(确切的说,是堆内存地址)。

注:对于第二种类型,除非是同一个new出来的对象,他们的比较后的结果为true,否则比较后结果为false。因为每new一次,都会重新开辟堆内存空间。

二、equals()方法介绍:

JAVA当中所有的类都是继承于Object这个超类的,在Object类中定义了一个equals的方法,equals的源码是这样写的:

public booleanequals(Object obj) {//this - s1//obj - s2

return (this ==obj);

}

可以看到,这个方法的初始默认行为是比较对象的内存地址值,一般来说,意义不大。所以,在一些类库当中这个方法被重写了,如String、Integer、Date。在这些类当中equals有其自身的实现(一般都是用来比较对象的成员变量值是否相同),而不再是比较类在堆内存中的存放地址了。

所以说,对于复合数据类型之间进行equals比较,在没有覆写equals方法的情况下,他们之间的比较还是内存中的存放位置的地址值,跟双等号(==)的结果相同;如果被复写,按照复写的要求来。

我们对上面的两段内容做个总结吧:

== 的作用:

基本类型:比较的就是值是否相同

引用类型:比较的就是地址值是否相同

equals 的作用:

引用类型:默认情况下,比较的是地址值。

注:不过,我们可以根据情况自己重写该方法。一般重写都是自动生成,比较对象的成员变量值是否相同

三、String类的equals()方法:

现在我们拿String类来举例:

我们去\src\java\lang目录中找到String类,发现equals方法被重写如下:

1 public booleanequals(Object anObject) {2 if (this ==anObject) {3 return true;4 }5 if (anObject instanceofString) {6 String anotherString =(String)anObject;7 int n =value.length;8 if (n ==anotherString.value.length) {9 char v1[] =value;10 char v2[] =anotherString.value;11 int i = 0;12 while (n-- != 0) {13 if (v1[i] !=v2[i])14 return false;15 i++;16 }17 return true;18 }19 }20 return false;21 }

上述代码可以看出,String类中被复写的equals()方法其实是比较两个字符串的内容。下面我们通过实际代码来看看String类的比较。

1、举例代码如下:

1 public classStringDemo {2 public static voidmain(String[] args) {3 String s1 = "Hello";4 String s2 = "Hello";5 System.out.println(s1 == s2); //true

6 }7 }

上方代码中,用“==”比较s1和s2,返回的结果是true。

2、稍微改动一下程序,会有奇怪的发现:

1 public classStringDemo {2 public static voidmain(String args[]) {3 String str1 = "Hello";4 String str2 = new String("Hello");5 String str3 = str2; //引用传递

6 System.out.println(str1 == str2); //false

7 System.out.println(str1 == str3); //false

8 System.out.println(str2 == str3); //true

9 System.out.println(str1.equals(str2)); //true

10 System.out.println(str1.equals(str3)); //true

11 System.out.println(str2.equals(str3)); //true

12 }13 }

上方第4行代码中,我们new了一个对象,用“==”比较s1和s2,返回的结果却是false;而用用“equals”比较s1和s2,返回的结果是true。

为了分析上面的代码,我们必须首先分析堆内存空间和栈内存空间,这一点非常重要:

95218f6eede1a06c84516dcda50ee6bf.png

看完上面的图,再结合上面的代码,就一目了然了。现在我们可以给自己出一道面试题:

面试题:请解释字符串比较之中“==”和equals()的区别?

==:比较的是两个字符串内存地址(堆内存)的数值是否相等,属于数值比较;

equals():比较的是两个字符串的内容,属于内容比较。

以后进行字符串相等判断的时候都使用equals()。

3、再次更改程序:

1 public classObjectDemo{2 public static voidmain(String[] args) {3 String s1 = "Hello";4 String s2 = new String("Hello");5 s2 =s2.intern();6 System.out.println(s1 == s2); //true

7 System.out.println(s1.equals(s2)); //true

8 }9 }

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

四、比较两个对象的值:

代码如下:

1 packagecom.smyh;2

3 public classObjectDemo {4 public static voidmain(String args[]){5 Student student1 = new Student("生命壹号",22,"成都");6 Student student2 = new Student("生命壹号",22,"成都");7 System.out.println(student1==student2);8 System.out.println(student1.equals(student2));9 }10 }11 classStudent {12 privateString name;13 private intage;14 privateString address;15 public Student(String name,intage,String address){16 this.name =name;17 this.age =age;18 this.address =address;19 }20 //重写Object类中的equals方法(比较两个对象的值是否相等)

21 public booleanequals(Object obj){22 //为了提高效率:如果两个内存地址相等,那么一定是指向同一个对内存中的对象,就无需比较两个对象的属性值(自己跟自己比,没啥意义嘛)

23 if(this==obj){24 return true;25 }26

27 //为了提供程序的健壮性28 //我先判断一下,obj是不是学生的一个对象,如果是,再做向下转型,如果不是,直接返回false。29 //这个时候,我们要判断的是对象是否是某个类的对象?30 //记住一个格式:对象名 instanceof 类名。表示:判断该对象是否是该类的一个对象

31 if(!(obj instanceofStudent)){32 return false;33 }34

35 //如果是就继续

36 Student s = (Student)obj;//强制转换,即向下转型(毕竟Object类型没有具体的对象属性)

37 return this.name.equals(s.name) && this.age == s.age && this.address.equals(s.address);//判断两个对象的属性值是否相等

38 }39 }

上述代码中,首先判断传递进来的对象与当前对象的地址是否相等,如果相等,则肯定是同一个堆内存中的对象。因为传递进来的参数是Object类型,所以任何对象都可以接收。一旦接收进来,就将Object类型的对象向下转型,然后判断具体的属性值。

运行效果:

44b7c893b39d0c5a4165905ba65c6ec4.png

其实,如果是在Eclipse中做开发的话,上面重写的equals()方法其实是可以自动生成的:

fe96d4ce015cba0a7319c4e1473a9abe.png

自动生成后的equals()方法如下:

1 @Override2 public booleanequals(Object obj) {3 if (this ==obj)4 return true;5 if (obj == null)6 return false;7 if (getClass() !=obj.getClass())8 return false;9 Student other =(Student) obj;10 if (address == null) {11 if (other.address != null)12 return false;13 } else if (!address.equals(other.address))14 return false;15 if (age !=other.age)16 return false;17 if (name == null) {18 if (other.name != null)19 return false;20 } else if (!name.equals(other.name))21 return false;22 return true;23 }

可以看到:我们自己重写的equals()方法和自动生成的equals()方法,二者效果是一样的。

【参考】

参考博客:

以及李兴华老师的视频讲解。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值