==和equals方法:
Java程序中判断两个变量是否相等有两种方式:
一、利用 == 运算符:
1.1、如果两个变量是基本类型变量,且都是数值型(不一定要求数值类型完全相同),则只要两个变量的值相同,就返回true
1.2、对于两个引用类型变量,只有它们指向同一个对象时,== 判断才会返回true
1.3、== 运算符不可用于比较类型上没有父子关系的对象
代码示例:
import static java.lang.System.*;
public class EqualTest{
public static void main(String[] args){
int it=65;
float f1=65.0f;
out.println("65和65.0f是否相等?"+(it==f1));
char ch='A';
out.println("65和A是否相等?"+(it==ch));
String str1=new String("hello");
String str2=new String("hello");
out.println("str1和str2是否相等?"+(str1==str2));
out.println("str1是否equals str2?"+(str1.equals(str2)));
//-由于java.lang.String与EqualTest类没有继承关系,下面代码会编译错误
//out.println("hello"==new EqualTest());
}
}
运行结果:

知识点拓展:
1、String str=“hello”;
2、String str=new String("hello");
如上两行代码中的两个变量str的区别:
1行代码中的“hello”被称为 字符串直接量(在编译时就计算出来的字符串值),JVM通过 常量池 来管理这些字符串
2行中的 new String("hello"),当使用这种定义方式时,JVM会先使用常量池来管理“hello”直接量,再调用String类的构造器来创建一个新的String对象,新创建的String对象被保存在堆内存中,
换句话说,new String("hello") 一共产生了两个字符串对象
常量池(constant pool):专门用于管理在编译时被确定并被保存在已编译的.class文件中的一些数据,包括了关于类、方法、接口中的常量,还包括字符串常量,如下代码示例:
import static java.lang.System.*;
public class CompareTest{
public static void main(String[] args){
//-a直接引用常量池中的"张三李四"
String a="张三李四";
String b="张三";
String c="李四";
//-d后面的字符串值在编译时就确定下来了
//-d直接引用常量池中的"张三李四"
String d="张三"+"李四";
//-e后面的字符串值在编译时就确定下来了
//-e直接引用常量池中的"张三李四"
String e="张"+"三"+"李四";
//-f后面的字符串值不能在编译时就确定下来了
//-f不能直接引用常量池中的字符串
String f=b+c;
//-使用new调用构造器会重新创建一个新的String对象
//-g引用堆内存中新创建的String对象
String g=new String("张三李四");
out.println(a==d);//-true
out.println(a==e);//-true
out.println(a==f);//-false
out.println(a==g);//-false
}
}
总结:
1、JVM常量池保证相同的字符串直接量只有一个,不会产生多个副本
2、例如上面代码中a、d、e所引用的字符串可以在编译期就确定下来,因此他们都将引用常量池中同一个字符串对象,所以用 == 运算符计算是返回 true 的
3、使用 new String()创建的字符串对象是运行时创建出来的,它被保存在运行时内存区内(即堆内存内),不会放入常量池中
二、利用 equals()方法:
1、equals()方法是Object类提供的一个实例方法,所有引用变量都可以引用这个方法判断与其它引用变量是否相等
2、判断是否相等的标准与 == 没有区别,都是看引用变量是否指向相同的对象,从这个作用上来说equals()方法没有太大意义
3、equals()方法最大的意义是可通过重写该方法,实现自定义的相等判断,如:String类就重写了equals()方法:只要两个字符串所包含的字符序列相等,即判断相等,返回true
重写equals()方法,实现两个类相等:
class Person{
//-重写Object类的equals()方法
public boolean equals(Object o){
//-相等规则自己定义
return true;
}
}
class Dog{
//-空类
}
public class OverrideEquals{
public static void main(String[] args){
Person p=new Person();
System.out.println(p.equals(new Dog()));
System.out.println(p.equals(new String("张三")));
}
}
运行结果:

有意义的去判断两个类相等。代码示例:
class Penson{
private String name;
private String id;
public Penson(){
}
public Penson(String name,String id){
this.name=name;
this.id=id;
}
public void setName(String name){
this.name=name;
}
public String getName(){
return this.name;
}
public void setId(String id){
this.id=id;
}
public String getId(){
return this.id;
}
public boolean equals(Object obj){
if(this==obj){
return true;
}
//-obj.getClass()==Penson.class用到了反射基础,暂时不用深究
if(obj!=null&&obj.getClass()==Penson.class){
//-强制转换,把将要比较的参数转换成Penson类,便于后面获取类变量id
Penson penson=(Penson)obj;
//-只有两个对象的id相等,才判定相等
if(this.getId().equals(penson.getId())){
return true;
}
}
return false;
}
}
public class OverrideEqualsRight{
public static void main(String[] args){
Penson p1=new Penson("张三","9527");
Penson p2=new Penson("李四","9527");
Penson p3=new Penson("王五","9526");
//-p1与p2的id相等,返回true
System.out.println(p1.equals(p2));
//-p1与p2的id不等,返回false
System.out.println(p1.equals(p3));
}
}
运行结果:

通常而言,重写equals()方法应满足下面的条件:
1、自反性:对任意x,x.equals(x)一定返回true
2、对称性:对任意x、y,如果x.equals(y)返回true,那么y.equals(x)一定返回true
3、传递性:对任意x、y、z,如果x.equals(y)返回true,y.equals(z)返回true,那么x.equals(z)一定返回true
4、一致性:对任意x、y,如果对象中用于等价比较的信息值没有发生改变,那么无论调用x.equals(y)多少次,返回的结果应该始终保持一致,要么一直true,要么一直false
5、对任何不是null的x,x.equals(null)一定返回false
浙公网安备 33010602011771号