手记

equals与==的区别

Java中的数据类型,可分为两类:

1.基本数据类型,也称原始数据类型:

byte、short、char、int、long、float、double、boolean它们之间的比较,应用双等号(==)比较的是他们的值。

2.引用数据类型:

JAVA当中所有的类都是继承于Object这个基类的,在Object中的基类中定义了一个equals的方法,这个方法的初始行为是比较变量(栈)内存中存放的对象的(堆)内存地址,用来判断两个对象的地址是否相同,即是否是指相同一个对象。比较的是真正意义上的指针操作。
但在一些类库当中这个方法被覆盖掉了,如String、Integer、Date。在这些类当中equals有其自身的实现,而不再是比较类在堆内存中的存放地址了。
对于引用数据类型之间进行equals比较,在没有重写equals方法的情况下,他们之间的比较还是基于他们在内存中的存放位置的地址值的,因为Object的equals方法也是用双等号进行比较的,所以比较后的结果跟双等号的结果相同。

注意:
1、比较的是操作符两端的操作数是否是同一个对象。
2、两边的操作数必须是同一类型的(可以是父子类之间)才能编译通过。
3、比较的是地址。如果是具体的阿拉伯数字的比较,值相等则为true,如:
int a=10 与 long b=10L 与 double c=10.0都是相同的(为true),因为他们都指向地址为10的堆。
4、String s=“abc"是一种非常特殊的形式,和new 有本质的区别。它是java中唯一不需要new就可以产生对象的途径。以 String s=“abc”; 形式赋值在java中叫直接量,它是在常量池中而不是象new一样放在压缩堆中。这种形式的字符串,在JVM内部发生字符串拘留,即当声明这样的一个字符串后,JVM会在常量池中先查找有有没有一个值为"abc"的对象。如果有,就会把它赋给当前引用。即原来那个引用和现在这个引用指向了同一对象。如果没有,则在常量池中新创建一个"abc”,下一次如果有 String s1 = “abc”; 又会将s1指向"abc"这个对象,即以这种形式声明的字符串,只要值相等,任何多个引用都指向同一对象。
  而 String s = new String(“abcd”); 和其它任何对象一样,每调用一次就产生一个对象,只要它们调用。
  也可以这么理解: String str = “hello”; 先在内存中找是不是有"hello"这个对象,如果有,就让str指向那个"hello"。如果内存里没有"hello",就创建一个新的对象保存"hello"。String str=new String (“hello”) 就是不管内存里是不是已经有"hello"这个对象,都新建一个对象保存"hello"。

具体可以看下面的代码:

public class test1 { 
    public static void main(String[] args) { 
         String a = new String("ab"); // a 为一个引用     
         String b = new String("ab"); // b为另一个引用,对象的内容一样
         String aa = "ab"; // 放在常量池中     
         String bb = "ab"; // 从常量池中查找   
         if (aa == bb){ 
            System.out.println("aa==bb");// true
         }       
         if (a == b){             
            System.out.println("a==b");// false,非同一对象 
         }      
        if (a.equals(b)){          
            System.out.println("aEQb");// true 
         }         
        if (42 == 42.0) {            
             System.out.println("true");15 // true   
        } 
     }
}

equals和==的区别:

equals方法最初是在所有类的基类Object中进行定义的,源码是

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

由equals的源码可以看出这里定义的equals与双等号是等效的(Object类中的equals与双等号没什么区别)。不同的原因就在于有些类(像String、Integer等类)对equals进行了重写,但是没有对equals进行重写的类(比如我们自己写的类)就只能从Object类中继承equals方法,其equals方法与==就也是等效的,除非我们在此类中重写equals。

对equals重新需要注意五点:

1、自反性:对任意引用值X,x.equals(x)的返回值一定为true;
2 、对称性:对于任何引用值x,y,当且仅当y.equals(x)返回值为true时,x.equals(y)的返回值一定为true;
3、传递性:如果x.equals(y)=true,y.equals(z)=true,则x.equals(z)=true ;
4、一致性:如果参与比较的对象没任何改变,则对象比较的结果也不应该有任何改变;
5、非空性:任何非空的引用值X,x.equals(null)的返回值一定为false 。

String类对equals的重写如下:

public boolean equals(Object var1) {
		if (this == var1) {
			return true;
		} else {
			if (var1 instanceof String) {
				String var2 = (String) var1;
				int var3 = this.value.length;
				if (var3 == var2.value.length) {
					char[] var4 = this.value;
					char[] var5 = var2.value;

					for (int var6 = 0; var3-- != 0; ++var6) {
						if (var4[var6] != var5[var6]) {
							return false;
						}
					}
					return true;
				}
			}
			return false;
		}
	}

另外,双等号比"equals"运行速度快,因为双等号"=="只是比较引用。

1人推荐
随时随地看视频
慕课网APP