JAVA面向对象6(Object类结构的剖析,Object类的方法 )

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.StringDateFile、包装类等都重写了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.StringDateFile、包装类等都重写了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函数那末尾有👆)
在这里插入图片描述

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值