《編寫高質(zhì)量Java》(五)

四十一、讓多重繼承成為現(xiàn)實

在Java中一個類可以多重實現(xiàn),但不能多重繼承,也就是說一個類可以同時實現(xiàn)多個接口,但不能同時繼承多個類。但有時候我們確實需要繼承多個類,比如希望擁有兩個類的行為功能,就很難使用單繼承來解決了。幸運的是Java提供的內(nèi)部類可以曲折的解決此問題。

內(nèi)部類的一個重要特性:內(nèi)部類可以繼承一個與外部類無關的類,保證了內(nèi)部類的獨立性,正是基于這一點,多繼承才能成為可能。

四十二、讓工具類不可實例化

設置其構造函數(shù)位private訪問權限。同時為了防止反射實例化該類,還應該拋出異常,代碼如下:

public class UtilsClass{
    private UtilsClass{
        throw new Error("不要實例化我!");
    }
}

如此做才能保證一個工具類不會實例化,并且保證所有的訪問都是通過類名來進行的。需要注意的一點是:此工具類最好不要做繼承的打算,因為如果子類可以實例化的話,那就要調(diào)用父類的構造函數(shù),可是父類并沒有可被訪問的構造函數(shù),于是就會出現(xiàn)問題。

  • 注意:如果一個類不允許實例化,就要保證“平常”渠道都不能實例化它。

四十三、避免對向的淺拷貝

我們知道一個類實現(xiàn)了Cloneable接口就表示它具備被拷貝的能力,如果再覆寫clone()方法就會完全具備拷貝能力。拷貝是在內(nèi)存中進行的,所以在性能方面要比直接new生成對象要快的多,特別是在大對象的生成上,這會使性能的提升非常顯著。但是對象拷貝也有一個比較容易忽略的問題:淺拷貝(shadow clone,也叫作影子拷貝)存在對象拷貝不徹底的問題。

Object提供了一個對象拷貝的默認方法,即super.clone()方法,但是該方法是有缺陷的,它提供的是一種淺拷貝方式,也就是說它不對把對象所有的屬性都拷貝一份,而是有選擇性的拷貝,拷貝規(guī)則如下:

  • 基本類型。如果變量是基本類型,則拷貝其值,比如int,float等
  • 對象。如果變量是一個實例對象,則拷貝地址引用,也就是說此時新拷貝出來的對象與原有的對象共享該實例變量,不受訪問權限的限制。
  • String字符串。這個比較特殊,拷貝的也是一個地址,是個引用,但在修改時, 它會從字符串池中重新生成新的字符串,原有的字符串對象保持不變,此處我們可以認為String是一個基本類型。

注意:淺拷貝只是Java提供的一種簡單拷貝機制,不便于直接使用。

四十四、推薦使用序列化實現(xiàn)對象的拷貝

可以通過序列化方式,在內(nèi)存中通過字節(jié)流的拷貝來實現(xiàn),也就是把母對象寫到一個字節(jié)流中,再從字節(jié)流中將其讀出來,這樣就可以重建一個新對象了,該對象與母對象之間不存在引用共享的問題,也就相當于深拷貝了一個新對象。代碼如下:

public class CloneUtils{
    //拷貝一個對象
    public static <T extends Serializable> T clone(T obj){
        T clonedObj = null;
        try{
            ByteArrayOutputStream baos = new ByteArrayOutputStream();
            ObjectOutputStream oos = new ObjectOutputStream(baos);
            oos.writeObject(obj);
            oos.close();

            ByteArrayInputStream bais = new ByteArrayInputStream(baos.toByteArray());  
            ObjectInputStream ois = new ObjectInputStream(bais);
            clonedObj = (T)ois.readObject(obj);
            ois.close();
        }catch(Exception e){
            e.printStackTrace();
        }

        return clonedObj;
    }
}

此工具類要求被拷貝的對象必須實現(xiàn) Serializable接口,否則是沒有辦法拷貝的(反射除外),當然,serialVersionUID常量還是要加上去的,然后我們就可以通過CloneUtils工具進行對象的深拷貝。需要注意兩點:

  • 對象的內(nèi)部屬性都是可序列化的。如果有內(nèi)部屬性不可序列化,則會拋出序列化異常。
  • 注意方法和屬性的特殊修飾符。比如final、static變量的序列化問題會被引入到對象拷貝中來,這點需要特別注意,同時transient變量(瞬態(tài)變量,不進行序列化的變量)也會影響到拷貝的效果。
    當然,采用序列化方式拷貝時還有一個更簡單的方式,即使用Apache下的Commons工具包中的SerializationUtils類,直接使用更加簡潔方便。

四十五、覆寫equals方法時不要識別不出自己

我們在寫一個JavaBean時,經(jīng)常會覆寫equals方法,其目的是根據(jù)業(yè)務規(guī)則判斷兩個對象是否相等,這在DAO層是經(jīng)常用到的。

equals方法的自反性原則:
對于任何非空引用X, X.equals(X)應該返回true

四十六、equals應該考慮null值情景

equals對稱性原則:對于任何引用x和y的情形,如果x.equals(y)返回true, 那么y.equals(x)也應該返回true。
注意:在比較之前先判斷要比較的引用是否為null

四十七、在equals中使用getClass進行類型判斷

使用getClass代替instanceof進行類型判斷。

四十八、覆寫equals方法必須覆寫hashCode方法

HashMap的底層處理機制是以數(shù)組的方式保存Map條目(Map Entry)的,這其中的關鍵是這個數(shù)組的下標處理機制:依據(jù)傳入元素hashCode方法的返回值決定其數(shù)組的下標,如果該數(shù)組位置上已經(jīng)有了Map條目,且與傳入的鍵值相等則不處理,若不相等則覆蓋;如果數(shù)組位置沒有條目則插入,并加入到Map條目的鏈表中。

重寫hashCode代碼如下:

@Override
public int hashCode(){
    return new HashCodeBuilder.append(name).toHashCode();
}

其中HashCodeBuilder是org.apache.commons.lang.builder包下的一個哈希碼生成工具,使用起來非常方便,可以直接在項目中集成。

四十九、推薦覆寫toString()方法

當Bean屬性較多時,可以使用apache的commons工具包中的ToStringBuilder類,簡潔,方便。

五十、 使用package-info類為包服務

Java中有一個特殊的類:package-info類,它是專門為本包服務的。它的特殊主要體現(xiàn)在三個方面:

  • 它不能隨便被創(chuàng)建。IDE直接創(chuàng)建會報錯,可以用記事本創(chuàng)建一個,然后拷貝進去改一下即可,或者是從別的項目中拷貝過來。
  • 它服務的對象很特殊。它主要描述和記錄本包信息的。
  • package-info類不能有實現(xiàn)代碼
    另外它不可以繼承,沒有接口,沒有類間關系(關聯(lián)、組合、聚合)等。

主要有三個作用:

  • 聲明友好類和包內(nèi)訪問常量。
  • 為在包上標注注解提供便利。
  • 提供包的整體注釋說明。

五十一、不要主動進行垃圾回收

System.gc是一個非常危險的動作,因為它要停止所有的響應,才能檢查出內(nèi)存中是否有可回收的對象,這對一個應用系統(tǒng)來說風險很大。

最后編輯于
?著作權歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,885評論 6 541
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 99,312評論 3 429
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 177,993評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,667評論 1 317
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,410評論 6 411
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,778評論 1 328
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,775評論 3 446
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,955評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 49,521評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 41,266評論 3 358
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,468評論 1 374
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,998評論 5 363
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 44,696評論 3 348
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,095評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,385評論 1 294
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,193評論 3 398
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,431評論 2 378

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