最近经过某大佬的建议准备阅读一下JDK的源码来提升一下自己

所以开始写JDK源码分析的文章

阅读JDK版本为1.8


  • 目录

    • Object结构图
    • 构造器
    • equals 方法
    • getClass 方法
    • hashCode 方法
    • toString 方法
    • finalize 方法
    • registerNatives 方法

1. Object结构图

2. 类构造器

  类构造器是创建Java对象的方法之一。一般我们都使用new关键字来进行实例,还可以在构造器中进行相应的初始化操作。

  在一个Java类中必须存在一个构造器,如果没有添加系统在编译时会默认创建一个无参构造。

  1. /*实例一个Object对象*/
  2. Object obj = new Object()

3. equals 方法

  在面试中面试官经常会问 equals() 方法和 == 运算符的区别,== 运算符用于比较基本类型的值是否相同而 equals 用于比较两个对象是否相等,那么有个问题来了,两个对象怎么才算是相等的呢。

看object中的equals实现

  1. public boolean equals(Object obj) {
  2. return (this == obj);
  3. }

在Object中equals和==是等价的。所以在Object中两个对象的引用相同,那么一定就是相同的。在我们自定义对象的时候一定要重写equals方法。我参考了以下网上的资料来分析一下String中重写的 equals方法:


  1. public boolean equals(Object anObject) {
  2. if (this == anObject) {
  3. return true;
  4. }
  5. if (anObject instanceof String) {
  6. String anotherString = (String)anObject;
  7. int n = value.length;
  8. if (n == anotherString.value.length) {
  9. char v1[] = value;
  10. char v2[] = anotherString.value;
  11. int i = 0;
  12. while (n-- != 0) {
  13. if (v1[i] != v2[i])
  14. return false;
  15. i++;
  16. }
  17. return true;
  18. }
  19. }
  20. return false;
  21. }

String 是引用类型,比较时不能比较引用是否相等,重点是字符串的内容是否相等。所以 String 类定义两个对象相等的标准是字符串内容都相同。

在Java规范中,对 equals 方法的使用必须遵循以下几个原则:

  • 自反性:对于任何非空引用值 x,x.equals(x) 都应返回 true。
  • 对称性:对于任何非空引用值 x 和 y,当且仅当 y.equals(x) 返回 true 时,x.equals(y) 才应返回 true。
  • 传递性:对于任何非空引用值 x、y 和 z,如果 x.equals(y) 返回 true,并且 y.equals(z) 返回 true,那么 x.equals(z) 应返回 true。
  • 一致性:对于任何非空引用值 x 和 y,多次调用 x.equals(y) 始终返回 true 或始终返回 false,前提是对象上 equals 比较中所用的信息没有被修改
  • 对于任何非空引用值 x,x.equals(null) 都应返回 false

下面定义一个类,在这个类中重写equals方法 对象属性相同则相等 否则不相等

  1. public class Student {
  2. private String name;
  3. /**
  4. * 无参构造方法
  5. */
  6. public Student() {
  7. }
  8. /**
  9. * 无参构造方法
  10. */
  11. public Student(String name) {
  12. this.name = name;
  13. }
  14. public String getName() {
  15. return name;
  16. }
  17. public void setName(String name) {
  18. this.name = name;
  19. }
  20. @Override
  21. public boolean equals(Object obj) {
  22. //引用相同 两个对象肯定是相同的
  23. if(this==obj){
  24. return true;
  25. }
  26. //对象等于空 或者不是Student 是不想等的
  27. if(obj==null || !(obj instanceof Student)){
  28. return false;
  29. }
  30. //转为Student对象
  31. Student student = (Student)obj;
  32. //属性相同 返回true
  33. return this.getName()==student.getName();
  34. }
  35. }

然后创建一个测试类来进行测试:

  1. Student t1 = new Student("yes");
  2. Student t2 = new Student("slm");
  3. System.out.println("对象不同 属性不同 == "+(t1==t2));
  4. System.out.println("对象不同 属性不同 equals "+(t1.equals(t2)));
  5. Student t3 = new Student("slm");
  6. System.out.println("对象不同 属性相同"+(t2.equals(t3)));

输出结果:

对象不同 属性不同 == false

对象不同 属性不同 equals false

对象不同 属性相同true

现在可以看出 如果在这里不重写equals方法的话永远只会执行Object的equals也就是通过==对比对象引用地址是否相同。



下面再看一个例子,这个时候如果出现一个Student的子类我们在对比一下

  1. /**
  2. * @Author: sunluomeng
  3. * @CreateTime: 2019-06-06 23:35
  4. * @Description:
  5. */
  6. public class Language extends Student{
  7. private String name;
  8. /**
  9. * 无参构造
  10. */
  11. public Language(){
  12. }
  13. /**
  14. * 有参构造
  15. * @param name
  16. */
  17. public Language(String name){
  18. this.name=name;
  19. }
  20. public String getName() {
  21. return name;
  22. }
  23. public void setName(String name) {
  24. this.name = name;
  25. }
  26. @Override
  27. public boolean equals(Object obj) {
  28. //引用相同 两个对象肯定是相同的
  29. if(this==obj){
  30. return true;
  31. }
  32. //对象等于空 或者不是Student 是不想等的
  33. if(obj==null || !(obj instanceof Language)){
  34. return false;
  35. }
  36. //转为Student对象
  37. Language language = (Language)obj;
  38. //属性相同 返回true
  39. return this.getName()==language.getName();
  40. }
  41. }

这个时候我们的新创建的Language类继承Student然后创建两个对象去做比较



输出结果:

父类对比子类 属性相同---true

子类对比父类 属性相同---false

可以看出父类去对比子类既 student.equals(language) 结果为true 而子类去对比父类 既 language.equals(student) 返回false

这样的话就违反了问哦们上面说到的对称性

对于任何非空引用值 x 和 y,当且仅当 y.equals(x) 返回 true 时,x.equals(y) 才应返回 true

如果y是Student x 是Language
那么现在就是 y.equals(x) 等于true 反过来x.equals(y)也应该返回true,但是现在为什么会返回false呢?

先来看一下代码



我们在判断的时候使用了instanceof关键字来判断运行的时候是否是指定的类型

java 中的instanceof 运算符是用来在运行时指出对象是否是特定类的一个实例。instanceof通过返回一个布尔值来指出,这个对象是否是这个特定类或者是它的子类的一个实例。

这样的话也就是说 Language是Student的子类 在用instanceof判断的时候是返回true,而Language虽然是继承Student 但是使用instanceof判断的时候会发现 Language和Student的类型不同 然后Student也不是Language的子类所以会返回false。



而解决的办法就是

然后我们在运行一下刚刚的代码

输出结果:

父类对比子类 属性相同---false

子类对比父类 属性相同---false

完美解决,满足对称性

注意:使用getClass是要根据情况而定,使用getClass 不符合多态的定义

那什么时候使用instanceof,什么时候使用getClass呢?

  • 如果子类能够拥有自己的相等概念,则对称性需求将强制采用 getClass 进行检测。
  • 如果有超类决定相等的概念,那么就可以使用 instanceof 进行检测,这样可以在不同的子类的对象之间进行相等的比较。

还有就是一定要注意无论何时重写此方法,通常都必须重写hashCode方法,以维护hashCode方法的一般约定,该方法声明相等对象必须具有相同的哈希代码。

4.getClass 方法

我们首先看一下getClass在Object中的实现。



我们看到getClass被native标识,这代表这是调用本地方法实现

关于native更多请百度。native是由操作系统帮我们实现

文档说明的是调用getClass返回一个运行时的类。什么意思呢 我们看下面的代码实现。



打印结果:

可以看出getClass是返回一个运行时的对象。class是返回编译的类对象

可以看到getClass方法被final修饰,说明此方法不能被重写。

5.hashCode

先看一下hashCode在Object中的实现:

hashCode也是一个被native修饰的本地方法

注释说明的是返回该对象的哈希值。那么它有什么作用呢?

主要是保证基于散列的集合,如HashSet、HashMap以及HashTable等,在插入元素时保证元素不可重复,同时为了提高元素的插入删除便利效率而设计;主要是为了查找的便捷性而存在。

就比如使用Set进行举例子。

Set集合是不可重复的,如果每次添加数据都使用equals去做对比的话,插入十万条数据就要对比十万次效率是非常慢的。



所以在添加数据的时候使用了哈希表,哈希算法也称之为散列算法,当添加一个值的时候先算出它的哈希值根据算出的哈希值将数据插入指定位置。这样的话就避免了一直调用equals造成的效率隐患。同时有以下条件:

  • 如果位置为空则直接添加
  • 如果位置不为空,判断两个元素是否相同如果相同则不存储。

还有一种情况是两个元素不相同,但是hashCode相同,这就是哈希碰撞。

如果发生了hash key相同的情况就在相同的元素创建一个链表。把所有相同的元素存放在链表中。



可以看出T1的哈希和T2相同,但是元素不同,所以现在会形成一个链来存储。

6.toString

先看toString的实现



可以看出toString是返回的类名加16进制无符号整数形式返回此哈希码的字符串表示形式。

运行输出结果:



直接输出对象和使用toString是一样的



如果想要toString输出属性内容则需要重写toString方法

7.finalize

源码中实现方法:



finalize用户垃圾回收是由JVM调用。

8.registerNatives

源码实现:



上面说到native是调用本地实现方法,而registerNatives则是对本地方法注册,装载本地库。在Object初始化时执行。

还有notify()/notifyAll()/wait()等写到多线程的时候在做分析

最后

小弟不才,如有错误请指出。喜欢请关注,慢慢更新JDK源码阅读笔记

** 小弟公众号,乱敲代码。欢迎点赞,关注**


JDK源码阅读(一):Object源码分析的更多相关文章

  1. JDK源码阅读:Object类阅读笔记

    Object 1. @HotSpotIntrinsicCandidate @HotSpotIntrinsicCandidate public final native Class<?> g ...

  2. 搜索引擎Hoot的源码阅读(提供源码)

    开门见山,最近阅读了一下一款开源引擎的源码,受益良多(学到了一些套路).外加好久没有写博客了(沉迷吃鸡,沉迷想念姑娘),特别开一篇.Hoot 的源码地址, 原理介绍地址.外加我看过之后的注释版本,当然 ...

  3. Spring源码阅读 之 搭建源码阅读环境(IDEA)

    检出源码: GitHub:https://github.com/spring-projects/spring-framework.git 可以按如下步骤:(须确保Git已正确安装) Git正确安装后, ...

  4. Redis源码阅读-sds字符串源码阅读

    redis使用sds代替char *字符串, 其定义如下: typedef char *sds; struct sdshdr { unsigned int len; unsigned int free ...

  5. 转-OpenJDK源码阅读导航跟编译

    OpenJDK源码阅读导航 OpenJDK源码阅读导航 博客分类: Virtual Machine HotSpot VM Java OpenJDK openjdk 这是链接帖.主体内容都在各链接中.  ...

  6. 《java.util.concurrent 包源码阅读》 结束语

    <java.util.concurrent 包源码阅读>系列文章已经全部写完了.开始的几篇文章是根据自己的读书笔记整理出来的(当时只阅读了部分的源代码),后面的大部分都是一边读源代码,一边 ...

  7. JDK源码阅读-------自学笔记(一)(java.lang.Object重写toString源码)

    一.前景提要 Object类中定义有public String toString()方法,其返回值是 String 类型. 二.默认返回组成 类名+@+16进制的hashcode,当使用打印方法打印的 ...

  8. jdk源码阅读笔记-LinkedHashMap

    Map是Java collection framework 中重要的组成部分,特别是HashMap是在我们在日常的开发的过程中使用的最多的一个集合.但是遗憾的是,存放在HashMap中元素都是无序的, ...

  9. jdk源码阅读笔记-HashSet

    通过阅读源码发现,HashSet底层的实现源码其实就是调用HashMap的方法实现的,所以如果你阅读过HashMap或对HashMap比较熟悉的话,那么阅读HashSet就很轻松,也很容易理解了.我之 ...

随机推荐

  1. windows添加本地文件托管到新增github库

    新增repositoy.登录gitHub,并点击“New Reposoitory” 写入名字  之后点击“create resposity” \ 按照上图中的步骤可以完成.以下为完成步骤. 2. 在本 ...

  2. 工具:sql server profiler(分析器)

    打开profiler新建->连接数据库进行监测 任何访问该数据库的都有记录   image 对于linq的检验:sql实际如何->运行程序,查看分析器记录   image         ...

  3. npm学习(-)

    了解npm请前往https://www.npmjs.cn/getting-started/what-is-npm/ npm 由三个独立的部分组成: 网站 注册表(registry) 命令行工具 (CL ...

  4. StepShot4.3.0安装包_KeyGen发布

    StepShot是一个可以方便快速的制作操作手册的软件,功能相当强悍. 请低调使用. -------------------------------华丽的分割线-------------------- ...

  5. IOS开发之把 JSON 数据转化成 Arrays 或者 Dictionaries

    1 前言通过 NSJSONSerialization 这个类的 JSONObjectWithData:options:error:方法来实现,把JSON 数据解析出来放在数据或者字典里面保存. 2 代 ...

  6. 图片处理拓展篇 : 图片转字符画(ascii)

    首先要明确思路, 图片是由像素组成的, 不同的像素有不同的颜色(rgb), 那么既然我们要转化为字符画, 最直接的办法就是利用字符串来替代像素, 也就是用不同的字符串来代表不同的像素. 另外图片一般来 ...

  7. Java底层知识学习:Bytecode and JMM

    最近在跟着耗子哥的程序员练级指南学习Java底层知识,结合<深入理解Java虚拟机>这本书在看,写笔记,看资料,成长中…… 目前看完了第二章JMM和各内存区OOM的情况 一篇图文并茂介绍字 ...

  8. nodejs redis遇到的一个问题解决

    v ar redis = require("redis"), client = redis.createClient({host:'tc-arch-osp33.tc', port: ...

  9. Sysinternals套件2016年11月更新发布,诸多工具被更新

    Sysinternals 实用程序可帮助您管理.解决和诊断 Windows 系统和应用程序,在 Sysinternals 的 2016 年 11 月更新中,微软增强 在 sysmon 对注册表和文件事 ...

  10. Delphi中取得汉字的首字母(十分巧妙)

    function Tdm.GetHzPy(const AHzStr: string): string;const  ChinaCode: array[0..25, 0..1] of Integer = ...