Effective Java 案例分享(二)

7曾沈、拋棄廢棄的對象引用

當一個對象不再使用時伐蒂,應該將該引用設置為null,避免該對象并不能被JVM回收量淌。文中列出以下幾種情況:

  • 數(shù)組中的對象,使用結束后要把對應的未知的引用設置為null嫌褪;
  • 注意緩存對象的存活時間呀枢;
  • 注意listener和callback的添加和移除;

8渔扎、避免使用FINALIZERS和CLEANERS

Java為對象提供了finalize方法硫狞,當對象被準備回收時調(diào)用,在Java 9之后廢棄finalize晃痴,使用Cleaner残吩,但是兩者都是不推薦使用的。 Java提供了便捷的內(nèi)存回收算法倘核,如果重寫了finalize泣侮,可能導致GC無法回收足夠的內(nèi)存空間,增加OutOfMemoryError的幾率紧唱,也增加了GC的工作時長活尊,所以我們不應該在finalize或者cleaner中嘗試修改對象的持久化狀態(tài)。為了防止finalize被惡意攻擊漏益,可以把finalize方法設置為final蛹锰。

finalize的常用場景:

public class FileOutputStream extends OutputStream{

    protected void finalize() throws IOException {
        // Android-added: CloseGuard support.
        if (guard != null) {
            guard.warnIfOpen();
        }

        if (fd != null) {
            if (fd == FileDescriptor.out || fd == FileDescriptor.err) {
                flush();
            } else {
                // Android-removed: Obsoleted comment about shared FileDescriptor handling.
                close();
            }
        }
    }

{

應該以finalize為時間點,為對象的回收做準備绰疤,以上面的FileOutputStream為例铜犬,在finalize執(zhí)行時,清除緩沖區(qū)或關閉文件流轻庆,保證對象在回收時不會產(chǎn)生內(nèi)存問題癣猾。如果你希望你的對象具有自動回收功能,可以實現(xiàn)AutoCloseable接口余爆,并在finalize中調(diào)用實現(xiàn)close方法纷宇。

9、使用try with resource 替換 try-finally

請注意蛾方,此語法Java 7以下不支持像捶。

我們經(jīng)常使用FileInputStream或FileOutStream等資源型的類上陕,需要在使用結束后,及時關閉資源作岖。在過去唆垃,使用try-finally是最好的方式:

static String firstLineOfFile(String path) throws IOException {
    BufferedReader br = new BufferedReader(newFileReader(path));
    try {
        return br.readLine();
    } finally {
        br.close();
    }
}

如果一次操作使用多次資源型的類,例如以下代碼:

static void copy(String src, String dst) throws IOException {
    InputStream in = new FileInputStream(src);
    try {
        OutputStream out = new FileOutputStream(dst);
        try {
            byte[] buf = new byte[BUFFER_SIZE];
            int n;
            while ((n = in.read(buf)) >= 0)
                out.write(buf, 0, n);
        } finally {
            out.close();
        }
    } finally {
            in.close();
    }
}

以上代碼存在以下缺點:

  • 因為要關閉資源痘儡,導致上面的代碼需要有多個try-finally辕万,可讀性上并不好
  • close方法本身也會拋出異常
  • 兩個資源型的操作互相影響

如果資源性的操作再復雜一些,代碼會更加復雜沉删,為了解決這個問題渐尿,在Java 7之后提供了新的語法:

static String firstLineOfFile(String path) throws IOException {
    try (BufferedReader br = new BufferedReader(new FileReader(path))) {
        return br.readLine();
    }
}

static void copy(String src, String dst) throws IOException {
    try (InputStream in = new FileInputStream(src); OutputStream out = new FileOutputStream(dst)) {
        byte[] buf = new byte[BUFFER_SIZE];
        int n;
        while ((n = in.read(buf)) >= 0)
        out.write(buf, 0, n);
    }
}

此語法要求try方法內(nèi)的資源必須實現(xiàn)AutoCloseable接口,如果發(fā)生了異常矾瑰,會自動調(diào)用該類的close方法砖茸。同樣支持try-catch捕獲指定的異常。

static String firstLineOfFile(String path, String defaultVal) {
    try (BufferedReader br = new BufferedReader(new FileReader(path))) {
        return br.readLine();
    } catch (IOException e) {
        return defaultVal;
    }
}

10殴穴、遵循重寫Object的equals方法的規(guī)則

重寫Object的equals方法雖然很簡單凉夯,但是有很多種方式出錯,并且后果很嚴重采幌。一般來說我們不必重寫equals方法劲够,因為每一個Object的equals已經(jīng)保證了它的唯一性,如果必須要重寫他休傍,要注意以下原則:

  • 自反性:x.equals(x)征绎,必須返回true;
  • 對稱性:x.equals(y) == y.equals(x);
  • 傳遞性:如果x.equals(y) ==true磨取,y.equals(z) == true, 所以x.equals(z) == true;
  • 一致性:x.equals(y)的結果人柿,如果未發(fā)生改變,無論調(diào)用多少次都會得到同一個結果忙厌;
  • 非空性:x非空凫岖,x.equals(null)必須返回false;

其他判斷相等的方法:

  • 數(shù)字的比較逢净,如果不是float和double隘截,可以直接使用==判斷相等,如果是float汹胃,使用Float.compare(float, float),如果是Double东臀,使用Double.compare(double, double)着饥;
  • 如果需要判斷數(shù)組的每一個元素相等,可以考慮使用Arrays.equals()方法惰赋;
  • 避免判斷空指針宰掉,可以使用Objects.equals()方法呵哨;

如果需要重寫equals,也一定還要重寫hashcode轨奄,并且遵循equals的方法孟害。

?著作權歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市挪拟,隨后出現(xiàn)的幾起案子挨务,更是在濱河造成了極大的恐慌,老刑警劉巖玉组,帶你破解...
    沈念sama閱讀 216,402評論 6 499
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件谎柄,死亡現(xiàn)場離奇詭異,居然都是意外死亡惯雳,警方通過查閱死者的電腦和手機朝巫,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,377評論 3 392
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來石景,“玉大人劈猿,你說我怎么就攤上這事〕蹦酰” “怎么了揪荣?”我有些...
    開封第一講書人閱讀 162,483評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長恩商。 經(jīng)常有香客問我变逃,道長,這世上最難降的妖魔是什么怠堪? 我笑而不...
    開封第一講書人閱讀 58,165評論 1 292
  • 正文 為了忘掉前任揽乱,我火速辦了婚禮,結果婚禮上粟矿,老公的妹妹穿的比我還像新娘凰棉。我一直安慰自己,他們只是感情好陌粹,可當我...
    茶點故事閱讀 67,176評論 6 388
  • 文/花漫 我一把揭開白布撒犀。 她就那樣靜靜地躺著,像睡著了一般掏秩。 火紅的嫁衣襯著肌膚如雪或舞。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,146評論 1 297
  • 那天蒙幻,我揣著相機與錄音映凳,去河邊找鬼。 笑死邮破,一個胖子當著我的面吹牛诈豌,可吹牛的內(nèi)容都是我干的仆救。 我是一名探鬼主播,決...
    沈念sama閱讀 40,032評論 3 417
  • 文/蒼蘭香墨 我猛地睜開眼矫渔,長吁一口氣:“原來是場噩夢啊……” “哼彤蔽!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起庙洼,我...
    開封第一講書人閱讀 38,896評論 0 274
  • 序言:老撾萬榮一對情侶失蹤顿痪,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后送膳,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體员魏,經(jīng)...
    沈念sama閱讀 45,311評論 1 310
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,536評論 2 332
  • 正文 我和宋清朗相戀三年叠聋,在試婚紗的時候發(fā)現(xiàn)自己被綠了撕阎。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,696評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡碌补,死狀恐怖上岗,靈堂內(nèi)的尸體忽然破棺而出奈揍,到底是詐尸還是另有隱情求橄,我是刑警寧澤钮莲,帶...
    沈念sama閱讀 35,413評論 5 343
  • 正文 年R本政府宣布,位于F島的核電站袜啃,受9級特大地震影響汗侵,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜群发,卻給世界環(huán)境...
    茶點故事閱讀 41,008評論 3 325
  • 文/蒙蒙 一晰韵、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧熟妓,春花似錦雪猪、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,659評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至抬虽,卻和暖如春官觅,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背阐污。 一陣腳步聲響...
    開封第一講書人閱讀 32,815評論 1 269
  • 我被黑心中介騙來泰國打工缰猴, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人疤剑。 一個月前我還...
    沈念sama閱讀 47,698評論 2 368
  • 正文 我出身青樓滑绒,卻偏偏與公主長得像,于是被迫代替她去往敵國和親隘膘。 傳聞我的和親對象是個殘疾皇子疑故,可洞房花燭夜當晚...
    茶點故事閱讀 44,592評論 2 353

推薦閱讀更多精彩內(nèi)容