ConcurrentModificationException源码逐条解析

一家之言 姑妄言之 絮絮叨叨 不足为训

ConcurrentModificationException类注释翻译:

   当不允许修改时,已经检测到对象并发修改的方法可能会抛出此异常。
   例如,当一个线程在集合上迭代时,通常不允许另一个线程修改集合。一般来说,迭代的结果在这种情况下是没有定义的。如果检测到上述行为,某些迭代器实现(包括JRE提供的所有通用集合实现)可能会选择抛出此异常。这样做的迭代器被称为故障快速fail-fast迭代器,因为它们快速而干净地失败了,而不是在将来某个不确定的时间冒着任意的、不确定的行为的风险。
   请注意,此异常并不总是表示对象已被其他线程并发修改。如果单个线程发出的一系列方法调用违反了对象的契约,则对象可能会抛出此异常。例如,如果线程在使用故障快速fail-fast迭代器遍历集合时直接修改集合,迭代器将抛出此异常。
   注意,快速故障行为fail-fast不能得到保证,因为通常来说,在非同步并发修改的情况下,不可能做出任何严格的保证。快速故障操作会在最大努力的基础上抛出ConcurrentModificationException异常。因此,编写一个依赖于这个异常来保证其正确性的程序是错误的:ConcurrentModificationException应该只用于检测bug。


笔者废话:

   其实这个异常类的注释已经写得很不错了,但是由于笔者翻译的问题(不敢加入个人色彩太多以避免文意曲解)导致可能看着有些拗口。
   实际上这个异常就是文案上的含义:
   1. 多线程时,发生并发修改时,这个异常会抛出;
   2. 单线程时,在遍历过程中修改了集合,这个异常也会抛出;
   3. 我们对这个异常是否可以正确抛出是不做严格保证的。因为这种机制是谁都不可能严格地去说,对,这里一定会抛出这个异常。所以,这里是非常不建议我们的程序依赖这个fail-fast机制来编写程序。它最佳的用处也就是检测bug,而不是依此来进行复杂业务处理。


ConcurrentModificationException类信息:

public class ConcurrentModificationException extends RuntimeException

   我们可以清楚的看到,ConcurrentModificationException 一个是继承自RuntimeException类的异常类。


ConcurrentModificationException静态变量信息:

/* 序列化版本编号 */
private static final long serialVersionUID = -3666751008965953603L;

   我们来看ConcurrentModificationException的静态变量信息。
   这里的静态变量信息唯独就是生成了一个序列化版本编号。这个类的整体集成方式是这样的:ConcurrentModificationException->RuntimeException->Exception->Throwable。而Throwable实现了java.io.Serializable接口。所以这里有一个序列化版本编号也不足为奇。


ConcurrentModificationException构造函数信息:

/**
 * 构造一个没有详细信息的ConcurrentModificationException异常。
 */
public ConcurrentModificationException() {
}

/**
 * 构造一个指定详细信息的ConcurrentModificationException异常。
 * @param message 与此异常相关的详细信息。
 */
public ConcurrentModificationException(String message) {
    super(message);
}

/**
 * 构造一个带有指定原因和详细信息
 * (cause==null ? null : cause.toString())
 * 的ConcurrentModificationException异常(其中通常包含类和指定
 * 原因的详细信息)。
 * @param  cause 错误原因(该原因将被Throwable.getCause()方法
 * 保存,供以后检索)(允许空值,并指示原因不存在或未知。)
 * @since  1.7
 */
public ConcurrentModificationException(Throwable cause) {
    super(cause);
}

/**
 * 构造一个带有指定原因和详细信息的ConcurrentModificationException异常
 * 请注意,与cause相关的详细信息不会自动合并到此异常的详细信息中。
 * @param  message 详细信息(它被保存起来供以后使用
 * Throwable.getMessage()方法检索)。
 * @param  cause 原因(该原因将被Throwable.getCause()方法保存,供以后检
 * 索)。(允许为空值,并指示原因不存在或未知。)
 * @since 1.7
 */
public ConcurrentModificationException(String message, Throwable cause) {
    super(message, cause);
}

   其实到这里你会发现整个类就已经结束了。对的,这个异常类的所有方法都是其本身的构造方法,是没有其他业务方法的。所以这里一连展示了其4个构造方法,有参的,无参的。同时,这里也一并对其介绍而不需要一个一个的讲解。
   为什么不需要一个一个讲解呢,我们仔细从代码体得知,这些构造方法无不在(除了无参构造器)都在调用其父类的构造方法。而这个父类你都可以追踪到它的祖宗辈儿Throwable类。
   我们整体来看这四个构造方法,其中第二个构造函数是第一个构造函数的扩展体,第四个构造函数是第三个构造函数的扩展体(这两个函数都是从Java1.7才开始有的)。
   那么,其实我们去搜寻现有的Java1.8源码包,也就第一无参构造器和第三有参构造器是被引用的。其中第一无参构造器被引用100次,第三有参构造器仅被引用4次(其中Map接口引用三次,DirectoryIteratorException异常类引用一次)。整个异常类都是在进行错误信息的收集。最主要的关注点其实还是它是如何被触发的,不过它的触发时机已经在上面简述过了,我们只需简单的知道这个类的大体构造就可以了~
   其实到这里我们ConcurrentModificationException到此全部解析完毕了,可能整篇文章显得无所重要,其实不然。因为这整个专题的部分源码分析都会涉及到此异常。这个解析已经是这个类可以提供的全部信息了,其他你想要的解析其实是需要具体到引用其类的源码上的。
   ConcurrentModificationException源码所提供的信息也就到此为止了(ಥ_ಥ)。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,826评论 6 506
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,968评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,234评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,562评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,611评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,482评论 1 302
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,271评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,166评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,608评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,814评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,926评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,644评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,249评论 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,866评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,991评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,063评论 3 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,871评论 2 354

推荐阅读更多精彩内容

  • java基础 集合承继包含图 Collection vs Collections 首先,"Collection" ...
    onlyHalfSoul阅读 1,318评论 0 5
  • Java源码研究之容器(1) 如何看源码 很多时候我们看源码, 看完了以后经常也没啥收获, 有些地方看得懂, 有些...
    骆驼骑士阅读 994评论 0 22
  • 面向对象的三个特征 封装,继承,多态.这个应该是人人皆知.有时候也会加上抽象. 多态的好处 允许不同类对象对同一消...
    Blizzard_liu阅读 1,327评论 0 6
  • java笔记第一天 == 和 equals ==比较的比较的是两个变量的值是否相等,对于引用型变量表示的是两个变量...
    jmychou阅读 1,497评论 0 3
  • 今天是开学的第二天
    史德娟阅读 178评论 0 0