理解Java中对象基础Object类

简介: Object类是所有类层级关系的Root节点,作为所有类的超类,包括数组也实现了该类的方法。

一、Object简述

源码注释:Object类是所有类层级关系的Root节点,作为所有类的超类,包括数组也实现了该类的方法,注意这里说的很明确,指类层面。

所以在Java中有一句常说的话,一切皆对象,这话并不离谱。

1、显式扩展

结论验证

既然Object作为所有类的父级别的类,则不需要在显式的添加继承关系,Each01编译期就会提示移除冗余。

public class Each01 extends Object {
    public static void main(String[] args) {
        System.out.println(new Each01().hashCode()+";"+new ObjEa02().hashCode());
    }
}
class ObjEa02 {}
class ObjEa03 extends ObjEa02{}

这里Each01ObjEa02对象实例都有Object类中的hashCode方法,这里对既有结论的验证。

编译文件

再从JVM编译层面看下字节码文件,是如何加载,使用javap -c命令查看编译后的文件,注意Jdk版本1.8

javap -c Each01.class
Compiled from "Each01.java"
public class com.base.object.each.Each01 {
  public com.base.object.each.Each01();
    Code:
       0: aload_0
       1: invokespecial #1 // Method java/lang/Object."<init>":()V
       4: return
}

javap -c ObjEa02.class 
Compiled from "Each01.java"
class com.base.object.each.ObjEa02 {
  com.base.object.each.ObjEa02();
    Code:
       0: aload_0
       1: invokespecial #1 // Method java/lang/Object."<init>":()V
       4: return
}

javap -c ObjEa03.class 
Compiled from "Each01.java"
class com.base.object.each.ObjEa03 extends com.base.object.each.ObjEa02 {
  com.base.object.each.ObjEa03();
    Code:
       0: aload_0
       1: invokespecial #1 // Method com/base/object/each/ObjEa02."<init>":()V
       4: return
}

invokespecial命令:可以查看Jvm的官方文档中的指令说明,调用实例化方法,和父类的初始化方法调用等,这里通过三个类的层级关系,再次说明Object超类不需要显式继承,即使显式声明但编译后源码依旧会清除冗余。

2、引用与对象

通常把下面过程称为:创建一个object对象;

Object object = new Object() ;

细节描述:声明对象引用object;通过new关键字创建对象并基于默认构造方法初始化;将对象引用object指向创建的对象。

这一点可以基于Jvm运行流程去理解,所以当对象一旦失去全部引用时,会被标记为垃圾对象,在垃圾收集器运行时清理。

接受任意数据类型对象的引用

既然Object作为Java中所有对象的超类,则根据继承关系的特点,以及向上转型机制,Object可以接受任意数据类型对象的引用,例如在集合容器或者传参过程,不确定对象类型时可以使用Object:

public class Each02 {
    public static void main(String[] args) {
        // 向上转型
        Object obj01 = new Each02Obj01("java") ;
        System.out.println(obj01);
        // 向下转型
        Each02Obj01 each02Obj01 = (Each02Obj01)obj01;
        System.out.println("name="+each02Obj01.getName());
    }
}
class Each02Obj01 {
    private String name ;
    public Each02Obj01(String name) { this.name = name; }
    @Override
    public String toString() {
        return "Each02Obj01{" +"name='" + name +'}';
    }
    public String getName() { return name; }
}

这里要强调一下这个向上转型的过程:

Object obj01 = new Each02Obj01("java") ;

通过上面流程分析,这里创建一个父类引用obj01,并指向子类Each02Obj01对象,所以在输出的时候,调用的是子类的toString方法。

二、基础方法

1、getClass

在程序运行时获取对象的实例类,进而可以获取详细的结构信息并进行操作:

public final native Class<?> getClass();

该方法在泛型,反射,动态代理等机制中有很多场景应用。

2、toString

返回对象的字符串描述形式,Object提供的是类名与无符号十六进制的哈希值组合表示,为了能返回一个信息明确的字符串,子类通常会覆盖该方法:

public String toString() {
    return getClass().getName()+"@"+Integer.toHexString(hashCode());
}

在Java中,打印对象的时候,会执行String.valueOf转换为字符串,该方法的底层依旧是对象的toString方法:

public void println(Object x) {
    String s = String.valueOf(x);
}
public static String valueOf(Object obj) {
    return (obj == null) ? "null" : obj.toString();
}

3、equals与hashCode

  • equals:判断两个对象是否相等;
  • hashCode:返回对象的哈希码值;
public native int hashCode();
public boolean equals(Object obj) {
    return (this == obj);
}

equals判断方法需要考量实际的场景与策略,例如常见的公民注册后分配的身份ID是不能修改的,但是名字可以修改,那么就可能存在这样的场景:

EachUser eachUser01 = new EachUser(1,"A") ;
EachUser eachUser02 = new EachUser(1,"B") ;
class EachUser {
    private Integer cardId ;
    private String name ;
}

从程序本身看,这确实是创建两个对象,但是放在场景下,这的确是描述同一个人,所以这时候可以在equals方法中定义比较规则,如果ID相同则视为同一个对象:

@Override
public boolean equals(Object obj) {
    if (obj != null){
        EachUser compareObj = (EachUser)obj ;
        return this.cardId.intValue()==compareObj.cardId ;
    }
    return Boolean.FALSE ;
}

这里还要注意值类型和引用类型的区别,如果出现null比较情况,要返回false。

通常在子类中会同时覆盖这两个方法,这样做法在集合容器的设计上已经体现的淋漓尽致。

4、thread相关

  • wait:线程进入waiting等待状态,不会争抢锁对象
  • notify:随机通知一个在该对象上等待的线程;
  • notifyAll:唤醒在该对象上所有等待的线程;
public final native void wait(long timeout) throws InterruptedException;
public final native void notify();
public final native void notifyAll();

注意这里:native关键字修饰的方法,即调用的是原生函数,也就是常说的基于C/C++实现的本地方法,以此提高和系统层面的交互效率降低交互复杂程度。

5、clone

返回当前对象的拷贝:

protected native Object clone() throws CloneNotSupportedException;

关于该方法的细节规则极度复杂,要注意下面几个核心点:

  • 对象必须实现Cloneable接口才可以被克隆;
  • 数据类型:值类型,String类型,引用类型;
  • 深浅拷贝的区别和与之对应的实现流程;
  • 在复杂的包装类型中,组合的不同变量类型;

6、finalize

当垃圾收集器确认该对象上没有引用时,会调用finalize方法,即清理内存释放资源:

protected void finalize() throws Throwable { }

通常子类不会覆盖该方法,除非在子类中有一些其他必要的资源清理动作。

三、生命周期

1、作用域

在下面main方法执行结束之后,无法再访问Each05Obj01的实例对象,因为对象的引用each05丢失:

public class Each05 {
    public static void main(String[] args) {
        Each05Obj01 each05 = new Each05Obj01 (99) ;
        System.out.println(each05);
    }
}

这里就会存在一个问题,引用丢失导致对象无法访问,但是对象在此时可能还是存在的,并没有释放内存的占用。

2、垃圾回收机制

Java通过new创建的对象会在堆中开辟内存空间存储,当对象失去所有引用时会被标记为垃圾对象,进而被回收;

这里涉及下面几个关键点:

  • Jvm中垃圾收集器会监控创建的对象 ;
  • 当判断对象不存在引用时,会执行清理动作;
  • 完成对象清理后会重新整理内存空间;

这里存在一个很难理解的概念,即对象不存在引用的判断,也就是常说的可达性分析算法:基于对象到根对象的引用链是否可达来判断对象是否可以被回收;GC-Roots根引用集合,也可以变相理解为存活对象的集合。(详见JVM系列)

通过Object对象的分析,结合Java方方面面的机制和设计,可以去意会一些所谓的编程思想。

相关文章
|
5天前
|
JSON Java Apache
Java基础-常用API-Object类
继承是面向对象编程的重要特性,允许从已有类派生新类。Java采用单继承机制,默认所有类继承自Object类。Object类提供了多个常用方法,如`clone()`用于复制对象,`equals()`判断对象是否相等,`hashCode()`计算哈希码,`toString()`返回对象的字符串表示,`wait()`、`notify()`和`notifyAll()`用于线程同步,`finalize()`在对象被垃圾回收时调用。掌握这些方法有助于更好地理解和使用Java中的对象行为。
|
1月前
|
安全 Java 编译器
Java对象一定分配在堆上吗?
本文探讨了Java对象的内存分配问题,重点介绍了JVM的逃逸分析技术及其优化策略。逃逸分析能判断对象是否会在作用域外被访问,从而决定对象是否需要分配到堆上。文章详细讲解了栈上分配、标量替换和同步消除三种优化策略,并通过示例代码说明了这些技术的应用场景。
Java对象一定分配在堆上吗?
|
2月前
|
Java API
Java 对象释放与 finalize 方法
关于 Java 对象释放的疑惑解答,以及 finalize 方法的相关知识。
54 17
|
1月前
|
存储 Java 程序员
Java基础的灵魂——Object类方法详解(社招面试不踩坑)
本文介绍了Java中`Object`类的几个重要方法,包括`toString`、`equals`、`hashCode`、`finalize`、`clone`、`getClass`、`notify`和`wait`。这些方法是面试中的常考点,掌握它们有助于理解Java对象的行为和实现多线程编程。作者通过具体示例和应用场景,详细解析了每个方法的作用和重写技巧,帮助读者更好地应对面试和技术开发。
135 4
|
2月前
|
存储 缓存 Java
java基础:IO流 理论与代码示例(详解、idea设置统一utf-8编码问题)
这篇文章详细介绍了Java中的IO流,包括字符与字节的概念、编码格式、File类的使用、IO流的分类和原理,以及通过代码示例展示了各种流的应用,如节点流、处理流、缓存流、转换流、对象流和随机访问文件流。同时,还探讨了IDEA中设置项目编码格式的方法,以及如何处理序列化和反序列化问题。
92 1
java基础:IO流 理论与代码示例(详解、idea设置统一utf-8编码问题)
|
1月前
|
存储 安全 Java
Java编程中的对象序列化与反序列化
【10月更文挑战第22天】在Java的世界里,对象序列化和反序列化是数据持久化和网络传输的关键技术。本文将带你了解如何在Java中实现对象的序列化与反序列化,并探讨其背后的原理。通过实际代码示例,我们将一步步展示如何将复杂数据结构转换为字节流,以及如何将这些字节流还原为Java对象。文章还将讨论在使用序列化时应注意的安全性问题,以确保你的应用程序既高效又安全。
|
2月前
|
存储 Java 数据管理
Java零基础-Java对象详解
【10月更文挑战第7天】Java零基础教学篇,手把手实践教学!
37 6
|
2月前
|
Oracle Java 关系型数据库
重新定义 Java 对象相等性
本文探讨了Java中的对象相等性问题,包括自反性、对称性、传递性和一致性等原则,并通过LaptopCharger类的例子展示了引用相等与内容相等的区别。文章还介绍了如何通过重写`equals`方法和使用`Comparator`接口来实现更复杂的相等度量,以满足特定的业务需求。
32 3
|
2月前
|
存储 Java
Java编程中的对象序列化与反序列化
【10月更文挑战第9天】在Java的世界里,对象序列化是连接数据持久化与网络通信的桥梁。本文将深入探讨Java对象序列化的机制、实践方法及反序列化过程,通过代码示例揭示其背后的原理。从基础概念到高级应用,我们将一步步揭开序列化技术的神秘面纱,让读者能够掌握这一强大工具,以应对数据存储和传输的挑战。
|
2月前
|
存储 Java 数据管理
Java零基础-Java对象详解
【10月更文挑战第3天】Java零基础教学篇,手把手实践教学!
35 1