畢業(yè)設(shè)計外文文獻(xiàn)翻譯---java垃圾收集器的工作方式_第1頁
已閱讀1頁,還剩9頁未讀, 繼續(xù)免費閱讀

下載本文檔

版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請進(jìn)行舉報或認(rèn)領(lǐng)

文檔簡介

1、<p><b>  畢業(yè)設(shè)計(論文)</b></p><p><b>  外文文獻(xiàn)翻譯</b></p><p><b>  信息工程學(xué)院</b></p><p><b>  英文原文</b></p><p>  How a garbage col

2、lector works of Java Language</p><p>  If you come from a programming language where allocating objects on the heap is expensive, you may naturally assume that Java’s scheme of allocating everything (except

3、primitives) on the heap is also expensive. However, it turns out that the garbage collector can have a significant impact on increasing the speed of object creation. This might sound a bit odd at first—that storage relea

4、se affects storage allocation—but it’s the way some JVMs work, and it means that allocating storage for heap obj</p><p>  For example, you can think of the C++ heap as a yard where each stakes out its own pi

5、ece of turf object. This real estate can become abandoned sometime later and must be reused. In some JVMs, the Java heap is quite different; it’s more like a conveyor belt that moves forward every time you allocate a new

6、 object. This means that object storage allocation is remarkably rapid. The “heap pointer” is simply moved forward into virgin territory, so it’s effectively the same as C++’s stack allocation. (</p><p>  Yo

7、u might observe that the heap isn’t in fact a conveyor belt, and if you treat it that way, you’ll start paging memory—moving it on and off disk, so that you can appear to have more memory than you actually do. Paging sig

8、nificantly impacts performance. Eventually, after you create enough objects, you’ll run out of memory. The trick is that the garbage collector steps in, and while it collects the garbage it compacts all the objects in th

9、e heap so that you’ve effectively moved the “heap pointer”</p><p>  To understand garbage collection in Java, it’s helpful learn how garbage-collection schemes work in other systems. A simple but slow garbag

10、e-collection technique is called reference counting. This means that each object contains a reference counter, and every time a reference is attached to that object, the reference count is increased. Every time a referen

11、ce goes out of scope or is set to null, the reference count is decreased. Thus, managing reference counts is a small but constant overhead th</p><p>  In faster schemes, garbage collection is not based on re

12、ference counting. Instead, it is based on the idea that any non-dead object must ultimately be traceable back to a reference that lives either on the stack or in static storage. The chain might go through several layers

13、of objects. Thus, if you start in the stack and in the static storage area and walk through all the references, you’ll find all the live objects. For each reference that you find, you must trace into the object that it p

14、oint</p><p>  In the approach described here, the JVM uses an adaptive garbage-collection scheme, and what it does with the live objects that it locates depends on the variant currently being used. One of th

15、ese variants is stop-and-copy. This means that—for reasons that will become apparent—the program is first stopped (this is not a background collection scheme). Then, each live object is copied from one heap to another, l

16、eaving behind all the garbage. In addition, as the objects are copied into the new heap</p><p>  Of course, when an object is moved from one place to another, all references that point at the object must be

17、changed. The reference that goes from the heap or the static storage area to the object can be changed right away, but there can be other references pointing to this object Initialization & Cleanup that will be encou

18、ntered later during the “walk.” These are fixed up as they are found (you could imagine a table that maps old addresses to new ones).</p><p>  There are two issues that make these so-called “copy collectors”

19、 inefficient. The first is the idea that you have two heaps and you slosh all the memory back and forth between these two separate heaps, maintaining twice as much memory as you actually need. Some JVMs deal with this by

20、 allocating the heap in chunks as needed and simply copying from one chunk to another.</p><p>  The second issue is the copying process itself. Once your program becomes stable, it might be generating little

21、 or no garbage. Despite that, a copy collector will still copy all the memory from one place to another, which is wasteful. To prevent this, some JVMs detect that no new garbage is being generated and switch to a differe

22、nt scheme (this is the “adaptive” part). This other scheme is called mark-and-sweep, and it’s what earlier versions of Sun’s JVM used all the time. For general use, mark</p><p>  However, each time it finds

23、a live object, that object is marked by setting a flag in it, but the object isn’t collected yet. Only when the marking process is finished does the sweep occur. During the sweep, the dead objects are released. However,

24、no copying happens, so if the collector chooses to compact a fragmented heap, it does so by shuffling objects around. “Stop-and-copy” refers to the idea that this type of garbage collection is not done in the background;

25、 Instead, the program is stopped</p><p>  As previously mentioned, in the JVM described here memory is allocated in big blocks. If you allocate a large object, it gets its own block. Strict stop-and-copy req

26、uires copying every live object from the source heap to a new heap before you can free the old one, which translates to lots of memory. With blocks, the garbage collection can typically copy objects to dead blocks as it

27、collects. Each block has a generation count to keep track of whether it’s alive. In the normal case, only the block</p><p>  The JVM monitors the efficiency of garbage collection and if it becomes a waste of

28、 time because all objects are long-lived, then it switches to mark-and sweep. Similarly, the JVM keeps track of how successful mark-and-sweep is, and if the heap starts to become fragmented, it switches back to stop-and-

29、copy. This is where the “adaptive” part comes in, so you end up with a mouthful: “Adaptive generational stop-and-copy mark-and sweep.”</p><p>  There are a number of additional speedups possible in a JVM. An

30、 especially important one involves the operation of the loader and what is called a just-in-time (JIT) compiler. A JIT compiler partially or fully converts a program into native machine code so that it doesn’t need to be

31、 interpreted by the JVM and thus runs much faster. When a class must be loaded (typically, the first time you want to create an object of that class), the .class file is located, and the byte codes for that class are <

32、;/p><p><b>  中文譯文</b></p><p>  Java垃圾收集器的工作方式 </p><p>  如果你學(xué)下過一種因為在堆里分配對象所以開銷過大的編程語言,很自然你可能會假定 Java 在堆里為每一樣?xùn)|西(除了 primitives)分配內(nèi)存資源的機(jī)制開銷也會很大。不過,事實上垃圾收集器能夠深刻影響對象的加速創(chuàng)建。 一開始聽起來

33、有些奇怪——存貯空間的釋放會影響存貯空間的分配,但是這的確是一些 JVMs 的工作方式,并且這意味著 Java 為堆對象分配存貯空間幾乎和別的語言里為棧分配存貯空間一樣地快。</p><p>  舉個例子,你可以認(rèn)為 C++的堆就如同一個堆放的工場,在這個工場里,每一個對象都立有的地皮占有權(quán)不久會被廢除無效,并且這塊地皮必須重新加以利用。在Java 的 JVM 里,堆的工作方式完全不同;每次為一個新的對象分配存貯

34、空間的時候,它就更像是一個不斷向前移動的傳送帶。 這就意味著對象存貯空間的分配速度明顯加快。 在這個過程中,“堆指針”簡單地向還沒被占用的空間領(lǐng)域移動,所以非常像 C++里棧的分配方式。(當(dāng)然,記錄工作會有一點額外的開銷,但是完全不同于 C++里那種在堆放工場里為尋找沒被利用的存貯空間而付出的開銷。)</p><p>  你或許觀察到實際上堆本身并不是一個傳送帶,如果你真的那樣看待堆,你就會啟用虛擬內(nèi)存——在硬盤

35、里不斷地裝卸,結(jié)果是看上去你會擁有比實際情況還要多的內(nèi)存空間。 最終,當(dāng)你創(chuàng)建了足夠多的對象后,你會耗盡內(nèi)存。 Java 的訣竅就在于垃圾搜集器插手于其中,當(dāng)垃圾收集器收集垃圾的時候,它會壓縮所有堆里的對象以便你能夠有效的將堆指針移動到相應(yīng)的地方從而遠(yuǎn)離了頁面錯誤。垃圾收集器重新安排了整個過程,這使得分配存貯空間的時候一種高速,無窮閑置的堆模式成為可能。</p><p>  要想理解 Java 的垃圾收集工作,先

36、了解一下別的語言系統(tǒng)里垃圾收集所使用的方案是有幫助的。 一種簡單的但卻較慢的垃圾收集技術(shù)就是引用記數(shù)(reference counting)。這種技術(shù)意味著每個對象都含有一個引用計數(shù)器,每一次一個引用指向那個對象的時候,引用記數(shù)就增加1 每一次對象引用離開作用域或者被設(shè)置為 null 的時候,引用記數(shù)就減1。 因此,應(yīng)付對象被引用的數(shù)量在你的程序的整個生命周期里是一筆較小但卻一直持續(xù)的開銷。垃圾收集器歷遍整組對象,當(dāng)它發(fā)現(xiàn)一個引用記數(shù)為

37、零的對象時就會釋放那個對象的存貯空間。(不過,只要記數(shù)為零,引用記數(shù)方案通常會立刻釋放對象)。 這種方案的一個缺點是如果對象之間循環(huán)著互相引用,那么這些對象的引用記數(shù)可能為非零,而垃圾收集器依然把它們當(dāng)作垃圾收集。 定位這種自我引用的對象組需要垃圾收集器付出大量額外的工作。引用記數(shù)通常被用來解釋一類垃圾收集的工作原理,但是它似乎沒被任何一種 JVM 所采納。有一種執(zhí)行更快的垃圾收集方案,這種方案中垃圾收集不是建立在引用記數(shù)的基礎(chǔ)上。&l

38、t;/p><p>  相反,它的思想是任何沒死的對象最終一定會在棧和靜態(tài)存貯器里找到相應(yīng)存活的引用。這種鏈?zhǔn)降牟檎曳绞娇赡軞v遍幾個層次的對象組。因此,如果從棧和靜態(tài)存貯器里開始并歷遍整個引用組,你會找到所有存活的對象。對于你找到的每個單引用,你必須找到它所指向的對象,然后發(fā)覺那個對象的所有引用,接著找到那些引用所指向的所有對象,依次類推,直到你歷遍整個由棧和靜態(tài)存貯器里的引用所形成的網(wǎng)。每個你找到的對象必須還存活著。

39、注意,這里不存在分離的自我引用的對象組——他們只是沒被查找到,因此被自動當(dāng)作垃圾。</p><p>  在上述提到的垃圾收集方案中,JVM 使用了一種自適應(yīng)的垃圾收集方案,它對查找到的存活對象采取的措施依賴于它正在使用的方案變體。其中的一個變體就是 stop-and-copy。 它意味著——基于一些明顯的原因——程序首先停止運行(這不是一種在后臺實施的垃圾收集方案)。然后,每一個活著的對象從一個堆里被拷貝到另一個

40、堆里,同時被拷貝的活對象和死的對象被當(dāng)作垃圾遺棄。并且,當(dāng)對象被拷貝到新的堆里后,他們在那里被一個挨一個塞緊,因此實現(xiàn)了壓縮新堆的目的(而且如前所述,這種方式騰出了壓縮后多余出來的新的空間)。當(dāng)然,對象從一個地方移動到另一個地方的時候,所有指向?qū)ο蟮囊帽仨毾鄳?yīng)改變。 指向堆或者靜態(tài)存貯器里某個被移動對象的引用可以立即得到改變,但是還存在其它后來“在走走”的時候才會碰到的指向該對象的引用。 這些引用一旦發(fā)現(xiàn)就會被修改。(你可以想象存在一

41、張映射舊新地址的表)。</p><p>  有兩個問題使這種所謂的“拷貝型收集器”缺乏效率。 第一個問題就是你使用了兩個堆,為了維護(hù)兩倍于你實際所需要的內(nèi)存空間,你得在這兩個堆之間來回攪動著整個內(nèi)存空間。 一些 JVMs 通過依據(jù)實際所需來為堆分配大塊內(nèi)存,然后很簡單地從一個塊拷貝對象到另一個。第二個問題是拷貝過程本身。 一旦你地程序趨向于穩(wěn)定的時候,它可能生成很少或者幾乎不生成垃圾。 然而 stop-and-c

42、opy 方案不管這些,拷貝型垃圾收集器依舊把活對象占用的空間從一個地方拷貝到另一個地方,這就形成了浪費。 為了阻止這種情況的發(fā)生,一些 JVMs會探測沒有新垃圾產(chǎn)生的時機(jī),并且會轉(zhuǎn)向?qū)嵤┝硗庖粋€完全不同的垃圾收集方案。 這種不同的方案被稱為 mark-and-sweep,并且它是 Sun 的早期 JVM 版本一直使用的方案。 處理一般的垃圾收集工作,mark-and-sweep 表現(xiàn)得相當(dāng)?shù)芈钱?dāng)你的程序生成很少或者不生成垃圾時,它

43、又運行得很快。</p><p>  Mark-and-sweep 遵循著和 stop-and-copy 一樣的邏輯:從棧和靜態(tài)存貯器里出發(fā),跟蹤所有的引用從而找到存活的對象。 不過,每次它找到活對象的時候,那個對象被做以標(biāo)記,而且對象還不會被收集起來。 只有在整個標(biāo)記過程完成后,清掃(sweep)工作才真正開始。在清掃過程中,死對象被釋放存貯空間。不過,Mark-and-sweep 方案的實施過程并沒有拷貝壓縮的

44、步驟發(fā)生,所以如果垃圾收集器打算壓縮已經(jīng)成為碎片的堆,它會采用如同洗牌一樣的方式來重新安排對象的散亂分布。stop-and-copy 的思想是垃圾收集不在后臺完成,相反,程序需要停止運行的同時垃圾收集開始工作。在 Sun 的文獻(xiàn)資料里,你會發(fā)現(xiàn)很多資料認(rèn)為垃圾收集是一種低優(yōu)先性的后臺進(jìn)程,但事實上垃圾收集在早期的 Sun JVM 版本里并不是這樣執(zhí)行地。相反,當(dāng)內(nèi)存閑置空間少的時候,Sun 的垃圾收集器會終止程序運行。 Mark-and

45、-sweep 也需要程序被終止。</p><p>  正如前面提到的,在這里描述的 JVM 里,內(nèi)存被分配成大的塊。 如果你指定了一個大的對象,它將會得到它自己的內(nèi)存塊。 嚴(yán)格意義上的 stop-and-copy 在可以釋放舊堆之前,需要從源堆里拷貝每一個活著的對象到新的堆里,這會耗費大量內(nèi)存。 而有了塊的概念,垃圾收集器在收集的時候就能夠拷貝對象到死的塊里。每一個塊都有一個生成數(shù)用來跟蹤它是否還活著。正常情況下

46、,只有自上次垃圾收集后創(chuàng)建的塊才被壓縮;所有別的塊,如果在什么地方被引用的話,相應(yīng)的生成記數(shù)會增加。 這種方式解決了通常情況下許多短期生存的暫時對象。徹底的清掃工作會周期性進(jìn)行。大的對象仍舊不拷貝(他們只是把自己的生成記數(shù)增加),而那些包含小對象的塊會被拷貝和壓縮。 JVM 會監(jiān)視垃圾收集的效率,如果是因為所有的對象都穩(wěn)定運行而使得收集工作比較浪費時間的話,垃圾收集會轉(zhuǎn)向 mark-and-sweep 模式。</p>&l

47、t;p>  同樣地,JVM 也會跟蹤 mark-and-sweep 的運行效果,如果堆開始變得零碎不堪,垃收集又會轉(zhuǎn)回 stop-and-copy 模式。 這就是“自適應(yīng)”概念的來源,所以你能總結(jié)出一句冗長拗口的話: “自適應(yīng)地分階段地 stop-and-copy mark-and-sweep?!盝VM 里,可能會有一些附帶的加速技術(shù)。 一項很重要的技術(shù)就涉及到那個加載程序的操作,它被稱為 just-in-time(JIT)編譯器

48、。 JIT 編譯器能部分或者全部把程序轉(zhuǎn)換成機(jī)器能夠識別的代碼,所以程序就不需要 JVM 的解釋了,結(jié)果是程序運行起來快很多。當(dāng)必須加載類的時候(特別是你第一次想創(chuàng)建那個類的對象的時候),首先定位 .class 文件,然后相應(yīng)的字節(jié)碼被送入內(nèi)存。 這個時候,一個可以利用的方式就是直接讓 JIT 編譯所有的代碼,但是這樣做有兩個缺點: 這種方式的應(yīng)用由于混雜在程序的整個生命周期里,所以能累加起來,從而導(dǎo)致程序執(zhí)行起來會花費一些額外的時間;

溫馨提示

  • 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
  • 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
  • 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
  • 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
  • 5. 眾賞文庫僅提供信息存儲空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護(hù)處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負(fù)責(zé)。
  • 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請與我們聯(lián)系,我們立即糾正。
  • 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時也不承擔(dān)用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。

評論

0/150

提交評論