读源码时的思考

最近在看concurrent包下线程池的源码,当我看到ThreadPoolExecutor类的时候,发现了JDK源码的一个问题。以下是ThreadPoolExecutor类的addWorker方法的代码片段:

  1. boolean workerStarted = false;
  2. boolean workerAdded = false;
  3. Worker w = null;
  4. try {
  5. w = new Worker(firstTask);
  6. final Thread t = w.thread;
  7. if (t != null) {
  8. final ReentrantLock mainLock = this.mainLock;
  9. mainLock.lock();
  10. try {
  11. int rs = runStateOf(ctl.get());
  12. if (rs < SHUTDOWN ||
  13. (rs == SHUTDOWN && firstTask == null)) {
  14. if (t.isAlive()) // precheck that t is startable
  15. throw new IllegalThreadStateException();
  16. workers.add(w);
  17. int s = workers.size();
  18. if (s > largestPoolSize)
  19. largestPoolSize = s;
  20. workerAdded = true;
  21. }
  22. } finally {
  23. mainLock.unlock();
  24. }
  25. if (workerAdded) {
  26. t.start();
  27. workerStarted = true;
  28. }
  29. }
  30. } finally {
  31. if (! workerStarted)
  32. addWorkerFailed(w);
  33. }
  34. return workerStarted;

这段代码的功能是完全没有问题的,但是如果使用卫语句,代码的可读性就会更高了。那么什么是卫语句呢?

欢迎关注微信公众号:万猫学社,每周一分享Java技术干货。

什么是卫语句?

条件表达式通常有两种表现形式,第一种形式是:所有分支都属于正常行为;第二种形式则是:条件表达式提供的答案中只有一种是正常行为,其他都是不常见的情况。这两类条件表达式有不同的用途,这一点应该通过代码表现出来。

如果两条分支都是正常行为,就应该使用形如if...else...的条件表达式;如果某个条件极其罕见,就应该单独检查该条件,并在该条件为真时立刻从函数中返回。这样的单独检查常常被称为“卫语句”(guard clauses)。只看概念干巴巴的,不好理解,我们来举两个例子。

欢迎关注微信公众号:万猫学社,每周一分享Java技术干货。

条件检查替换

这是一个计算员工薪资的方法,其中以特殊规则处理驻外员工和退休员工的薪资。这些情况不常有,但的确会偶尔出现。

  1. public double getSalary() {
  2. double result;
  3. if (this.isSeparated) {//驻外员工
  4. result = this.separatedSalary();
  5. } else {
  6. if (this.isRetired) {//退休员工
  7. result = this.retiredSalary();
  8. } else {//正常员工
  9. result = this.normalSalary();
  10. }
  11. }
  12. return result;
  13. }

这段代码中,非正常情况的检查掩盖了正常情况的检查,所以应该用卫语句来取代这些条件检查,以提高程序清晰度。对于每个检查,放进一个卫语句。卫语句要不就从函数中返回,要不就抛出一个异常。


  1. public double getSalary() {
  2. if (this.isSeparated) {//卫语句
  3. return this.separatedSalary();
  4. }
  5. if (this.isRetired) {//卫语句
  6. return this.retiredSalary();
  7. }
  8. return this.normalSalary();
  9. }

欢迎关注微信公众号:万猫学社,每周一分享Java技术干货。

反转条件替换

这是一个已知长宽高求长方体体积的方法,但有个特殊的需求:高大于0时,打印万猫学社。(惊不惊喜?意不意外?突不突兀?变不变态?是的,有时候我们接到的需求就是这样的。)代码是这样的:

  1. public double getVolume(double length, double width, double height) {
  2. double result = 0.0;
  3. if (height > 0.0) {
  4. System.out.println("万猫学社");
  5. if (length > 0.0 && width > 0.0) {
  6. result = length * width * height;
  7. }
  8. }
  9. return result;
  10. }

还是用卫语句替换条件检查,但是我们需要将相应的条件反转过来,也就是做逻辑非运算。

  1. public double getVolume(double length, double width, double height) {
  2. if (height <= 0.0) {//卫语句,!(height > 0)
  3. return 0.0;
  4. }
  5. System.out.println("万猫学社");
  6. if (length <= 0.0 || width <= 0.0) {//卫语句,!(length > 0 && width > 0)
  7. return 0.0;
  8. }
  9. return length * width * height;
  10. }

欢迎关注微信公众号:万猫学社,每周一分享Java技术干货。

为什么要使用卫语句?

卫语句的精髓是:给某一条分支以特别的重视。如果使用if...else...结构,你对if分支和else分支的重视是同等的。这样的代码结构传递给阅读者的消息就是:各个分支有同样的重要性。

卫语句就不同了,它告诉阅读者:“这种情况很罕见,如果它真的发生了,请做一些必要的整理工作,然后退出。”如果对方法剩余部分不再有兴趣,当然应该立刻退出。引导代码的阅读者去看一个没有用的else区段,只会妨碍他们的理解。用了卫语句以后,代码更容易被理解,被维护。

欢迎关注微信公众号:万猫学社,每周一分享Java技术干货。

优化JDK代码

看了上面的讲解,addWorker方法的代码片段就可以优化为:

  1. boolean workerStarted = false;
  2. boolean workerAdded = false;
  3. Worker w = null;
  4. try {
  5. w = new Worker(firstTask);
  6. final Thread t = w.thread;
  7. if (t == null) {//卫语句
  8. return workerStarted;
  9. }
  10. final ReentrantLock mainLock = this.mainLock;
  11. mainLock.lock();
  12. try {
  13. int rs = runStateOf(ctl.get());
  14. //卫语句
  15. if (rs > SHUTDOWN || (rs == SHUTDOWN && firstTask != null)) {
  16. return workerStarted;
  17. }
  18. if (t.isAlive())// precheck that t is startable
  19. throw new IllegalThreadStateException();
  20. workers.add(w);
  21. int s = workers.size();
  22. if (s > largestPoolSize)
  23. largestPoolSize = s;
  24. workerAdded = true;
  25. } finally {
  26. mainLock.unlock();
  27. }
  28. if (workerAdded) {
  29. t.start();
  30. workerStarted = true;
  31. }
  32. } finally {
  33. if (!workerStarted)
  34. addWorkerFailed(w);
  35. }
  36. return workerStarted;

修改后的代码,理解起来是不是更容易了?假如再加入新的功能,可以更容易修改代码。

欢迎关注微信公众号:万猫学社,每周一分享Java技术干货。

结语

这段JDK源码在功能上没有任何问题,架构设计也堪称完美,不过我认为在可读性上还是可以优化的。类似这样的嵌套条件表达式的代码在JDK源码中不止这一处,可能是作者当时没有考虑到使用卫语句,或者没有像我这样的吹毛求疵。希望大家在自己编码过程中,也可以尽量使用卫语句取代嵌套条件表达式,以提高代码的清晰度和可维护性。

欢迎关注微信公众号:万猫学社,每周一分享Java技术干货。

参考文献:《重构:改善既有代码的设计》

震惊!我竟然发现了JDK源码的问题的更多相关文章

  1. 调试过程中发现按f5无法走进jdk源码

    debug 模式 ,在fis=new FileInputStream(file); 行打断点 调试过程中发现按f5无法走进jdk源码 package com.lzl.spring.test; impo ...

  2. 重新编译jdk源码,启用debug信息

    我有一个不知道是好还是不好的习惯,搞不懂的一些玩意儿,喜欢调试然后单步执行看这玩意儿到底是怎么运行的. 今天看到正则表达式的时候,appendReplacement()这个方法怎么也看不明白它是怎么工 ...

  3. eclipse下导入jdk源码

    一直想好好看看jdk的源码,虽然可以直接解压jdk下的src看,但是终究不方便!后来发现可以导入到eclipse中,就在网上找了一些方法,下面就和大家分共享: step1:打开eclipse选择Win ...

  4. Timer的故事----Jdk源码解读

    咱们今天也来说说定时器Timer Timer是什么? Timer  n. [电子] 定时器:计时器:计时员 从翻译来看,我们可以知道Timer的本意是,定时定点. 而JDK中Timer类也的确是这个本 ...

  5. JDK源码包结构分类

    最近查看JDK源码时,无意间发现几个类在陌生包里:com.sun.*.sun.*.org.*,google了一把总结了下以备他人搜索,如内容有误欢迎指正!   Jre库包含的jar文件(jdk1.6) ...

  6. jdk源码调试功能

    JDK源码重新编译——支持eclipse调试JDK源码--转载 最近在研究jdk源码,发现debug时无法查看源码里的变量值. 因为sun提供的jdk并不能查看运行中的局部变量,需要重新编译一下rt. ...

  7. JDK源码重新编译——支持eclipse调试JDK源码--转载

    最近在研究jdk源码,发现debug时无法查看源码里的变量值. 因为sun提供的jdk并不能查看运行中的局部变量,需要重新编译一下rt.jar. 下面这六步是编译jdk的具体步骤: Step 1:   ...

  8. JDK源码学习--String篇(二) 关于String采用final修饰的思考

    JDK源码学习String篇中,有一处错误,String类用final[不能被改变的]修饰,而我却写成静态的,感谢CTO-淼淼的指正. 风一样的码农提出的String为何采用final的设计,阅读JD ...

  9. JDK源码分析—— ArrayBlockingQueue 和 LinkedBlockingQueue

    JDK源码分析—— ArrayBlockingQueue 和 LinkedBlockingQueue 目的:本文通过分析JDK源码来对比ArrayBlockingQueue 和LinkedBlocki ...

随机推荐

  1. css3:bacground-size

    个人博客: https://chenjiahao.xyz CSS3之背景尺寸Background-size是CSS3中新加的一个有关背景的属性,这个属性是改变背景尺寸的通过各种不同是属性值改变背景尺寸 ...

  2. 扛住阿里双十一高并发流量,Sentinel是怎么做到的?

    Sentinel 承接了阿里巴巴近 10 年的双十一大促流量的核心场景 本文介绍阿里开源限流熔断方案Sentinel功能.原理.架构.快速入门以及相关框架比较 基本介绍 1 名词解释 服务限流 :当系 ...

  3. Rust入坑指南:常规套路

    搭建好了开发环境之后,就算是正式跳进Rust的坑了,今天我就要开始继续向下挖了. 由于我们初来乍到 ,对Rust还不熟悉,所以我决定先走一遍常规套路. 变不变的变量 学习一门语言第一个要了解的当然就是 ...

  4. django-rest-framework解析请求参数

    django-rest-framework解析请求参数 前言 前面的文章中编写了接口, 调通了接口文档. 接口文档可以直接填写参数进行请求, 接下来的问题是如何接受参数, 由于请求方式与参数序列化形式 ...

  5. Python基础库之jieba库的使用(第三方中文词汇函数库)

    各位学python的朋友,是否也曾遇到过这样的问题,举个例子如下: “I am proud of my motherland” 如果我们需要提取中间的单词要走如何做? 自然是调用string中的spl ...

  6. LeetCode 第 287 号问题:寻找重复数,一道非常简单的数组遍历题,加上四个条件后感觉无从下手

    今天分享的题目来源于 LeetCode 第 287 号问题:寻找重复数. 题目描述 给定一个包含 n + 1 个整数的数组 nums,其数字都在 1 到 n 之间(包括 1 和 n),可知至少存在一个 ...

  7. MySQL学习(四)深入理解乐观锁与悲观锁

    转载自:http://www.hollischuang.com/archives/934 在数据库的锁机制中介绍过,数据库管理系统(DBMS)中的并发控制的任务是确保在多个事务同时存取数据库中同一数据 ...

  8. 关于Stream的知识分享

    一.什么是Stream 查了一下MSDN,他是这么解释的:提供字节序列的一般视图. 这个解释有点太笼统了,下面,我们来仔细的捋一下 1.什么是字节序列? 字节序列指的是:字节对象被存储为连续的字节序列 ...

  9. shell传递参数(三)

    $n:n代表一个数字,指执行脚本的第n个参数.特别地,$0指执行的文件名 [root@ipha-dev71- exercise_shell]# cat test.sh #!/bin/bash echo ...

  10. “无处不在” 的系统核心服务 —— ActivityManagerService 启动流程解析

    本文基于 Android 9.0 , 代码仓库地址 : android_9.0.0_r45 系列文章目录: Java 世界的盘古和女娲 -- Zygote Zygote 家的大儿子 -- System ...