OpenJDK源码研究笔记(十五):吐槽JDK中的10个富有争议的设计
前14篇文章,分享了JDK中值得学习和借鉴的编码和设计方法。
每个硬币都是有两面的。Every coin has two sides。
当然,JDK中也有很多值得改进或者说富有争议的设计。
本篇,就来详细吐槽下JDK中的10个富有争议的设计。
网友指出
1.求和抛异常,错误提示不合理。
在OpenJDK源码研究笔记(一)-参数检查&抛出带关键错误提示信息的异常这篇文章中,有个网友指出:“第三个if的提示信息有点问题。前两个if说明了position和size都不可能为负,求和通常不可能为负,只有一种情况是 求和 溢出了吧,此时异常信息再这样抛,反而是信息不明确”
protected FileLock(AsynchronousFileChannel channel, long position,
long size, boolean shared) {
if (position < 0)
throw new IllegalArgumentException("Negative position");
if (size < 0)
throw new IllegalArgumentException("Negative size");
if (position + size < 0)
throw new IllegalArgumentException("Negative position + size");
this.channel = channel;
this.position = position;
this.size = size;
this.shared = shared;
}
2.标记接口可以用注解替代。
在OpenJDK源码研究笔记(三)-RandomAccess等标记接口的作用这篇文章中,有个网友指出:“空接口,比如 Serializable 之类的,都只是一个记号, 其实用annotation 更合适。”
比如ArrayList的定义是 class ArrayList implements Serialiable{}。
可以修改为@Serialize
class ArrayList{}
几个观点
a.注解是JDK1.5才有的。之前的API当然还是应该支持。
注解的话,我觉得也是可以的。可能看起来不够简洁。
实现一个空的接口,用instanceof挺方便的。
b.实现接口也好,用注解也好。最终,都是一种设计的2种不同实现。
实现接口用Instance of,用注解 调用方法。
用接口能够更好地兼容以前的代码。
c.从兼容的角度看,确实是的。
但是从设计目的来看,注解更好。
而且不影响对象继承结构,就是加了个标记。
方法定义
3.重复定义方法。
public interface List<E> extends Collection<E>{ int size();
boolean isEmpty(); }
既然继承了父接口Collection,为何还要 重复去定义父类中已经存在的接口??
public interface Set<E> extends Collection<E>等接口的定义 也是如此。
4.双端队列同一种操作有2种类似的实现。
java.util.Deque<E>
/**
* 查询队列中的第1个元素。
* 当队列为空时,返回null。 */
E peekFirst(); /**
* 查询队列中的第1个元素。
* 与peekFirst方法不同的是,当队列为空时,抛出异常。
* @throws NoSuchElementException if this deque is empty
*/
E getFirst();
为什么要定义2套API呢?显得有些重复,容易让人产生疑惑。
5.集合接口Set的返回值不同。
java.util.Set public abstract E get(int index);
public E set(int index, E element); public boolean add(E e);
为什么就这个方法没有返回值呢?
public void add(int index, E element) ;
上面的3个方法都是有返回值的,同样是add(E e)也是有boolean返回值。
设计问题
6.List生成子列表后,就不能再操作原列表。
List<E> subList(int fromIndex, int toIndex);
List<String> list = new ArrayList<String>(); list.add("A"); list.add("B"); List<String> subList = list.subList(0,2); subList.add("C");
以上代码会抛出java.util.ConcurrentModificationException
有的时候,我们确实有这个需求。
既然JDK这样设计,我们只能尽可能避开上述用法。
7.不可变集合的实现方式。
java.util.Collections.unmodifiableList(list);
这种方法的实现,其实返回一个“不可变”的List实现。
static class UnmodifiableList<E> extends UnmodifiableCollection<E>
implements List<E> {
private static final long serialVersionUID = -283967356065247728L;
final List<? extends E> list; UnmodifiableList(List<? extends E> list) {
super(list);
this.list = list;
} public E get(int index) {
return list.get(index);
} public E set(int index, E element) {
throw new UnsupportedOperationException();
} public void add(int index, E element) {
throw new UnsupportedOperationException();
} public E remove(int index) {
throw new UnsupportedOperationException();
} }
把那些会修改List的方法,以抛出异常的方式实现。
在看源码之前,我总觉得有其它方法实现。
看到源码是这种方式实现的,我有点失望。
看来是我想多了。
8.参数检查不合理。
java.util.logging.Handler.setFormatter(Formatter) public void setFormatter(Formatter newFormatter) throws SecurityException {
checkAccess();
// Check for a null pointer:
newFormatter.getClass();
formatter = newFormatter;
}
就是简单地调用一个类的方法newFormatter.getClass();来检查该类是否为null。
这种方式是否更好呢?
if(newFormatter==null){
throw new NullPointerException("");
}
编码习惯
9.一条If语句的括号问题。
大部分的只有一条语句的if语句是这么写的。
public E getFirst() {
final Node<E> f = first;
if (f == null)
throw new NoSuchElementException();
return f.item;
}
我比较习惯以下写法,防止“看走眼”。
if (f == null){
throw new NoSuchElementException();
}
另外,我经常会写下面这样的注释,
if (f == null)
//抛出异常,或者加一条打印语句
throw new NoSuchElementException();
这种写法,很容易看错。
这种类似的代码有很多,如果if语句再嵌套一层,就更容易看错了。
public ScriptEngineManager() {
ClassLoader ctxtLoader = Thread.currentThread().getContextClassLoader();
if (canCallerAccessLoader(ctxtLoader)) {
if (DEBUG)
System.out.println("using " + ctxtLoader);
init(ctxtLoader);
} else {
if (DEBUG)
System.out.println("using bootstrap loader");
init(null);
}
}
10.没有用的变量。
java.util.logging.LogManager中有如下变量定义。
private final static Handler[] emptyHandlers = {};
上面的代码是没有用到的。
Eclipse会有警告,黄色的感叹号还是比较烦人的。
相关阅读
OpenJDK源码研究过程中整理的学习笔记。 OpenJDK是GPL许可(GPL-licensed)的Java平台的开源实现。
原文参见:http://FansUnion.cn/articles/3174(小雷网-FansUnion.cn)
OpenJDK源码研究笔记(十五):吐槽JDK中的10个富有争议的设计的更多相关文章
- OpenJDK源码研究笔记(十六):在Java中使用JavaScript脚本语言
友情提示 本文主要参考了51CTO上的一篇文章,代码经过自己的模仿和整理,还算凑合. 本文中的代码注释比较多,不再过多解释. 更多用法,还是得看JDK的API或者看原文http://developer ...
- OpenJDK源码研究笔记(十):枚举的高级用法,枚举实现接口,竟是别有洞天
在研究OpenJDK,Java编译器javac源码的过程中,发现以下代码. 顿时发现枚举类竟然也有如此"高端大气上档次"的用法. 沙场点兵(用法源码) com.sun.tools. ...
- OpenJDK源码研究笔记(十二):JDBC中的元数据,数据库元数据(DatabaseMetaData),参数元数据(ParameterMetaData),结果集元数据(ResultSetMetaDa
元数据最本质.最抽象的定义为:data about data (关于数据的数据).它是一种广泛存在的现象,在许多领域有其具体的定义和应用. JDBC中的元数据,有数据库元数据(DatabaseMeta ...
- OpenJDK源码研究笔记(十三):Javac编译过程中的上下文容器(Context)、单例(Singleton)和延迟创建(LazyCreation)3种模式
在阅读Javac源码的过程中,发现一个上下文对象Context. 这个对象用来确保一次编译过程中的用到的类都只有一个实例,即实现我们经常提到的"单例模式". 今天,特意对这个上下文 ...
- OpenJDK源码研究笔记(十四):三种经典的设计方法,接口,接口-抽象类-具体实现类,接口-具体实现类
在研究OpenJDK源码过程中,我发现常用的设计方法就是2种:接口,接口-抽象类-具体实现类 . 在一些其它开源框架和业务开发中,经常存在着第3种设计,接口-具体实现类. 1.只有接口,没有实现类. ...
- ABP源码分析十五:ABP中的实用扩展方法
类名 扩展的类型 方法名 参数 作用 XmlNodeExtensions XmlNode GetAttributeValueOrNull attributeName Gets an attribu ...
- OpenJDK源码研究笔记(五)-缓存Integer等类型的频繁使用的数据和对象,大幅度提升性能(一道经典的Java笔试题)
摘要 本文先给出一个看似很简单实则有深意的Java笔试面试题,引出JDK内部的缓存. JDK内部的缓存,主要是为了提高Java程序的性能. 你能答对这道"看似简单,实则有深意"的J ...
- [置顶] OpenJDK源码研究笔记(九)-可恨却又可亲的的异常(NullPointerException)
可恨的异常 程序开发过程中,最讨厌异常了. 异常代表着程序出了问题,一旦出现,控制台会出现一屏又一屏的堆栈错误信息. 看着就让人心烦. 对于一个新人来讲,遇到异常经常会压力大,手忙脚乱,心生畏惧. 可 ...
- OpenJDK源码研究笔记(四)-编写和组织可复用的工具类和方法
本篇主要讲解java.util.Arrays这个针对数组的工具类. 1.可复用的工具类和方法. 这个工具类里,包含很多针对数组的工具方法,如 排序.交换.二分查找.比较.填充.复制.hashcode ...
随机推荐
- Android 提示: The connection to adb is down, and a severe error has occured.
今天早上打开Eclipse,一直提示 The connection to adb is down, and a severe error has occured,无法执行程序.重新启动Eclipse. ...
- bzoj1070: [SCOI2007]修车(费用流)
1070: [SCOI2007]修车 题目:传送门 题解: 一道挺简单的费用流吧...胡乱建模走起 贴个代码... #include<cstdio> #include<cstring ...
- 设置和获取Android中各种音量
通过程序获取android系统手机的铃声和音量.同样,设置铃声和音量的方法也很简单! AudioManager am = (AudioManager) getSystemService(Context ...
- C#中使用Dictionary实现Map数据结构——VC编程网
转载自: http://blog.51cto.com/psnx168 在VC中使用过CMap以及在Java中使用过Map的朋友应该很熟悉,使用Map可以方便实现基于键值对数据的处理,在C#中,你就需要 ...
- Windows10通过VNC远程连接Ubuntu18.04
1.打开终端输入:sudo apt-get install xrdp vnc4server xbase-clients dconf-editor 2.接着在终端输入: 进入到下面这个界面: 接着按照这 ...
- Windows 10 计划带来颜文字和Sandbox
在最新的 Window 10 测试版 Build 18305 中,Windows 10 增加了对颜文字(kaomoji)的支持. Kaomoji 是由日本符号序列组成的面脸部表情的名称.虽然有些人,比 ...
- 认识Vue组件
前言 Vue.js是一套构建用户界面的渐进式框架(官方说明).通俗点来说,Vue.js是一个轻量级的,易上手易使用的,便捷,灵活性强的前端MVVM框架.简洁的API,良好健全的中文文档,使开发者能够较 ...
- 一个Web报表项目的性能分析和优化实践(二):MySQL数据库连接不够用(TooManyConnections)问题的一次分析和解决案例
最近,项目中遇到了数据库连接不够的问题. 异常信息com.mysql.jdbc.exceptions.jdbc4.MySQLNonTransientConnectionException: Data ...
- 【Henu ACM Round#20 D】 Devu and Partitioning of the Array
[链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 一开始所有的数字单独成一个集合. 然后用v[0]和v[1]记录集合的和为偶数和奇数的集合它们的根节点(并查集 然后先让v[0]的大小 ...
- Java基础学习总结(19)——Java环境变量配置
前言 学习java的第一步就要搭建java的学习环境,首先是要安装JDK,JDK安装好之后,还需要在电脑上配置"JAVA_HOME"."path"." ...