Java中的RAII
資源獲取即初始化( RAII )是Bjarne Stroustrup用C ++引入的一種用于異常安全資源管理的設計思想。 感謝垃圾回收,Java 沒有此功能,但是我們可以使用try-with-resources實現類似的功能。
約翰·哈德斯(John Huddles)在薩赫姆農場(1998)
RAII解決的問題很明顯; 看一下這段代碼(我確定您知道Semaphore是什么以及它在Java中的工作方式):
class Foo {private Semaphore sem = new Semaphore(5);void print(int x) throws Exception {this.sem.acquire();if (x > 1000) {throw new Exception("Too large!");}System.out.printf("x = %d", x);this.sem.release();} }該代碼相當原始,沒有任何用處,但您很可能會明白:如果從多個并行線程調用方法print() ,則只允許其中五個并行打印。 有時,如果x大于1000 ,將不允許其中的一些打印,并且將引發異常。
該代碼的問題是資源泄漏 。 x大于1000每個print()調用都會從信號量中獲取一個許可,并且不會返回它。 在五個異常調用中,信號量將為空,所有其他線程將不打印任何內容。
解決辦法是什么? 這里是:
class Foo {private Semaphore sem = new Semaphore(5);void print(int x) throws Exception {this.sem.acquire();if (x > 1000) {this.sem.release();throw new Exception("Too large!");}System.out.printf("x = %d", x);this.sem.release();} }拋出異常之前,我們必須釋放許可證。
但是,還有另一個問題出現:代碼重復。 我們在兩個地方發放許可證。 如果我們添加更多的throw指令,我們還必須添加更多的sem.release()調用。
在C ++中引入了一個非常優雅的解決方案,稱為RAII。 這是在Java中的外觀:
class Permit {private Semaphore sem;Permit(Semaphore s) {this.sem = s;this.sem.acquire();}@Overridepublic void finalize() {this.sem.release();} } class Foo {private Semaphore sem = new Semaphore(5);void print(int x) throws Exception {new Permit(this.sem);if (x > 1000) {throw new Exception("Too large!");}System.out.printf("x = %d", x);} }看看方法Foo.print()的代碼多么漂亮。 我們只創建了Permit類的實例,它立即在信號量上獲得了一個新的許可。 然后我們通過異常或正常方式退出方法print() ,然后方法Permit.finalize()釋放許可。
優雅,不是嗎? 是的,是的,但是在Java中不起作用。
它不起作用,因為與C ++不同,Java在可見性關閉時不會破壞對象。 當我們退出方法print()時, Permit類的對象不會被破壞。 它最終將被銷毀,但我們不知道確切的時間。 獲得信號燈中的所有許可之后,很可能將其破壞,并且我們將被阻止。
Java中也有一個解決方案。 它不像C ++那樣優雅,但確實有效。 這里是:
class Permit implements Closeable {private Semaphore sem;Permit(Semaphore s) {this.sem = s;}@Overridepublic void close() {this.sem.release();}public Permit acquire() {this.sem.acquire();return this;} } class Foo {private Semaphore sem = new Semaphore(5);void print(int x) throws Exception {try (Permit p = new Permit(this.sem).acquire()) {if (x > 1000) {throw new Exception("Too large!");}System.out.printf("x = %d", x);}} }請注意try塊和Permit類現在實現的Closeable接口。 try塊退出時,對象p將被“關閉”。 它可以在結尾處退出,也可以通過return或throw語句退出。 無論哪種情況,都會調用Permit.close() :這就是try-with-resources在Java中的工作方式。
我介紹的方法acquire()和感動sem.acquire()出的Permit構造,因為我相信這構造函數必須是免費的代碼。
總而言之,RAII是一個完美的設計 圖案 處理可能泄漏的資源時的方法。 即使Java沒有開箱即用,我們也可以通過try-with-resources和Closeable實現它。
翻譯自: https://www.javacodegeeks.com/2017/08/raii-in-java.html
總結
以上是生活随笔為你收集整理的Java中的RAII的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: iPhone 15系列京东预约人数达22
- 下一篇: 2023年第二季度紫光展锐智能手机芯片全