目录
java.lang.Object 类是 Java 语言中的根类,即所有类的父类。它中描述的所有方法子类都可以使用。在对象实例化的时候,最终找的父类就是Object 。 如果一个类没有特别指定父类, 那么默认则继承自Object 类。例如:
public class MyClass /*extends Object*/ {
// ...
}
一、toString
1、方法摘要
public String toString() 返回该对象的字符串表示。toString 方法返回该对象的字符串表示,其实该字符串内容就是对象的类型 +@+ 内存地址值。 由于 toString 方法返回的结果是内存地址,而在开发中,经常需要按照对象的属性得到相应的字符串表现形式,因 此也需要重写它。
2、覆盖重写
如果不希望使用 toString 方法的默认行为(即重写前:打印的是包名类名@地址值),则可以对它进行覆盖重写。例如自定义的 Person 类:
public class Person {
private String name;
private int age;
@Override
public String toString() {
return "Person{" + "name='" + name + '\'' + ", age=" + age + '}';
}
// 省略构造器与Getter Setter
}
3、代码
在我们直接使用输出语句输出对象名的时候, 其实通过该对象调用了其 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;
}
}
重写前:打印的是包名类名@地址值
重写后:打印的是对象中的属性值(依重写后为准,大多是这种情况)
二、equals
1、方法摘要
public boolean equals(Object obj) 指示其他某个对象是否与此对象 “ 相等 ” 。调用成员方法 equals 并指定参数为另一个对象,则可以判断这两个对象是否是相同的。这里的 “ 相同 ” 有默认和自定义两种方式。
2、默认地址比较
如果没有覆盖重写 equals 方法,那么 Object 类中默认进行 == 运算符的对象地址比较,只要不是同一个对象,结果必然为false 。3、对象内容比较
如果希望进行对象的内容比较,即所有或指定的部分成员变量相同就判定两个对象相同,则可以覆盖重写 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
重写前:比较的是对象的地址值
重写和:比较的是对象的属性值(依重写后为准,大多是这种情况)
4、优化
Objects 工具类,它提供了一些方法来操作对象,它由一些静态的实用方法组成,这些方法是null-save(空指针安全的)或 null-tolerant (容忍空指针的,用于计算对象的 hashcode 、返回对象的字符串表示形式、比较两个对象。 在比较两个对象的时候, Object 的 equals 方法容易抛出空指针异常,而 Objects 类中的 equals 方法就优化了这个问题。方法如下:public static boolean equals(Object a, Object b) 判断两个对象是否相等。
源码:
public static boolean equals(Object a, Object b) {
return (a == b) || (a != null && a.equals(b));
}
5、代码
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;
}
}