目录
一、toString
1、方法摘要
2、覆盖重写
3、代码
二、equals
1、方法摘要
2、默认地址比较
3、对象内容比较
4、优化
5、代码
public class MyClass /*extends Object*/ { // ... }
public String toString() 返回该对象的字符串表示。toString 方法返回该对象的字符串表示,其实该字符串内容就是对象的类型 +@+ 内存地址值。 由于 toString 方法返回的结果是内存地址,而在开发中,经常需要按照对象的属性得到相应的字符串表现形式,因 此也需要重写它。
如果不希望使用 toString 方法的默认行为(即重写前:打印的是包名类名@地址值),则可以对它进行覆盖重写。例如自定义的 Person 类:
public class Person { private String name; private int age; @Override public String toString() { return "Person{" + "name='" + name + '\'' + ", age=" + age + '}'; } // 省略构造器与Getter Setter }
在我们直接使用输出语句输出对象名的时候, 其实通过该对象调用了其 toString() 方法。
/* java.lang.Object 类 Object 是类层次结构的根(父)类。 每个类(Person,Student...)都使用 Object 作为超(父)类。 所有对象(包括数组)都实现这个类的方法。 */ import java.util.ArrayList; import java.util.Random; import java.util.Scanner; public class Demo01ToString { public static void main(String[] args) { /* Person类默认继承了Object类,所以可以使用Object类中的toString方法 String toString() 返回该对象的字符串表示。 */ Person p = new Person("Specter", 18); String s = p.toString(); System.out.println(s); // API.Demo01.object.Person@75412c2f | abc | Person{name='Specter', age=18} //直接打印对象的名字,其实就是调用对象的toString p=p.toString(); System.out.println(p); // API.Demo01.object.Person@75412c2f | abc | Person{name='Specter', age=18} /* 看一个类是否重写了toString,直接打印这个类的对象即可, 如果没有重写toString方法那么打印的是对象的地址值 如果重写toString方法,那么就按照重写的方式打印 */ Random r = new Random(); System.out.println(r); // java.util.Random@13b6d03 没有重写toString方法 Scanner sc = new Scanner(System.in); System.out.println(sc); // ava.util.Scanner[delimiters=\p{javaWhitespace}…… 重写toString方法 ArrayList<Integer> list = new ArrayList<>(); list.add(1); list.add(2); list.add(3); System.out.println(list);//[1, 2, 3] 重写toString方法 } }
直接打印对象的地址值没有意义,需要重写Object类中的toString方法,打印对象的属性(name,age)
public class Person { private String name; private int age; public Person() { } public Person(String name, int age) { this.name = name; this.age = age; } /* 直接打印对象的地址值没有意义,需要重写Object类中的toString方法 打印对象的属性(name,age) */ /*@Override public String toString() { // return "abc"; return "Person{" + "name='" + name + '\'' + ", age=" + age + '}'; }*/ 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; } }
重写前:打印的是包名类名@地址值
重写后:打印的是对象中的属性值(依重写后为准,大多是这种情况)
public boolean equals(Object obj) 指示其他某个对象是否与此对象 “ 相等 ” 。调用成员方法 equals 并指定参数为另一个对象,则可以判断这两个对象是否是相同的。这里的 “ 相同 ” 有默认和自定义两种方式。
import java.util.Objects; public class Person { private String name; private int age; @Override public boolean equals(Object o) { // 如果对象地址一样,则认为相同 if (this == o) return true; // 如果参数为空,或者类型信息不一样,则认为不同 if (o == null || getClass() != o.getClass()) return false; // 转换为当前类型 Person person = (Person) o; // 要求基本类型相等,并且将引用类型交给java.util.Objects类的equals静态方法取用结果 return age == person.age && Objects.equals(name, person.name); } }
Person类默认继承了Object类,所以可以使用Object类的equals方法 boolean equals(Object obj) 指示其他某个对象是否与此对象“相等”。 equals方法源码: public boolean equals(Object obj) { return (this == obj); } 参数: Object obj:可以传递任意的对象 == 比较运算符,返回的是一个布尔值 true false 基本数据类型:比较的是值 引用数据类型:比价的是两个对象的地址值 this是谁?那个对象调用的方法,方法中的this就是那个对象;p1调用的equals方法所以this就是p1 obj是谁?传递过来的参数p2 this==obj -->p1==p2
重写前:比较的是对象的地址值
重写和:比较的是对象的属性值(依重写后为准,大多是这种情况)
public static boolean equals(Object a, Object b) 判断两个对象是否相等。
源码:
public static boolean equals(Object a, Object b) { return (a == b) || (a != null && a.equals(b)); }
import java.util.Objects; public class Person { private String name; private int age; public Person() { } public Person(String name, int age) { this.name = name; this.age = age; } /* Object类的equals方法,默认比较的是两个对象的地址值,没有意义 所以我们要重写equals方法,比较两个对象的属性(name,age) 问题: 隐含着一个多态 多态的弊端:无法使用子类特有的内容(属性和方法) Object obj = p2 = new Person("KCaNaMgAl",18); 解决:可以使用向下转型(强转)把Object类型转换为Person */ /* @Override public boolean equals(Object obj) { //增加一个判断,传递的参数obj如果是this本身,直接返回true,提高程序的效率 if(obj==this){ return true; } //增加一个判断,传递的参数obj如果是null,直接返回false,提高程序的效率 if(obj==null){ return false; } //增加一个判断,防止类型转换一次ClassCastException if(obj instanceof Person){ //使用向下转型,把obj转换为Person类型 Person p = (Person)obj; //比较两个对象的属性,一个对象是this(p1),一个对象是p(obj->p2) boolean b = this.name.equals(p.name) && this.age==p.age; //这里的是字符串重写的equals return b; } //不是Person类型直接返回false return false; }*/ @Override public boolean equals(Object o) { if (this == o) return true; /* getClass() != o.getClass() 使用反射技术,判断o是否是Person类型 等效于 obj instanceof Person */ if (o == null || getClass() != o.getClass()) return false; Person person = (Person) o; return age == person.age && Objects.equals(name, person.name); } @Override public int hashCode() { return Objects.hash(name, age); } 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; } }