理解Java中对象基础Object类
一、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{}
这里Each01
与ObjEa02
对象实例都有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方方面面的机制和设计,可以去意会一些所谓的编程思想。
同系列:Wiki文档 | List分析 | Map分析 | IO流核心 | 动态代理 | 面向对象
四、源代码地址
GitEE·地址
https://gitee.com/cicadasmile/java-base-parent
Wiki·地址
https://gitee.com/cicadasmile/butte-java-note/wikis
阅读标签
【Java基础】【设计模式】【结构与算法】【Linux系统】【数据库】
【分布式架构】【微服务】【大数据组件】【SpringBoot进阶】【Spring&Boot基础】
理解Java中对象基础Object类的更多相关文章
- Java 中对象锁和类锁的区别? 关键字 Synchronized的用法?
一 对象锁和类锁的关系 /* * 对象锁和[类锁] 全局锁的关系? 对象锁是用于对象实例方法,或者一个对象实例上的 this 类锁是用于类的静态方法或者一个类的class对象上的. Ag.class ...
- 个人对java中对象锁与类锁的一些理解与实例
一 什么是对象锁 对象锁也叫方法锁,是针对一个对象实例的,它只在该对象的某个内存位置声明一个标识该对象是否拥有锁,所有它只会锁住当前的对象,而并不会对其他对象实例的锁产生任何影响,不同对象访问同一个 ...
- EBS OAF开发中的Java 实体对象(Entity Object)验证功能补充
EBS OAF开发中的Java 实体对象(Entity Object)验证功能补充 (版权声明,本人原创或者翻译的文章如需转载,如转载用于个人学习,请注明出处:否则请与本人联系,违者必究) EO理论上 ...
- 深入理解Java中的不可变对象
深入理解Java中的不可变对象 不可变对象想必大部分朋友都不陌生,大家在平时写代码的过程中100%会使用到不可变对象,比如最常见的String对象.包装器对象等,那么到底为何Java语言要这么设计,真 ...
- 深刻理解Java中final的作用(一):从final的作用剖析String被设计成不可变类的深层原因
声明:本博客为原创博客,未经同意,不得转载!小伙伴们假设是在别的地方看到的话,建议还是来csdn上看吧(原文链接为http://blog.csdn.net/bettarwang/article/det ...
- Java中直接输出一个类的对象
例如 package com.atguigu.java.fanshe; public class Person { String name; private int age; public Strin ...
- Java中的不可变类理解
一.Java中的不可变类 不可变类(Immutable Objects):当类的实例一经创建,其内容便不可改变,即无法修改其成员变量. 可变类(Mutable Objects):类的实例创建后,可以修 ...
- Java中对象和引用的理解
偶然想起Java中对象和引用的基本概念,为了加深下对此的理解和认识,特地整理一下相关的知识点,通过具体实例从两者的概念和区别两方面去更形象的认识理解,再去记忆. 一.对象和引用的概念: 在Java中万 ...
- 你真的了解JAVA中对象和类、this、super和static关键字吗
作者:小牛呼噜噜 | https://xiaoniuhululu.com 计算机内功.JAVA底层.面试相关资料等更多精彩文章在公众号「小牛呼噜噜 」 目录 Java对象究竟是什么? 创建对象的过程 ...
随机推荐
- PAT乙级:1066 图像过滤 (15分)
PAT乙级:1066 图像过滤 (15分) 题干 图像过滤是把图像中不重要的像素都染成背景色,使得重要部分被凸显出来.现给定一幅黑白图像,要求你将灰度值位于某指定区间内的所有像素颜色都用一种指定的颜色 ...
- Leetcode:559. N叉树的最大深度
Leetcode:559. N叉树的最大深度 Leetcode:559. N叉树的最大深度 Talk is cheap . Show me the code . /* // Definition fo ...
- 【队列+模拟】机器翻译 luogu-1540
题目描述 小晨的电脑上安装了一个机器翻译软件,他经常用这个软件来翻译英语文章. 这个翻译软件的原理很简单,它只是从头到尾,依次将每个英文单词用对应的中文含义来替换.对于每个英文单词,软件会先在内存中查 ...
- Vue__npm run build npm run dev
npm run build npm run dev 一.以前一直错的做法 以前,git完项目之后就,执行1.npm install 2.npm run build 3.npm run dev.今天ma ...
- 入门Kubernetes-数据存储
一.Volume介绍: 在k8s中Pod的生命周期可能很短,会被频繁地销毁和创建.容器销毁时,保存在容器内部文件系统中的数据都会被清除. 为了持久化保存容器数据,k8s 提供了卷(Volume)的抽象 ...
- 看视频学Bootstrap—在微软虚拟学院学习Bootstrap
Bootstrap 是目前最流行的 HTML.CSS 和 JS 框架,用于开发响应式布局.移动设备优先的 WEB项目. 如果您希望在几个小时内对Bootstrap有一个直观的了解,观看微软虚拟学院(M ...
- CentOS 8.0 安装docker 报错:Problem: package docker-ce-3:19.03.4-3.el7.x86_64 requires containerd.io >= 1
1.错误内容 package docker-ce-3:19.03.2-3.el7.x86_64 requires containerd.io >= 1.2.2-3, but none of th ...
- HCNA Routing&Switching之STP基础
前文我们了解了VLAN动态注册协议GVRP相关话题,回顾请参考https://www.cnblogs.com/qiuhom-1874/p/15113770.html:今天我们来讨论下二层环路和STP相 ...
- WPF Combox实现下拉多选,可选中多个值
自定义多选MultiCombox,可以实现下拉列表多选 using System; using System.Collections.Generic; using System.Collections ...
- 812考试总结(NOIP模拟37)[数列·数对·最小距离·真相]
前言 考得挺憋屈的... 先是搞了两个半小时的 T1 后来发现假了,又没多想跳了.. 然后一看 T2 这不是队长快跑嘛... 先是根据自己的想法打了一遍(考完之后发现是对的..) 然后回想了一下之前的 ...