這篇文章主要介紹了Java中fail-fast指的是什么意思,具有一定借鑒價(jià)值,感興趣的朋友可以參考下,希望大家閱讀完這篇文章之后大有收獲,下面讓小編帶著大家一起了解一下。
十載的龍馬潭網(wǎng)站建設(shè)經(jīng)驗(yàn),針對(duì)設(shè)計(jì)、前端、開(kāi)發(fā)、售后、文案、推廣等六對(duì)一服務(wù),響應(yīng)快,48小時(shí)及時(shí)工作處理。網(wǎng)絡(luò)營(yíng)銷(xiāo)推廣的優(yōu)勢(shì)是能夠根據(jù)用戶(hù)設(shè)備顯示端的尺寸不同,自動(dòng)調(diào)整龍馬潭建站的顯示方式,使網(wǎng)站能夠適用不同顯示終端,在瀏覽器中調(diào)整網(wǎng)站的寬度,無(wú)論在任何一種瀏覽器上瀏覽網(wǎng)站,都能展現(xiàn)優(yōu)雅布局與設(shè)計(jì),從而大程度地提升瀏覽體驗(yàn)。成都創(chuàng)新互聯(lián)從事“龍馬潭網(wǎng)站設(shè)計(jì)”,“龍馬潭網(wǎng)站推廣”以來(lái),每個(gè)客戶(hù)項(xiàng)目都認(rèn)真落實(shí)執(zhí)行。
01、前言
說(shuō)起來(lái)真特么慚愧:十年 IT 老兵,Java 菜鳥(niǎo)一枚。今天我才了解到 Java 還有 fail-fast 一說(shuō)。不得不感慨啊,學(xué)習(xí)真的是沒(méi)有止境。只要肯學(xué),就會(huì)有巨多巨多別人眼中的“舊”知識(shí)涌現(xiàn)出來(lái),并且在我這全是新的。
能怎么辦呢?除了羞愧,就只能趕緊全身心地投入學(xué)習(xí),把這些知識(shí)掌握。
為了鎮(zhèn)樓,必須搬一段英文來(lái)解釋一下 fail-fast。
In systems design, a fail-fast system is one which immediately reports at its interface any condition that is likely to indicate a failure. Fail-fast systems are usually designed to stop normal operation rather than attempt to continue a possibly flawed process. Such designs often check the system's state at several points in an operation, so any failures can be detected early. The responsibility of a fail-fast module is detecting errors, then letting the next-highest level of the system handle them.
大家不嫌棄的話(huà),我就用蹩腳的英語(yǔ)能力翻譯一下。某場(chǎng)戰(zhàn)役當(dāng)中,政委發(fā)現(xiàn)司令員在亂指揮的話(huà),就立馬報(bào)告給權(quán)限更高的中央軍委——這樣可以有效地避免更嚴(yán)重的后果出現(xiàn)。當(dāng)然了,如果司令員是李云龍的話(huà),報(bào)告也沒(méi)啥用。
不過(guò),Java 的世界里不存在李云龍。fail-fast 扮演的就是政委的角色,一旦報(bào)告給上級(jí),后面的行動(dòng)就別想執(zhí)行。
怎么和代碼關(guān)聯(lián)起來(lái)呢?看下面這段代碼。
public void test(Wanger wanger) { if (wanger == null) { throw new RuntimeException("wanger 不能為空"); } System.out.println(wanger.toString()); }
一旦檢測(cè)到 wanger 為 null,就立馬拋出異常,讓調(diào)用者來(lái)決定這種情況下該怎么處理,下一步 wanger.toString() 就不會(huì)執(zhí)行了——避免更嚴(yán)重的錯(cuò)誤出現(xiàn),這段代碼由于太過(guò)簡(jiǎn)單,體現(xiàn)不出來(lái),后面會(huì)講到。
瞧,fail-fast 就是這個(gè)鬼,沒(méi)什么神秘的。如果大家源碼看得比較多的話(huà),這種例子多得就像旅游高峰期的人頭。
然后呢,沒(méi)了?三秒鐘,別著急,我們繼續(xù)。
02、for each 中集合的 remove 操作
很長(zhǎng)一段時(shí)間里,我都不明白為什么不能在 for each 循環(huán)里進(jìn)行元素的 remove。今天我們就來(lái)借機(jī)來(lái)體驗(yàn)一把。
Listlist = new ArrayList<>(); list.add("沉默王二"); list.add("沉默王三"); list.add("一個(gè)文章真特么有趣的程序員"); for (String str : list) { if ("沉默王二".equals(str)) { list.remove(str); } } System.out.println(list);
這段代碼看起來(lái)沒(méi)有任何問(wèn)題,但運(yùn)行起來(lái)就糟糕了。
Exception in thread "main" java.util.ConcurrentModificationException at java.util.ArrayList$Itr.checkForComodification(ArrayList.java:909) at java.util.ArrayList$Itr.next(ArrayList.java:859) at com.cmower.java_demo.str.Cmower3.main(Cmower3.java:14)
為毛呢?
03、分析問(wèn)題的殺手锏
這時(shí)候就只能看源碼了,ArrayList.java 的 909 行代碼是這樣的。
final void checkForComodification() { if (modCount != expectedModCount) throw new ConcurrentModificationException(); }
也就是說(shuō),remove 的時(shí)候執(zhí)行了 checkForComodification 方法,該方法對(duì) modCount 和 expectedModCount 進(jìn)行了比較,發(fā)現(xiàn)兩者不等,就拋出了 ConcurrentModificationException 異常。
可為什么會(huì)執(zhí)行 checkForComodification 方法呢?這就需要反編譯一下 for each 那段代碼了。
Listlist = new ArrayList(); list.add("沉默王二"); list.add("沉默王三"); list.add("一個(gè)文章真特么有趣的程序員"); Iterator var3 = list.iterator(); while (var3.hasNext()) { String str = (String) var3.next(); if ("沉默王二".equals(str)) { list.remove(str); } } System.out.println(list);
原來(lái) for each 是通過(guò)迭代器 Iterator 配合 while 循環(huán)實(shí)現(xiàn)的。
1)ArrayList.iterator() 返回的 Iterator 其實(shí)是 ArrayList 的一個(gè)內(nèi)部類(lèi) Itr。
public Iteratoriterator() { return new Itr(); }
Itr 實(shí)現(xiàn)了 Iterator 接口。
private class Itr implements Iterator{ int cursor; // index of next element to return int lastRet = -1; // index of last element returned; -1 if no such int expectedModCount = modCount; Itr() {} public boolean hasNext() { return cursor != size; } @SuppressWarnings("unchecked") public E next() { checkForComodification(); int i = cursor; Object[] elementData = ArrayList.this.elementData; if (i >= elementData.length) throw new ConcurrentModificationException(); cursor = i + 1; return (E) elementData[lastRet = i]; } }
也就是說(shuō) new Itr() 的時(shí)候 expectedModCount 被賦值為 modCount,而 modCount 是 List 的一個(gè)成員變量,表示集合被修改的次數(shù)。由于 list 此前執(zhí)行了 3 次 add 方法,所以 modCount 的值為 3;expectedModCount 的值也為 3。
可當(dāng)執(zhí)行 list.remove(str) 后,modCount 的值變成了 4。
private void fastRemove(int index) { modCount++; int numMoved = size - index - 1; if (numMoved > 0) System.arraycopy(elementData, index+1, elementData, index, numMoved); elementData[--size] = null; // clear to let GC do its work }
注:remove 方法內(nèi)部調(diào)用了 fastRemove 方法。
下一次循環(huán)執(zhí)行到 String str = (String) var3.next(); 的時(shí)候,就會(huì)調(diào)用 checkForComodification 方法,此時(shí)一個(gè)為 3,一個(gè)為
4,就只好拋出異常 ConcurrentModificationException 了。
不信,可以直接在 ArrayList 類(lèi)的 909 行打個(gè)斷點(diǎn) debug 一下。
真的耶,一個(gè)是 4 一個(gè)是 3。
總結(jié)一下。在 for each 循環(huán)中,集合遍歷其實(shí)是通過(guò)迭代器 Iterator 配合 while 循環(huán)實(shí)現(xiàn)的,但是元素的 remove 卻直接使用的集合類(lèi)自身的方法。這就導(dǎo)致 Iterator 在遍歷的時(shí)候,會(huì)發(fā)現(xiàn)元素在自己不知情的情況下被修改了,它覺(jué)得很難接受,就拋出了異常。
讀者朋友們,你們是不是覺(jué)得我跑題了,fail-fast 和 for each 中集合的 remove 操作有什么關(guān)系呢?
有!Iterator 使用了 fail-fast 的保護(hù)機(jī)制。
04、怎么避開(kāi) fail-fast 保護(hù)機(jī)制呢
通過(guò)上面的分析,相信大家都明白為什么不能在 for each 循環(huán)里進(jìn)行元素的 remove 了。
那怎么避開(kāi) fail-fast 保護(hù)機(jī)制呢?畢竟刪除元素是常規(guī)操作,咱不能因噎廢食啊。
1)remove 后 break
Listlist = new ArrayList<>(); list.add("沉默王二"); list.add("沉默王三"); list.add("一個(gè)文章真特么有趣的程序員"); for (String str : list) { if ("沉默王二".equals(str)) { list.remove(str); break; } }
我怎么這么聰明,忍不住驕傲一下。有讀者不明白為什么嗎?那我上面的源碼分析可就白分析了,爬樓再看一遍吧!
略微透露一下原因:break 后循環(huán)就不再遍歷了,意味著 Iterator 的 next 方法不再執(zhí)行了,也就意味著 checkForComodification 方法不再執(zhí)行了,所以異常也就不會(huì)拋出了。
但是呢,當(dāng) List 中有重復(fù)元素要?jiǎng)h除的時(shí)候,break 就不合適了。
2)for 循環(huán)
Listlist = new ArrayList<>(); list.add("沉默王二"); list.add("沉默王三"); list.add("一個(gè)文章真特么有趣的程序員"); for (int i = 0, n = list.size(); i < n; i++) { String str = list.get(i); if ("沉默王二".equals(str)) { list.remove(str); } }
for 循環(huán)雖然可以避開(kāi) fail-fast 保護(hù)機(jī)制,也就說(shuō) remove 元素后不再拋出異常;但是呢,這段程序在原則上是有問(wèn)題的。為什么呢?
第一次循環(huán)的時(shí)候,i 為 0,list.size() 為 3,當(dāng)執(zhí)行完 remove 方法后,i 為 1,list.size() 卻變成了 2,因?yàn)?list 的大小在 remove 后發(fā)生了變化,也就意味著“沉默王三”這個(gè)元素被跳過(guò)了。能明白嗎?
remove 之前 list.get(1) 為“沉默王三”;但 remove 之后 list.get(1) 變成了“一個(gè)文章真特么有趣的程序員”,而 list.get(0) 變成了“沉默王三”。
3)Iterator
Listlist = new ArrayList<>(); list.add("沉默王二"); list.add("沉默王三"); list.add("一個(gè)文章真特么有趣的程序員"); Iterator itr = list.iterator(); while (itr.hasNext()) { String str = itr.next(); if ("沉默王二".equals(str)) { itr.remove(); } }
為什么使用 Iterator 的 remove 方法就可以避開(kāi) fail-fast 保護(hù)機(jī)制呢?看一下 remove 的源碼就明白了。
public void remove() { if (lastRet < 0) throw new IllegalStateException(); checkForComodification(); try { ArrayList.this.remove(lastRet); cursor = lastRet; lastRet = -1; expectedModCount = modCount; } catch (IndexOutOfBoundsException ex) { throw new ConcurrentModificationException(); } }
雖然刪除元素依然使用的是 ArrayList 的 remove 方法,但是刪除完會(huì)執(zhí)行 expectedModCount = modCount,保證了 expectedModCount 與 modCount 的同步。
05、最后
在 Java 中,fail-fast 從狹義上講是針對(duì)多線(xiàn)程情況下的集合迭代器而言的。這一點(diǎn)可以從 ConcurrentModificationException 定義上看得出來(lái)。
This exception may be thrown by methods that have detected concurrent
modification of an object when such modification is not permissible.For example, it is not generally permissible for one thread to modify a Collectionwhile another thread is iterating over it. In general, the results of theiteration are undefined under these circumstances. Some Iteratorimplementations (including those of all the general purpose collection implementationsprovided by the JRE) may choose to throw this exception if this behavior isdetected. Iterators that do this are known as fail-fast iterators,as they fail quickly and cleanly, rather that risking arbitrary,non-deterministic behavior at an undetermined time in the future.
再次拙劣地翻譯一下。
該異常可能由于檢測(cè)到對(duì)象在并發(fā)情況下被修改而拋出的,而這種修改是不允許的。
通常,這種操作是不允許的,比如說(shuō)一個(gè)線(xiàn)程在修改集合,而另一個(gè)線(xiàn)程在迭代它。這種情況下,迭代的結(jié)果是不確定的。如果檢測(cè)到這種行為,一些 Iterator(比如說(shuō) ArrayList 的內(nèi)部類(lèi) Itr)就會(huì)選擇拋出該異常。這樣的迭代器被稱(chēng)為 fail-fast 迭代器,因?yàn)楸M早的失敗比未來(lái)出現(xiàn)不確定的風(fēng)險(xiǎn)更好。
既然是針對(duì)多線(xiàn)程,為什么我們之前的分析都是基于單線(xiàn)程的呢?因?yàn)閺膹V義上講,fail-fast 指的是當(dāng)有異常或者錯(cuò)誤發(fā)生時(shí)就立即中斷執(zhí)行的這種設(shè)計(jì),從單線(xiàn)程的角度去分析,大家更容易明白。
1.SpringMVC,Spring Web MVC是一種基于Java的實(shí)現(xiàn)了Web MVC設(shè)計(jì)模式的請(qǐng)求驅(qū)動(dòng)類(lèi)型的輕量級(jí)Web框架。2.Shiro,Apache Shiro是Java的一個(gè)安全框架。3.Mybatis,MyBatis 是支持普通 SQL查詢(xún),存儲(chǔ)過(guò)程和高級(jí)映射的優(yōu)秀持久層框架。4.Dubbo,Dubbo是一個(gè)分布式服務(wù)框架。5.Maven,Maven是個(gè)項(xiàng)目管理和構(gòu)建自動(dòng)化工具。6.RabbitMQ,RabbitMQ是用Erlang實(shí)現(xiàn)的一個(gè)高并發(fā)高可靠AMQP消息隊(duì)列服務(wù)器。7.Ehcache,EhCache 是一個(gè)純Java的進(jìn)程內(nèi)緩存框架。
感謝你能夠認(rèn)真閱讀完這篇文章,希望小編分享的“Java中fail-fast指的是什么意思”這篇文章對(duì)大家有幫助,同時(shí)也希望大家多多支持創(chuàng)新互聯(lián),關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道,更多相關(guān)知識(shí)等著你來(lái)學(xué)習(xí)!