Object类结构的剖析
java.lang.Object类
1.Object类是所有Java类的根父类
2.如果在类的声明中未使用extends关键字指明其父类,则默认父类为java.lang.Object类
3.Object类中的功能(属性、方法)就具有通用性。
属性:无
方法:equals() / toString() / getClass() /hashCode() 返回当前对象的一个哈希值/ clone()克隆 / finalize()
wait() 、 notify()、notifyAll() //线程时讲
4. Object类只声明了一个空参的构造器(意味着所以其他类的对象最好都会调到空参构造器Object() )
Object类是所有Java类的根父类和getClass()方法
public class ObjectTest {
public static void main(String[] args) {
Order order = new Order();
System.out.println(order.getClass());//获得这个对象是什么类
System.out.println(order.getClass().getSuperclass());//获得这个对象的父类是什么类
}
}
class Order{
}
结果:
Object类的clone()的使用
Object类的clone()的使用
克隆一个对象
//Object类的clone()的使用
public class CloneTest {
public static void main(String[] args) {
Animal a1 = new Animal("花花");
try {
Animal a2 = (Animal) a1.clone();
System.out.println("原始对象:" + a1);
a2.setName("毛毛");
System.out.println("clone之后的对象:" + a2);
} catch (CloneNotSupportedException e) {
e.printStackTrace();
}
}
}
class Animal implements Cloneable{
private String name;
public Animal() {
super();
}
public Animal(String name) {
super();
this.name = name;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
@Override
public String toString() {
return "Animal [name=" + name + "]";
}
@Override
protected Object clone() throws CloneNotSupportedException {
// TODO Auto-generated method stub
return super.clone();
}
}
Object类的finalize()的使用
垃圾回收
finalize()是方法不是关键字。
在对象回收之前会调用finalize()方法。
Object类的finalize()的使用
public class FinalizeTest {
public static void main(String[] args) {
Person p = new Person("Peter", 12);
System.out.println(p);
p = null;//此时对象实体就是垃圾对象,等待被回收。但时间不确定。
System.gc();//强制性释放空间
}
}
class Person{
private String name;
private int age;
public Person(String name, int age) {
super();
this.name = name;
this.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;
}
//子类重写此方法,可在释放对象前进行某些操作
@Override
protected void finalize() throws Throwable {
System.out.println("对象被释放--->" + this);
}
@Override
public String toString() {
return "Person [name=" + name + ", age=" + age + "]";
}
}
结果:
以下重点讲解
==运算符回顾(==和equals区别)
知识点总领。
面试题: == 和 equals() 区别
*
* 一、回顾 == 的使用:
* == :运算符
* 1. 可以使用在基本数据类型变量和引用数据类型变量中
* 2. 如果比较的是基本数据类型变量:比较两个变量保存的数据是否相等。(不一定类型要相同)
* 如果比较的是引用数据类型变量:比较两个对象的地址值是否相同.即两个引用是否指向同一个对象实体
* 补充: == 符号使用时,必须保证符号左右两边的变量类型一致。
* //"hello"==new java.util.Date();会编译错误,因为两边完全不是一个类型。
* 二、equals()方法的使用:
* 1. 是一个方法,而非运算符
* 2. 只能适用于引用数据类型
* 3. Object类中equals()的定义:
* public boolean equals(Object obj) {
return (this == obj);
}
* 说明:Object类中定义的equals()和==的作用是相同的:比较两个对象的地址值是否相同.即两个引用是否指向同一个对象实体
*
* 4. 像String、Date、File、包装类等都重写了Object类中的equals()方法。重写以后,比较的不是
* 两个引用的地址是否相同,而是比较两个对象的"实体内容"是否相同。但是一般写的类里面仍然没有重写,仍然是Object类中定义的equals(),和==的作用是相同的,如需比较实体内容是否相等,还需自己重写equals()函数。
*
* 5. 通常情况下,我们自定义的类如果使用equals()的话,也通常是比较两个对象的"实体内容"是否相同。那么,我们
* 就需要对Object类中的equals()进行重写.
* 重写的原则:比较两个对象的实体内容是否相同.
举例实验
以下两个类是上面的举例。
Customer.java
public class Customer {
private String name;
private int 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 Customer() {
super();
}
public Customer(String name, int age) {
super();
this.name = name;
this.age = age;
}
//自动生成的equals(),以下↓,比较两个实体内容是否相等。
@Override
public boolean equals(Object obj) {
if (this == obj)
return true;
if (obj == null)
return false;
if (getClass() != obj.getClass())
return false;
Customer other = (Customer) obj;
if (age != other.age)
return false;
if (name == null) {
if (other.name != null)
return false;
} else if (!name.equals(other.name))
return false;
return true;
}
Object类中定义的equals()和==的作用是相同的:比较两个对象的地址值是否相同.即两个引用是否指向同一个对象实体
如果我们需要比较实体内容是否相同,有时我们需要重写equals()函数。
//重写的原则:比较两个对象的实体内容(即:name和age)是否相同
//手动实现equals()的重写 以下↓
// @Override
// public boolean equals(Object obj) {
//
System.out.println("Customer equals()....");
// if (this == obj) {
// return true;
// }
//
// if(obj instanceof Customer){
// Customer cust = (Customer)obj;
// //比较两个对象的每个属性是否都相同
if(this.age == cust.age && this.name.equals(cust.name)){
return true;
}else{
return false;
}
//
// //或
// return this.age == cust.age && this.name.equals(cust.name);
// }else{
// return false;
//
// }
//
// }
自定义类也可以重写toString()方法,当调用此内容时,返回对象的实体内容。
//未重写时
//public String toString(){
//return super.toString();
}
//手动实现
// @Override
// public String toString() {
// return "Customer[name = " + name + ",age = " + age + "]";
// }
//自动实现
@Override
public String toString() {
return "Customer [name=" + name + ", age=" + age + "]";
}
}
EqualsTest.java
public class EqualsTest {
public static void main(String[] args) {
//1、可以使用在基本数据类型变量和引用数据类型变量中
//2、如果比较的是基本数据类型变量:比较两个变量保存的数据是否相等。(不一定类型要相同,比如下面的i==d,i会自动类型提升为double 10.0和d比较,然后相等)
//基本数据类型
int i = 10;
int j = 10;
double d = 10.0;
System.out.println(i == j);//true
System.out.println(i == d);//true
boolean b = true;
// System.out.println(i == b);基本数据类型不和布尔型的比较,这句编译出错
char c = 10;
System.out.println(i == c);//true
char c1 = 'A';
char c2 = 65;
System.out.println(c1 == c2);//true
//引用类型:
//3、如果比较的是引用数据类型变量:比较两个对象的地址值是否相同.即两个引用是否指向同一个对象实体
Customer cust1 = new Customer("Tom",21);
Customer cust2 = new Customer("Tom",21);
System.out.println(cust1 == cust2);//false
//像String、Date、File、包装类等都重写了Object类中的equals()方法。重写以后,比较的不是
// 两个引用的地址是否相同,而是比较两个对象的"实体内容"是否相同。
String str1 = new String("atguigu");
String str2 = new String("atguigu");
System.out.println(str1 == str2);//false
System.out.println("****************************");
System.out.println(cust1.equals(cust2));//true
System.out.println(str1.equals(str2));//true
Date date1 = new Date(32432525324L);
Date date2 = new Date(32432525324L);
System.out.println(date1.equals(date2));//true (两个对象的实体内容相等)
}
}
易错点:String是引用数据类型,不能用 == 来比较内容是否相同。
String是引用数据类型,不能用 == 来比较内容是否相同。所以下面的红框部分只能用this.name.equals(cust.name),而不能是this.name=cust.name(这句就是比较地址值是否相等了)
重写equals()方法的原则。
一些练习
一些练习
小知识(String定义的变量在常量池!)
常量池特点,如果再定义一个变量,如果跟已有的变量相同,就直接赋用了,意味着两个赋用的地址是相同的。String定义的变量在常量池!
重点在倒数第9行。↓
/*
* 编写Order类,有int型的orderId,String型的orderName,
* 相应的getter()和setter()方法,两个参数的构造器,
* 重写父类的equals()方法:public boolean equals(Object obj),
* 并判断测试类中创建的两个对象是否相等。
*/
public class OrderTest {
public static void main(String[] args) {
Order order1 = new Order(1001, "AA");
Order order2 = new Order(1001, "BB");
//如果 Order order2 = new Order(1001, new String("BB"));那么System.out.println(order2.equals(order3));就是false了,这里是下面判断equals不能用==的原因。
System.out.println(order1.equals(order2));//false
Order order3 = new Order(1001, "BB");
System.out.println(order2.equals(order3));//true
// String s1 = "BB";
// String s2 = "BB";
// System.out.println(s1 == s2);//true
}
}
class Order{
private int orderId;
private String orderName;
public int getOrderId() {
return orderId;
}
public void setOrderId(int orderId) {
this.orderId = orderId;
}
public String getOrderName() {
return orderName;
}
public void setOrderName(String orderName) {
this.orderName = orderName;
}
public Order(int orderId, String orderName) {
super();
this.orderId = orderId;
this.orderName = orderName;
}
@Override
public boolean equals(Object obj) {
if(this == obj){
return true;
}
if(obj instanceof Order){
Order order = (Order)obj;
//正确的:
return this.orderId == order.orderId &&
this.orderName.equals(order.orderName);
//错误的:(因为这个new Order(1001, new String("BB"));)
重点!!!!!!!!!!!
(这个打开后//String s1 = "BB";
// String s2 = "BB";
// System.out.println(s1 == s2);//true仍然是true。String定义的变量在常量池!常量池特点,如果再定义一个变量,如果跟已有的变量相同,就直接赋用了,意味着两个赋用的地址是相同的。所以仍然是true。)
// return this.orderId == order.orderId &&
// this.orderName == order.orderName;
}
return false;
}
}
引用和对象的区分
1、什么是对象:是类的实例化。是一组相关数据的组织单位(状态)围绕这组数据的各种操作(方法)
2、什么又是引用?:用来操作对象。
3、引用指向对象
3.1、同一时刻,要么指向对象,要
么不指向对象(null)
3.2、对象不能指向引用,也没有对
象指向对象一说。
4、
toString()
知识点总领
* Object类中toString()的使用:
*
* 1. 当我们输出一个对象的引用时,实际上就是调用当前对象的toString()
*
* 2. Object类中toString()的定义:
* public String toString() {
return getClass().getName() + "@" + Integer.toHexString(hashCode());
}//前半部分getClass().getName()是类的类名,后半部分Integer.toHexString(hashCode()是通过hashCode值计算出对空间中的存储位置,并把它转化为16进制。Java中说内存地址是虚拟的地址
*
* 3. 像String、Date、File、包装类等都重写了Object类中的toString()方法。
* 使得在调用对象的toString()时,返回"实体内容"信息
*
* 4. 自定义类也可以重写toString()方法,当调用此方法时,返回对象的"实体内容"
*/
代码举例证明上述知识点
public class ToStringTest {
public static void main(String[] args) {
Customer cust1 = new Customer("Tom",21);
System.out.println(cust1.toString());//com.atguigu.java1.Customer@15db9742-->Customer[name = Tom,age = 21]
System.out.println(cust1);//com.atguigu.java1.Customer@15db9742-->Customer[name = Tom,age = 21]
String str = new String("MM");
System.out.println(str);//MM(因为String重写过)
Date date = new Date(4534534534543L);
System.out.println(date.toString());//Mon Sep 11 08:55:34 GMT+08:00 2113(这也重写过所以输出的不是对象的引用。)
}
}
结果
自定义类也可以重写toString()方法,当调用此方法时,返回对象的"实体内容"(此示例在上面Customer.java函数那末尾有👆)