当前位置: 首页 > news >正文

如何阅读jdk源码?

2019独角兽企业重金招聘Python工程师标准>>> hot3.png

简介

这篇文章主要讲述jdk本身的源码该如何阅读,关于各种框架的源码阅读我们后面再一起探讨。

笔者认为阅读源码主要包括下面几个步骤。

设定目标

凡事皆有目的,阅读源码也是一样。

从大的方面来说,我们阅读源码的目的是为了提升自己的技术能力,运用到工作中,遇到问题快速定位,升职加薪等等。

从小的方面来说,阅读某一段源码的目的就是要搞清楚它的原理,就是死磕,就是那种探索真相的固执。

目的是抽象的,目标是具体的,我们阅读源码之前一定要给自己设定一个目标。

比如,下一章我们将要一起学习的ConcurrentHashMap,我们可以设定以下目标:

(1)熟悉ConcurrentHashMap的存储结构;

(2)熟悉ConcurrentHashMap中主要方法的实现过程;

(3)探索ConcurrentHashMap中出现的新技术;

提出问题

有了目标之后,我们要试着提出一些问题。

还是以ConcurrentHashMap为例,笔者提出了以下这些问题:

(1)ConcurrentHashMap与HashMap的数据结构是否一样?

(2)HashMap在多线程环境下何时会出现并发安全问题?

(3)ConcurrentHashMap是怎么解决并发安全问题的?

(4)ConcurrentHashMap使用了哪些锁?

(5)ConcurrentHashMap的扩容是怎么进行的?

(6)ConcurrentHashMap是否是强一致性的?

(7)ConcurrentHashMap不能解决哪些问题?

(8)ConcurrentHashMap除了并发安全,还有哪些与HashMap不同的地方,为什么要那么实现?

(8)ConcurrentHashMap中有哪些不常见的技术值得学习?

如何提出问题

很多人会说,我也知道要提出问题,但是该怎么提出问题呢?

这确实是很困难的一件事,笔者认为主要是三点:

(1)问自己

把自己当成面试官问自己,往死里问的那种。

如果问自己问不出几个问题,也不要紧,请看下面。

(2)问互联网

很多问题可能自己也想不到,那就需要上网大概查一下相关的博客,看人家有没有提出什么问题。

或者,查询相关面试题。

比如,笔者学习ConcurrentHashMap这个类时,上网一查很多都是基于jdk7的,那这时候就可以提出一个问题,jdk8与jdk7中ConcurrentHashMap这个类的实现方式有何不同?jdk8对jdk7作了哪些优化?

(3)不断发现问题

在源码阅读的过程中,可能看着看着就遇到个问题,这是非常常见的,这种问题也应该保留下来研究研究。

比如,ConcurrentHashMap中size()方法是怎么实现的?@sun.misc.Contended这玩意是什么鬼东西?然后上网一查,与是为了避免伪共享,我X,伪共享又是啥?然后你再查一下伪共享,又出来了CPU多级缓存?学完CPU多级缓存,是不是觉得跟jvm的内存模型很像?问完这一连串问题,是不是感觉世界都清晰了?^_^

看吧,问题是源源不断地被发现的。

所以,一开始提不出几个问题也不要紧,关键是要看,看了才能发现更多的问题。

带着问题阅读源码,忽略不必要的细节,死磕重要的细节

首先,一定要带着问题阅读源码。

其次,一定要忽略不必要的细节。

再次,一定要死磕重要的细节。

乍一看,后面两步似乎有所矛盾,其实不然,忽略不必要的细节是为了不迷失在源码的世界中,死磕重要的细节是为了弄清楚源码的真相。

这里的细节是忽略还是死磕,主要是看跟问题的相关性。

jdk源码还是比较好阅读的,如果后面看spring的源码,做不到忽略不必要的细节,真的是会迷失的,先埋个伏笔哈~~

举个例子,之前阅读过ArrayList的序列化相关的代码中的readObject()方法。

s.readInt();这行是干嘛的?省略行不行?这时候就要去了解序列化相关的知识,然后看看writeObject()里面的实现,这就是要死磕的代码。

SharedSecrets.getJavaOISAccess().checkArray(s, Object[].class, capacity);这行又是干嘛的?乍一看,好像是跟权限相关的代码,跟我们的问题“序列化”无关,忽略之,如果实在想知道,先打个标记,等把序列化的问题解决了再来研究这个东西。

private void readObject(java.io.ObjectInputStream s)
        throws java.io.IOException, ClassNotFoundException {
    // 声明为空数组
    elementData = EMPTY_ELEMENTDATA;

    // 读入非transient非static属性(会读取size属性)
    s.defaultReadObject();

    // 读入元素个数,没什么用,只是因为写出的时候写了size属性,读的时候也要按顺序来读
    s.readInt();

    if (size > 0) {
        // 计算容量
        int capacity = calculateCapacity(elementData, size);
        SharedSecrets.getJavaOISAccess().checkArray(s, Object[].class, capacity);
        // 检查是否需要扩容
        ensureCapacityInternal(size);
        
        Object[] a = elementData;
        // 依次读取元素到数组中
        for (int i=0; i<size; i++) {
            a[i] = s.readObject();
        }
    }
}

多做比较

在阅读jdk源码的时候,还有很重要的一点,就是要多做比较,比较也可以分为横向比较和纵向比较。

(1)横向比较

就是与相似的类做比较。比如,集合模块中,基本都是各种插入、查询、删除元素,那这时候可以从数据结构、时间复杂度等维度进行比较,这就是横向比较。

(2)纵向比较

可以从集合发展的历史进行比较。比如,HashMap的发展史,从(单个数组)实现(没错,可以直接用一个数组实现HashMap),到(多数组+链表)实现,再到jdk8中的(多数组+链表+红黑树)实现,这就是纵向比较。

多做实验

最后一步,最最最最重要的就是要多做实验。

比如,ConcurrentHashMap是不是强一致性的?

可以启动多个线程去不断调用get()、put()、size()方法,看看是不是强一致性的。

耐心&坚持

这一点我不多说,大家都懂得。

不管是什么领域,耐心&坚持都是最难能可贵的品质。

阅读源码也是一样,只要耐心地坚持下去,终将有所收获。

彩蛋

哎呀,一不小心透露了下一章ConcurrentHashMap的内容。

大家可以用本篇所说的方法试着阅读一下ConcurrentHashMap的源码,下一章我们再一起学习哈哈~~


qrcode

转载于:https://my.oschina.net/u/4108008/blog/3033339

相关文章:

  • 判断语句与循环语句
  • 类(初学)
  • 如何理解MVVM?
  • 程序员着装的退化史:曾经也是体面人
  • kubernetes1.4新特性(一):支持sysctl命令
  • Linux通配符应用详解
  • 蚂蚁金服CTO程立:真正的技术革命才刚刚开始
  • 性能测试学习 第四课
  • Markdown 编辑器 1.1.7 发布,新增全屏模式
  • pdf如何解密?解密的方法有这些
  • linux安装docker
  • docker使用阿里云镜像仓库
  • JDK基础---组合知识疑问
  • Spring Cloud学习笔记--Spring Boot初次搭建
  • Unity3D - 异步加载游戏场景与异步加载游戏资源进度条 ...
  • (ckeditor+ckfinder用法)Jquery,js获取ckeditor值
  • 【知识碎片】第三方登录弹窗效果
  • C++11: atomic 头文件
  • Fundebug计费标准解释:事件数是如何定义的?
  • iOS动画编程-View动画[ 1 ] 基础View动画
  • JS+CSS实现数字滚动
  • MySQL用户中的%到底包不包括localhost?
  • Mysql优化
  • PyCharm搭建GO开发环境(GO语言学习第1课)
  • vuex 学习笔记 01
  • Vue小说阅读器(仿追书神器)
  • 大整数乘法-表格法
  • 动态规划入门(以爬楼梯为例)
  • 跨域
  • 离散点最小(凸)包围边界查找
  • 利用DataURL技术在网页上显示图片
  • 模型微调
  • 排序(1):冒泡排序
  • 嵌入式文件系统
  • 实战:基于Spring Boot快速开发RESTful风格API接口
  • 通过获取异步加载JS文件进度实现一个canvas环形loading图
  • 学习JavaScript数据结构与算法 — 树
  • AI又要和人类“对打”,Deepmind宣布《星战Ⅱ》即将开始 ...
  • 容器镜像
  • # 达梦数据库知识点
  • #设计模式#4.6 Flyweight(享元) 对象结构型模式
  • (31)对象的克隆
  • (C语言)输入自定义个数的整数,打印出最大值和最小值
  • (Redis使用系列) Springboot 使用Redis+Session实现Session共享 ,简单的单点登录 五
  • (论文阅读11/100)Fast R-CNN
  • (转)JVM内存分配 -Xms128m -Xmx512m -XX:PermSize=128m -XX:MaxPermSize=512m
  • (轉貼)《OOD启思录》:61条面向对象设计的经验原则 (OO)
  • .NET : 在VS2008中计算代码度量值
  • .NET CORE使用Redis分布式锁续命(续期)问题
  • .net core使用RPC方式进行高效的HTTP服务访问
  • .NET Framework 的 bug?try-catch-when 中如果 when 语句抛出异常,程序将彻底崩溃
  • .Net MVC4 上传大文件,并保存表单
  • .NET 服务 ServiceController
  • .net6Api后台+uniapp导出Excel
  • .Net开发笔记(二十)创建一个需要授权的第三方组件