java编程规约
整理于 ?阿里巴巴Java開(kāi)發(fā)手冊(cè)?
一、編程規(guī)約?
(一) 命名風(fēng)格?
1. 【強(qiáng)制】 代碼中的命名均不能以下劃線或美元符號(hào)開(kāi)始,也不能以下劃線或美元符號(hào)結(jié)束。?
反例: _name / __name / $Object / name_ / name$ / Object$ 、
2. 【強(qiáng)制】 代碼中的命名嚴(yán)禁使用拼音與英文混合的方式,更不允許直接使用中文的方式。?
說(shuō)明:正確的英文拼寫(xiě)和語(yǔ)法可以讓閱讀者易于理解,避免歧義。注意,即使純拼音命名方式
也要避免采用。?
正例: alibaba / taobao / youku / hangzhou 等國(guó)際通用的名稱,可視同英文。?
反例: DaZhePromotion [打折] / getPingfenByName() [評(píng)分] / int 某變量 = 3?
3. 【強(qiáng)制】類名使用 UpperCamelCase 風(fēng)格,必須遵從駝峰形式,但以下情形例外:DO / BO /?
DTO / VO / AO?
正例:MarcoPolo / UserDO / XmlService / TcpUdpDeal / TaPromotion?
反例:macroPolo / UserDo / XMLService / TCPUDPDeal / TAPromotion?
?4. 【強(qiáng)制】方法名、參數(shù)名、成員變量、局部變量都統(tǒng)一使用 lowerCamelCase 風(fēng)格,必須遵從
駝峰形式。
正例: localValue / getHttpMessage() / inputUserId?
5. 【強(qiáng)制】常量命名全部大寫(xiě),單詞間用下劃線隔開(kāi),力求語(yǔ)義表達(dá)完整清楚,不要嫌名字長(zhǎng)。?
正例: MAX_STOCK_COUNT?
反例: MAX_COUNT?
6. 【強(qiáng)制】抽象類命名使用 Abstract 或 Base 開(kāi)頭;異常類命名使用 Exception 結(jié)尾;測(cè)試類
命名以它要測(cè)試的類的名稱開(kāi)始,以 Test 結(jié)尾。?
7. 【強(qiáng)制】中括號(hào)是數(shù)組類型的一部分,數(shù)組定義如下:String[] args;?
反例:使用 String args[]的方式來(lái)定義。?
8. 【強(qiáng)制】POJO 類中布爾類型的變量,都不要加 is,否則部分框架解析會(huì)引起序列化錯(cuò)誤。?
反例:定義為基本數(shù)據(jù)類型 Boolean isDeleted;的屬性,它的方法也是 isDeleted(),RPC阿里巴巴 Java 開(kāi)發(fā)手冊(cè)?
框架在反向解析的時(shí)候,“以為”對(duì)應(yīng)的屬性名稱是 deleted,導(dǎo)致屬性獲取不到,進(jìn)而拋出異
常。?
9. 【強(qiáng)制】包名統(tǒng)一使用小寫(xiě),點(diǎn)分隔符之間有且僅有一個(gè)自然語(yǔ)義的英語(yǔ)單詞。包名統(tǒng)一使用
單數(shù)形式,但是類名如果有復(fù)數(shù)含義,類名可以使用復(fù)數(shù)形式。?
正例: 應(yīng)用工具類包名為 com.alibaba.open.util、類名為 MessageUtils(此規(guī)則參考
spring 的框架結(jié)構(gòu))?
10. 【強(qiáng)制】杜絕完全不規(guī)范的縮寫(xiě),避免望文不知義。?
反例: AbstractClass“縮寫(xiě)”命名成 AbsClass;condition“縮寫(xiě)”命名成 condi,此類
隨意縮寫(xiě)嚴(yán)重降低了代碼的可閱讀性。?
11. 【推薦】如果使用到了設(shè)計(jì)模式,建議在類名中體現(xiàn)出具體模式。?
說(shuō)明:將設(shè)計(jì)模式體現(xiàn)在名字中,有利于閱讀者快速理解架構(gòu)設(shè)計(jì)思想。?
正例:
?
12. 【推薦】接口類中的方法和屬性不要加任何修飾符號(hào)(public 也不要加),保持代碼的簡(jiǎn)潔
性,并加上有效的 Javadoc 注釋。盡量不要在接口里定義變量,如果一定要定義變量,肯定是
與接口方法相關(guān),并且是整個(gè)應(yīng)用的基礎(chǔ)常量。?
正例:接口方法簽名:
?接口基礎(chǔ)常量表示:
反例:接口方法定義:
說(shuō)明:JDK8 中接口允許有默認(rèn)實(shí)現(xiàn),那么這個(gè) default 方法,是對(duì)所有實(shí)現(xiàn)類都有價(jià)值的默
認(rèn)實(shí)現(xiàn)。?
13. 接口和實(shí)現(xiàn)類的命名有兩套規(guī)則:?
?1)【強(qiáng)制】對(duì)于 Service 和 DAO 類,基于 SOA 的理念,暴露出來(lái)的服務(wù)一定是接口,內(nèi)部
的實(shí)現(xiàn)類用 Impl 的后綴與接口區(qū)別。?
正例:CacheServiceImpl 實(shí)現(xiàn) CacheService 接口。?
?2) 【推薦】 如果是形容能力的接口名稱,取對(duì)應(yīng)的形容詞做接口名(通常是–able 的形式)。?
正例:AbstractTranslator 實(shí)現(xiàn) Translatable。?
14. 【參考】枚舉類名建議帶上 Enum 后綴,枚舉成員名稱需要全大寫(xiě),單詞間用下劃線隔開(kāi)。?
說(shuō)明:枚舉其實(shí)就是特殊的常量類,且構(gòu)造方法被默認(rèn)強(qiáng)制是私有。?
正例:枚舉名字:DealStatusEnum,成員名稱:SUCCESS / UNKOWN_REASON。?
15. 【參考】各層命名規(guī)約:?
A) Service/DAO 層方法命名規(guī)約?
?1) 獲取單個(gè)對(duì)象的方法用 get 做前綴。 阿里巴巴 Java 開(kāi)發(fā)手冊(cè)?
?2) 獲取多個(gè)對(duì)象的方法用 list 做前綴。?
?3) 獲取統(tǒng)計(jì)值的方法用 count 做前綴。?
?4) 插入的方法用 save(推薦)或 insert 做前綴。?
?5) 刪除的方法用 remove(推薦)或 delete 做前綴。?
?6) 修改的方法用 update 做前綴。?
B) 領(lǐng)域模型命名規(guī)約?
?1) 數(shù)據(jù)對(duì)象:xxxDO,xxx 即為數(shù)據(jù)表名。?
?2) 數(shù)據(jù)傳輸對(duì)象:xxxDTO,xxx 為業(yè)務(wù)領(lǐng)域相關(guān)的名稱。?
?3) 展示對(duì)象:xxxVO,xxx 一般為網(wǎng)頁(yè)名稱。?
?4) POJO 是 DO/DTO/BO/VO 的統(tǒng)稱,禁止命名成 xxxPOJO。?
?
?
(二) 常量定義?
1. 【強(qiáng)制】不允許任何魔法值(即未經(jīng)定義的常量)直接出現(xiàn)在代碼中。?
?反例:
?
2. 【強(qiáng)制】long 或者 Long 初始賦值時(shí),必須使用大寫(xiě)的 L,不能是小寫(xiě)的 l,小寫(xiě)容易跟數(shù)字
1 混淆,造成誤解。?
說(shuō)明:Long a = 2l; 寫(xiě)的是數(shù)字的 21,還是 Long 型的 2??
3. 【推薦】不要使用一個(gè)常量類維護(hù)所有常量,應(yīng)該按常量功能進(jìn)行歸類,分開(kāi)維護(hù)。
如:緩存
相關(guān)的常量放在類:CacheConsts 下;系統(tǒng)配置相關(guān)的常量放在類:ConfigConsts 下。?
說(shuō)明:大而全的常量類,非得使用查找功能才能定位到修改的常量,不利于理解和維護(hù)。
?
4. 【推薦】常量的復(fù)用層次有五層:跨應(yīng)用共享常量、應(yīng)用內(nèi)共享常量、子工程內(nèi)共享常量、包
內(nèi)共享常量、類內(nèi)共享常量。?
?1) 跨應(yīng)用共享常量:放置在二方庫(kù)中,通常是 client.jar 中的 constant 目錄下。?
?2) 應(yīng)用內(nèi)共享常量:放置在一方庫(kù)的 modules 中的 constant 目錄下。?
?反例:易懂變量也要統(tǒng)一定義成應(yīng)用內(nèi)共享常量,兩位攻城師在兩個(gè)類中分別定義了表示
“是”的變量:?
?類 A 中:public static final String YES = "yes";?
?類 B 中:public static final String YES = "y";?
?A.YES.equals(B.YES),預(yù)期是 true,但實(shí)際返回為 false,導(dǎo)致線上問(wèn)題。?
?3) 子工程內(nèi)部共享常量:即在當(dāng)前子工程的 constant 目錄下。?
?4) 包內(nèi)共享常量:即在當(dāng)前包下單獨(dú)的 constant 目錄下。?
?5) 類內(nèi)共享常量:直接在類內(nèi)部 private static final 定義。 阿里巴巴 Java 開(kāi)發(fā)手冊(cè)?
?
5. 【推薦】如果變量值僅在一個(gè)范圍內(nèi)變化,且?guī)в忻Q之外的延伸屬性,定義為枚舉類。下面
正例中的數(shù)字就是延伸信息,表示星期幾。?
正例:public Enum { MONDAY(1), TUESDAY(2), WEDNESDAY(3), THURSDAY(4), FRIDAY(5), SATURDAY(6),?
SUNDAY(7);}?
?
(三) 代碼格式?
1. 【強(qiáng)制】大括號(hào)的使用約定。如果是大括號(hào)內(nèi)為空,則簡(jiǎn)潔地寫(xiě)成{}即可,不需要換行;如果
是非空代碼塊則:?
?1) 左大括號(hào)前不換行。?
?2) 左大括號(hào)后換行。?
?3) 右大括號(hào)前換行。?
?4) 右大括號(hào)后還有 else 等代碼則不換行;表示終止的右大括號(hào)后必須換行。?
2. 【強(qiáng)制】 左小括號(hào)和字符之間不出現(xiàn)空格;同樣,右小括號(hào)和字符之間也不出現(xiàn)空格。詳見(jiàn)
第 5 條下方正例提示。?
反例:if (空格 a == b 空格)?
3. 【強(qiáng)制】if/for/while/switch/do 等保留字與括號(hào)之間都必須加空格。?
4. 【強(qiáng)制】任何二目、三目運(yùn)算符的左右兩邊都需要加一個(gè)空格。?
說(shuō)明:運(yùn)算符包括賦值運(yùn)算符=、邏輯運(yùn)算符&&、加減乘除符號(hào)等。
?
5. 【強(qiáng)制】縮進(jìn)采用 4 個(gè)空格,禁止使用 tab 字符。?
說(shuō)明:如果使用 tab 縮進(jìn),必須設(shè)置 1 個(gè) tab 為 4 個(gè)空格。IDEA 設(shè)置 tab 為 4 個(gè)空格時(shí),
請(qǐng)勿勾選 Use tab character;而在 eclipse 中,必須勾選 insert spaces for tabs。?
正例: (涉及 1-5 點(diǎn))?
public static void main(String[] args) {?
// 縮進(jìn) 4 個(gè)空格?
String say = "hello";?
// 運(yùn)算符的左右必須有一個(gè)空格?
int flag = 0;?
// 關(guān)鍵詞 if 與括號(hào)之間必須有一個(gè)空格,括號(hào)內(nèi)的 f 與左括號(hào),0 與右括號(hào)不需要空格?
if (flag == 0) {?
System.out.println(say);?
}?
// 左大括號(hào)前加空格且不換行;左大括號(hào)后換行?
if (flag == 1) {?
System.out.println("world");?
// 右大括號(hào)前換行,右大括號(hào)后有 else,不用換行?
} else {?
System.out.println("ok"); 阿里巴巴 Java 開(kāi)發(fā)手冊(cè)?
?
?
(四) OOP 規(guī)約?
1. 【強(qiáng)制】避免通過(guò)一個(gè)類的對(duì)象引用訪問(wèn)此類的靜態(tài)變量或靜態(tài)方法,無(wú)謂增加編譯器解析成
本,直接用類名來(lái)訪問(wèn)即可。?
2. 【強(qiáng)制】所有的覆寫(xiě)方法,必須加@Override 注解。?
說(shuō)明:getObject()與 get0bject()的問(wèn)題。一個(gè)是字母的 O,一個(gè)是數(shù)字的 0,加@Override
可以準(zhǔn)確判斷是否覆蓋成功。另外,如果在抽象類中對(duì)方法簽名進(jìn)行修改,其實(shí)現(xiàn)類會(huì)馬上編
譯報(bào)錯(cuò)。?
3. 【強(qiáng)制】相同參數(shù)類型,相同業(yè)務(wù)含義,才可以使用 Java 的可變參數(shù),避免使用 Object。?
說(shuō)明:可變參數(shù)必須放置在參數(shù)列表的最后。(提倡同學(xué)們盡量不用可變參數(shù)編程)?
正例:public User getUsers(String type, Integer... ids) {...}?
4. 【強(qiáng)制】外部正在調(diào)用或者二方庫(kù)依賴的接口,不允許修改方法簽名,避免對(duì)接口調(diào)用方產(chǎn)生
影響。接口過(guò)時(shí)必須加@Deprecated 注解,并清晰地說(shuō)明采用的新接口或者新服務(wù)是什么。?
5. 【強(qiáng)制】不能使用過(guò)時(shí)的類或方法。?
說(shuō)明:java.net.URLDecoder 中的方法 decode(String encodeStr) 這個(gè)方法已經(jīng)過(guò)時(shí),應(yīng)
該使用雙參數(shù) decode(String source, String encode)。接口提供方既然明確是過(guò)時(shí)接口,
那么有義務(wù)同時(shí)提供新的接口;作為調(diào)用方來(lái)說(shuō),有義務(wù)去考證過(guò)時(shí)方法的新實(shí)現(xiàn)是什么。?
6. 【強(qiáng)制】Object 的 equals 方法容易拋空指針異常,應(yīng)使用常量或確定有值的對(duì)象來(lái)調(diào)用
equals。?
正例: "test".equals(object);?
反例: object.equals("test");?
說(shuō)明:推薦使用 java.util.Objects#equals (JDK7 引入的工具類)?
7. 【強(qiáng)制】所有的相同類型的包裝類對(duì)象之間值的比較,全部使用 equals 方法比較。?
說(shuō)明:對(duì)于 Integer var = ? 在-128 至 127 范圍內(nèi)的賦值,Integer 對(duì)象是在
IntegerCache.cache 產(chǎn)生,會(huì)復(fù)用已有對(duì)象,這個(gè)區(qū)間內(nèi)的 Integer 值可以直接使用==進(jìn)行
判斷,但是這個(gè)區(qū)間之外的所有數(shù)據(jù),都會(huì)在堆上產(chǎn)生,并不會(huì)復(fù)用已有對(duì)象,這是一個(gè)大坑,
推薦使用 equals 方法進(jìn)行判斷。?
8. 關(guān)于基本數(shù)據(jù)類型與包裝數(shù)據(jù)類型的使用標(biāo)準(zhǔn)如下:?
?1) 【強(qiáng)制】所有的 POJO 類屬性必須使用包裝數(shù)據(jù)類型。?
?2) 【強(qiáng)制】RPC 方法的返回值和參數(shù)必須使用包裝數(shù)據(jù)類型。?
?3) 【推薦】所有的局部變量使用基本數(shù)據(jù)類型。?
?說(shuō)明:POJO 類屬性沒(méi)有初值是提醒使用者在需要使用時(shí),必須自己顯式地進(jìn)行賦值,任何
NPE 問(wèn)題,或者入庫(kù)檢查,都由使用者來(lái)保證。?
?正例:數(shù)據(jù)庫(kù)的查詢結(jié)果可能是 null,因?yàn)樽詣?dòng)拆箱,用基本數(shù)據(jù)類型接收有 NPE 風(fēng)險(xiǎn)。 阿里巴巴 Java 開(kāi)發(fā)手冊(cè)?
?反例:比如顯示成交總額漲跌情況,即正負(fù) x%,x 為基本數(shù)據(jù)類型,調(diào)用的 RPC 服務(wù),調(diào)用
不成功時(shí),返回的是默認(rèn)值,頁(yè)面顯示:0%,這是不合理的,應(yīng)該顯示成中劃線-。所以包裝
數(shù)據(jù)類型的 null 值,能夠表示額外的信息,如:遠(yuǎn)程調(diào)用失敗,異常退出。?
?
9. 【強(qiáng)制】定義 DO/DTO/VO 等 POJO 類時(shí),不要設(shè)定任何屬性默認(rèn)值。?
反例:POJO 類的 gmtCreate 默認(rèn)值為 new Date();但是這個(gè)屬性在數(shù)據(jù)提取時(shí)并沒(méi)有置入具
體值,在更新其它字段時(shí)又附帶更新了此字段,導(dǎo)致創(chuàng)建時(shí)間被修改成當(dāng)前時(shí)間。?
10. 【強(qiáng)制】序列化類新增屬性時(shí),請(qǐng)不要修改 serialVersionUID 字段,避免反序列失敗;如
果完全不兼容升級(jí),避免反序列化混亂,那么請(qǐng)修改 serialVersionUID 值。?
說(shuō)明:注意 serialVersionUID 不一致會(huì)拋出序列化運(yùn)行時(shí)異常。?
11. 【強(qiáng)制】構(gòu)造方法里面禁止加入任何業(yè)務(wù)邏輯,如果有初始化邏輯,請(qǐng)放在 init 方法中。?
12. 【強(qiáng)制】POJO 類必須寫(xiě) toString 方法。使用 IDE 的中工具:source> generate toString
時(shí),如果繼承了另一個(gè) POJO 類,注意在前面加一下 super.toString。?
說(shuō)明:在方法執(zhí)行拋出異常時(shí),可以直接調(diào)用 POJO 的 toString()方法打印其屬性值,便于排
查問(wèn)題。?
13. 【推薦】使用索引訪問(wèn)用 String 的 split 方法得到的數(shù)組時(shí),需做最后一個(gè)分隔符后有無(wú)
內(nèi)容的檢查,否則會(huì)有拋 IndexOutOfBoundsException 的風(fēng)險(xiǎn)。?
說(shuō)明:?
String str = "a,b,c,,";?
String[] ary = str.split(",");?
//預(yù)期大于 3,結(jié)果是 3?
System.out.println(ary.length);?
14. 【推薦】當(dāng)一個(gè)類有多個(gè)構(gòu)造方法,或者多個(gè)同名方法,這些方法應(yīng)該按順序放置在一起,
便于閱讀。?
15. 【推薦】 類內(nèi)方法定義順序依次是:公有方法或保護(hù)方法 > 私有方法 > getter/setter
方法。?
說(shuō)明:公有方法是類的調(diào)用者和維護(hù)者最關(guān)心的方法,首屏展示最好;保護(hù)方法雖然只是子類
關(guān)心,也可能是“模板設(shè)計(jì)模式”下的核心方法;而私有方法外部一般不需要特別關(guān)心,是一個(gè)
黑盒實(shí)現(xiàn);因?yàn)榉椒ㄐ畔r(jià)值較低,所有 Service 和 DAO 的 getter/setter 方法放在類體最
后。?
16. 【推薦】setter 方法中,參數(shù)名稱與類成員變量名稱一致,this.成員名 = 參數(shù)名。在
getter/setter 方法中,不要增加業(yè)務(wù)邏輯,增加排查問(wèn)題的難度。?
反例:?
?
17. 【推薦】循環(huán)體內(nèi),字符串的連接方式,使用 StringBuilder 的 append 方法進(jìn)行擴(kuò)展。?
說(shuō)明:反編譯出的字節(jié)碼文件顯示每次循環(huán)都會(huì) new 出一個(gè) StringBuilder 對(duì)象,然后進(jìn)行
append 操作,最后通過(guò) toString 方法返回 String 對(duì)象,造成內(nèi)存資源浪費(fèi)。?
反例:?
?
18. 【推薦】final 可以聲明類、成員變量、方法、以及本地變量,下列情況使用 final 關(guān)鍵字:?
?1) 不允許被繼承的類,如:String 類。?
?2) 不允許修改引用的域?qū)ο?#xff0c;如:POJO 類的域變量。?
?3) 不允許被重寫(xiě)的方法,如:POJO 類的 setter 方法。?
?4) 不允許運(yùn)行過(guò)程中重新賦值的局部變量。?
?5) 避免上下文重復(fù)使用一個(gè)變量,使用 final 描述可以強(qiáng)制重新定義一個(gè)變量,方便更好
地進(jìn)行重構(gòu)。?
19. 【推薦】慎用 Object 的 clone 方法來(lái)拷貝對(duì)象。?
說(shuō)明:對(duì)象的 clone 方法默認(rèn)是淺拷貝,若想實(shí)現(xiàn)深拷貝需要重寫(xiě) clone 方法實(shí)現(xiàn)屬性對(duì)象
的拷貝。?
20. 【推薦】類成員與方法訪問(wèn)控制從嚴(yán):?
?1) 如果不允許外部直接通過(guò) new 來(lái)創(chuàng)建對(duì)象,那么構(gòu)造方法必須是 private。?
?2) 工具類不允許有 public 或 default 構(gòu)造方法。?
?3) 類非 static 成員變量并且與子類共享,必須是 protected。?
?4) 類非 static 成員變量并且僅在本類使用,必須是 private。?
?5) 類 static 成員變量如果僅在本類使用,必須是 private。?
?6) 若是 static 成員變量,必須考慮是否為 final。?
?7) 類成員方法只供類內(nèi)部調(diào)用,必須是 private。?
?8) 類成員方法只對(duì)繼承類公開(kāi),那么限制為 protected。?
說(shuō)明:任何類、方法、參數(shù)、變量,嚴(yán)控訪問(wèn)范圍。過(guò)于寬泛的訪問(wèn)范圍,不利于模塊解耦。
思考:如果是一個(gè) private 的方法,想刪除就刪除,可是一個(gè) public 的 service 方法,或者
一個(gè) public 的成員變量,刪除一下,不得手心冒點(diǎn)汗嗎?變量像自己的小孩,盡量在自己的
視線內(nèi),變量作用域太大,如果無(wú)限制的到處跑,那么你會(huì)擔(dān)心的。?
?
?
(五) 集合處理?
1. 【強(qiáng)制】關(guān)于 hashCode 和 equals 的處理,遵循如下規(guī)則:?
1) 只要重寫(xiě) equals,就必須重寫(xiě) hashCode。?
2) 因?yàn)?Set 存儲(chǔ)的是不重復(fù)的對(duì)象,依據(jù) hashCode 和 equals 進(jìn)行判斷,所以 Set 存儲(chǔ)的
對(duì)象必須重寫(xiě)這兩個(gè)方法。?
3) 如果自定義對(duì)象做為 Map 的鍵,那么必須重寫(xiě) hashCode 和 equals。?
說(shuō)明:String 重寫(xiě)了 hashCode 和 equals 方法,所以我們可以非常愉快地使用 String 對(duì)象
作為 key 來(lái)使用。?
2. 【強(qiáng)制】 ArrayList的subList結(jié)果不可強(qiáng)轉(zhuǎn)成ArrayList,否則會(huì)拋出ClassCastException
異常:java.util.RandomAccessSubList cannot be cast to java.util.ArrayList ;?
說(shuō)明:subList 返回的是 ArrayList 的內(nèi)部類 SubList,并不是 ArrayList ,而是?
ArrayList 的一個(gè)視圖,對(duì)于 SubList 子列表的所有操作最終會(huì)反映到原列表上。?
3. 【強(qiáng)制】 在 subList 場(chǎng)景中,高度注意對(duì)原集合元素個(gè)數(shù)的修改,會(huì)導(dǎo)致子列表的遍歷、增
加、刪除均產(chǎn)生 ConcurrentModificationException 異常。?
4. 【強(qiáng)制】使用集合轉(zhuǎn)數(shù)組的方法,必須使用集合的 toArray(T[] array),傳入的是類型完全
一樣的數(shù)組,大小就是 list.size()。?
說(shuō)明:使用 toArray 帶參方法,入?yún)⒎峙涞臄?shù)組空間不夠大時(shí),toArray 方法內(nèi)部將重新分配
內(nèi)存空間,并返回新數(shù)組地址;如果數(shù)組元素大于實(shí)際所需,下標(biāo)為[ list.size() ]的數(shù)組
元素將被置為 null,其它數(shù)組元素保持原值,因此最好將方法入?yún)?shù)組大小定義與集合元素
個(gè)數(shù)一致。?
正例:?
List<String> list = new ArrayList<String>(2);?
list.add("guan");?
list.add("bao");?
String[] array = new String[list.size()];?
array = list.toArray(array);?
反例:直接使用 toArray 無(wú)參方法存在問(wèn)題,此方法返回值只能是 Object[]類,若強(qiáng)轉(zhuǎn)其它
類型數(shù)組將出現(xiàn) ClassCastException 錯(cuò)誤。?
5. 【強(qiáng)制】使用工具類 Arrays.asList()把數(shù)組轉(zhuǎn)換成集合時(shí),不能使用其修改集合相關(guān)的方
法,它的 add/remove/clear 方法會(huì)拋出 UnsupportedOperationException 異常。?
說(shuō)明:asList 的返回對(duì)象是一個(gè) Arrays 內(nèi)部類,并沒(méi)有實(shí)現(xiàn)集合的修改方法。Arrays.asList
體現(xiàn)的是適配器模式,只是轉(zhuǎn)換接口,后臺(tái)的數(shù)據(jù)仍是數(shù)組。?
?String[] str = new String[] { "a", "b" };?
?List list = Arrays.asList(str); 阿里巴巴 Java 開(kāi)發(fā)手冊(cè)?
第一種情況:list.add("c"); 運(yùn)行時(shí)異常。?
第二種情況:str[0] = "gujin"; 那么 list.get(0)也會(huì)隨之修改。?
6. 【強(qiáng)制】泛型通配符<? extends T>來(lái)接收返回的數(shù)據(jù),此寫(xiě)法的泛型集合不能使用 add 方
法,而<? super T>不能使用 get 方法,做為接口調(diào)用賦值時(shí)易出錯(cuò)。?
說(shuō)明:擴(kuò)展說(shuō)一下 PECS(Producer Extends Consumer Super)原則:1)頻繁往外讀取內(nèi)容
的,適合用上界 Extends。2)經(jīng)常往里插入的,適合用下界 Super。
?
7. 【強(qiáng)制】不要在 foreach 循環(huán)里進(jìn)行元素的 remove/add 操作。remove 元素請(qǐng)使用 Iterator
方式,如果并發(fā)操作,需要對(duì) Iterator 對(duì)象加鎖。?
正例:?
反例:?
說(shuō)明:以上代碼的執(zhí)行結(jié)果肯定會(huì)出乎大家的意料,那么試一下把“1”換成“2”,會(huì)是同樣的
結(jié)果嗎??
8. 【強(qiáng)制】 在 JDK7 版本及以上,Comparator 要滿足如下三個(gè)條件,不然 Arrays.sort,
Collections.sort 會(huì)報(bào) IllegalArgumentException 異常。?
說(shuō)明:?
?1) x,y 的比較結(jié)果和 y,x 的比較結(jié)果相反。?
?2) x>y,y>z,則 x>z。?
?3) x=y,則 x,z 比較結(jié)果和 y,z 比較結(jié)果相同。?
反例:下例中沒(méi)有處理相等的情況,實(shí)際使用中可能會(huì)出現(xiàn)異常:?
?
9. 【推薦】集合初始化時(shí),指定集合初始值大小。?
說(shuō)明:HashMap 使用 HashMap(int initialCapacity) 初始化,?
正例:initialCapacity = (需要存儲(chǔ)的元素個(gè)數(shù) / 負(fù)載因子) + 1。注意負(fù)載因子(即 loader?
factor)默認(rèn)為 0.75,如果暫時(shí)無(wú)法確定初始值大小,請(qǐng)?jiān)O(shè)置為 16。?
反例:HashMap 需要放置 1024 個(gè)元素,由于沒(méi)有設(shè)置容量初始大小,隨著元素不斷增加,容
量 7 次被迫擴(kuò)大,resize 需要重建 hash 表,嚴(yán)重影響性能。
?
10. 【推薦】使用 entrySet 遍歷 Map 類集合 KV,而不是 keySet 方式進(jìn)行遍歷。?
說(shuō)明:keySet 其實(shí)是遍歷了 2 次,一次是轉(zhuǎn)為 Iterator 對(duì)象,另一次是從 hashMap 中取出
key 所對(duì)應(yīng)的 value。而 entrySet 只是遍歷了一次就把 key 和 value 都放到了 entry 中,效
率更高。如果是 JDK8,使用 Map.foreach 方法。?
正例:values()返回的是 V 值集合,是一個(gè) list 集合對(duì)象;keySet()返回的是 K 值集合,是
一個(gè) Set 集合對(duì)象;entrySet()返回的是 K-V 值組合集合。?
11. 【推薦】高度注意 Map 類集合 K/V 能不能存儲(chǔ) null 值的情況,如下表格:?
集合類 Key Value Super 說(shuō)明?
Hashtable 不允許為 null 不允許為 null Dictionary 線程安全?
ConcurrentHashMap 不允許為 null 不允許為 null AbstractMap 分段鎖技術(shù)?
TreeMap 不允許為 null 允許為 null AbstractMap 線程不安全?
HashMap 允許為 null 允許為 null AbstractMap 線程不安全?
反例: 由于 HashMap 的干擾,很多人認(rèn)為 ConcurrentHashMap 是可以置入 null 值,而事實(shí)上,
存儲(chǔ) null 值時(shí)會(huì)拋出 NPE 異常。?
12. 【參考】合理利用好集合的有序性(sort)和穩(wěn)定性(order),避免集合的無(wú)序性(unsort)和
不穩(wěn)定性(unorder)帶來(lái)的負(fù)面影響。?
說(shuō)明:有序性是指遍歷的結(jié)果是按某種比較規(guī)則依次排列的。穩(wěn)定性指集合每次遍歷的元素次
序是一定的。如:ArrayList 是 order/unsort;HashMap 是 unorder/unsort;TreeSet 是
order/sort。?
13. 【參考】利用 Set 元素唯一的特性,可以快速對(duì)一個(gè)集合進(jìn)行去重操作,避免使用 List 的
contains 方法進(jìn)行遍歷、對(duì)比、去重操作。?
?
(六) 并發(fā)處理?
1. 【強(qiáng)制】獲取單例對(duì)象需要保證線程安全,其中的方法也要保證線程安全。?
說(shuō)明:資源驅(qū)動(dòng)類、工具類、單例工廠類都需要注意。?
2. 【強(qiáng)制】創(chuàng)建線程或線程池時(shí)請(qǐng)指定有意義的線程名稱,方便出錯(cuò)時(shí)回溯。?
正例:?
?
?
3. 【強(qiáng)制】線程資源必須通過(guò)線程池提供,不允許在應(yīng)用中自行顯式創(chuàng)建線程。?
說(shuō)明:使用線程池的好處是減少在創(chuàng)建和銷毀線程上所花的時(shí)間以及系統(tǒng)資源的開(kāi)銷,解決資
源不足的問(wèn)題。如果不使用線程池,有可能造成系統(tǒng)創(chuàng)建大量同類線程而導(dǎo)致消耗完內(nèi)存或者
“過(guò)度切換”的問(wèn)題。?
4. 【強(qiáng)制】線程池不允許使用 Executors 去創(chuàng)建,而是通過(guò) ThreadPoolExecutor 的方式,這樣
的處理方式讓寫(xiě)的同學(xué)更加明確線程池的運(yùn)行規(guī)則,規(guī)避資源耗盡的風(fēng)險(xiǎn)。?
說(shuō)明:Executors 返回的線程池對(duì)象的弊端如下:?
1)FixedThreadPool 和 SingleThreadPool:?
?允許的請(qǐng)求隊(duì)列長(zhǎng)度為 Integer.MAX_VALUE,可能會(huì)堆積大量的請(qǐng)求,從而導(dǎo)致 OOM。?
2)CachedThreadPool 和 ScheduledThreadPool:?
?允許的創(chuàng)建線程數(shù)量為 Integer.MAX_VALUE,可能會(huì)創(chuàng)建大量的線程,從而導(dǎo)致 OOM。?
5. 【強(qiáng)制】SimpleDateFormat 是線程不安全的類,一般不要定義為 static 變量,如果定義為
static,必須加鎖,或者使用 DateUtils 工具類。?
正例:注意線程安全,使用 DateUtils。亦推薦如下處理:?
說(shuō)明:如果是 JDK8 的應(yīng)用,可以使用 Instant 代替 Date,LocalDateTime 代替 Calendar,
DateTimeFormatter代替Simpledateformatter,官方給出的解釋:simple beautiful strong?
immutable thread-safe。?
6. 【強(qiáng)制】高并發(fā)時(shí),同步調(diào)用應(yīng)該去考量鎖的性能損耗。能用無(wú)鎖數(shù)據(jù)結(jié)構(gòu),就不要用鎖;能
鎖區(qū)塊,就不要鎖整個(gè)方法體;能用對(duì)象鎖,就不要用類鎖。?
說(shuō)明:盡可能使加鎖的代碼塊工作量盡可能的小,避免在鎖代碼塊中調(diào)用 RPC 方法。?
7. 【強(qiáng)制】對(duì)多個(gè)資源、數(shù)據(jù)庫(kù)表、對(duì)象同時(shí)加鎖時(shí),需要保持一致的加鎖順序,否則可能會(huì)造
成死鎖。?
說(shuō)明:線程一需要對(duì)表 A、B、C 依次全部加鎖后才可以進(jìn)行更新操作,那么線程二的加鎖順序
也必須是 A、B、C,否則可能出現(xiàn)死鎖。??
?
8. 【強(qiáng)制】并發(fā)修改同一記錄時(shí),避免更新丟失,需要加鎖。要么在應(yīng)用層加鎖,要么在緩存加
鎖,要么在數(shù)據(jù)庫(kù)層使用樂(lè)觀鎖,使用 version 作為更新依據(jù)。?
說(shuō)明:如果每次訪問(wèn)沖突概率小于 20%,推薦使用樂(lè)觀鎖,否則使用悲觀鎖。樂(lè)觀鎖的重試次
數(shù)不得小于 3 次。
?
9. 【強(qiáng)制】多線程并行處理定時(shí)任務(wù)時(shí),Timer 運(yùn)行多個(gè) TimeTask 時(shí),只要其中之一沒(méi)有捕獲
拋出的異常,其它任務(wù)便會(huì)自動(dòng)終止運(yùn)行,使用 ScheduledExecutorService 則沒(méi)有這個(gè)問(wèn)題。?
10. 【推薦】使用 CountDownLatch 進(jìn)行異步轉(zhuǎn)同步操作,每個(gè)線程退出前必須調(diào)用 countDown
方法,線程執(zhí)行代碼注意 catch 異常,確保 countDown 方法可以執(zhí)行,避免主線程無(wú)法執(zhí)行
至 await 方法,直到超時(shí)才返回結(jié)果。?
說(shuō)明:注意,子線程拋出異常堆棧,不能在主線程 try-catch 到。?
11. 【推薦】避免 Random 實(shí)例被多線程使用,雖然共享該實(shí)例是線程安全的,但會(huì)因競(jìng)爭(zhēng)同一
seed 導(dǎo)致的性能下降。?
說(shuō)明:Random 實(shí)例包括 java.util.Random 的實(shí)例或者 Math.random()的方式。?
正例:在 JDK7 之后,可以直接使用 API ThreadLocalRandom,而在 JDK7 之前,需要編碼保
證每個(gè)線程持有一個(gè)實(shí)例。?
12. 【推薦】在并發(fā)場(chǎng)景下,通過(guò)雙重檢查鎖(double-checked locking)實(shí)現(xiàn)延遲初始化的優(yōu)
化問(wèn)題隱患(可參考 The "Double-Checked Locking is Broken" Declaration),推薦問(wèn)
題解決方案中較為簡(jiǎn)單一種(適用于 JDK5 及以上版本),將目標(biāo)屬性聲明為 volatile 型。?
反例:?
?
?
13. 【參考】volatile 解決多線程內(nèi)存不可見(jiàn)問(wèn)題。對(duì)于一寫(xiě)多讀,是可以解決變量同步問(wèn)題,
但是如果多寫(xiě),同樣無(wú)法解決線程安全問(wèn)題。如果是 count++操作,使用如下類實(shí)現(xiàn):
AtomicInteger count = new AtomicInteger(); count.addAndGet(1); 如果是 JDK8,推
薦使用 LongAdder 對(duì)象,比 AtomicLong 性能更好(減少樂(lè)觀鎖的重試次數(shù))。
?
14. 【參考】 HashMap 在容量不夠進(jìn)行 resize 時(shí)由于高并發(fā)可能出現(xiàn)死鏈,導(dǎo)致 CPU 飆升,在
開(kāi)發(fā)過(guò)程中可以使用其它數(shù)據(jù)結(jié)構(gòu)或加鎖來(lái)規(guī)避此風(fēng)險(xiǎn)。?
?
15. 【參考】ThreadLocal 無(wú)法解決共享對(duì)象的更新問(wèn)題,ThreadLocal 對(duì)象建議使用 static
修飾。這個(gè)變量是針對(duì)一個(gè)線程內(nèi)所有操作共有的,所以設(shè)置為靜態(tài)變量,所有此類實(shí)例共享
此靜態(tài)變量 ,也就是說(shuō)在類第一次被使用時(shí)裝載,只分配一塊存儲(chǔ)空間,所有此類的對(duì)象(只
要是這個(gè)線程內(nèi)定義的)都可以操控這個(gè)變量。?
?
(七) 控制語(yǔ)句?
1. 【強(qiáng)制】在一個(gè) switch 塊內(nèi),每個(gè) case 要么通過(guò) break/return 等來(lái)終止,要么注釋說(shuō)明程
序?qū)⒗^續(xù)執(zhí)行到哪一個(gè) case 為止;在一個(gè) switch 塊內(nèi),都必須包含一個(gè) default 語(yǔ)句并且
放在最后,即使它什么代碼也沒(méi)有。?
2. 【強(qiáng)制】在 if/else/for/while/do 語(yǔ)句中必須使用大括號(hào)。即使只有一行代碼,避免使用
單行的形式:if (condition) statements;?
3. 【推薦】表達(dá)異常的分支時(shí),少用 if-else 方式,這種方式可以改寫(xiě)成:?
// 接著寫(xiě) else 的業(yè)務(wù)邏輯代碼;?
說(shuō)明:如果非得使用 if()...else if()...else...方式表達(dá)邏輯,【強(qiáng)制】避免后續(xù)代碼維
護(hù)困難,請(qǐng)勿超過(guò) 3 層。?
正例:邏輯上超過(guò) 3 層的 if-else 代碼可以使用衛(wèi)語(yǔ)句,或者狀態(tài)模式來(lái)實(shí)現(xiàn)。衛(wèi)語(yǔ)句示例
如下:?
?
?
4. 【推薦】除常用方法(如 getXxx/isXxx)等外,不要在條件判斷中執(zhí)行其它復(fù)雜的語(yǔ)句,將復(fù)
雜邏輯判斷的結(jié)果賦值給一個(gè)有意義的布爾變量名,以提高可讀性。?
說(shuō)明:很多 if 語(yǔ)句內(nèi)的邏輯相當(dāng)復(fù)雜,閱讀者需要分析條件表達(dá)式的最終結(jié)果,才能明確什么阿里巴巴 Java 開(kāi)發(fā)手冊(cè)?
?——禁止用于商業(yè)用途,違者必究—— 15 / 33
?
樣的條件執(zhí)行什么樣的語(yǔ)句,那么,如果閱讀者分析邏輯表達(dá)式錯(cuò)誤呢??
正例:?
//偽代碼如下?
?
?
5. 【推薦】循環(huán)體中的語(yǔ)句要考量性能,以下操作盡量移至循環(huán)體外處理,如定義對(duì)象、變量、
獲取數(shù)據(jù)庫(kù)連接,進(jìn)行不必要的 try-catch 操作(這個(gè) try-catch 是否可以移至循環(huán)體外)。
?
6. 【推薦】接口入?yún)⒈Wo(hù),這種場(chǎng)景常見(jiàn)的是用于做批量操作的接口。
?
7. 【參考】下列情形,需要進(jìn)行參數(shù)校驗(yàn):?
?1) 調(diào)用頻次低的方法。?
?2) 執(zhí)行時(shí)間開(kāi)銷很大的方法。此情形中,參數(shù)校驗(yàn)時(shí)間幾乎可以忽略不計(jì),但如果因?yàn)閰?br /> 數(shù)錯(cuò)誤導(dǎo)致中間執(zhí)行回退,或者錯(cuò)誤,那得不償失。?
?3) 需要極高穩(wěn)定性和可用性的方法。?
?4) 對(duì)外提供的開(kāi)放接口,不管是 RPC/API/HTTP 接口。?
?5) 敏感權(quán)限入口。?
8. 【參考】下列情形,不需要進(jìn)行參數(shù)校驗(yàn):?
?1) 極有可能被循環(huán)調(diào)用的方法。但在方法說(shuō)明里必須注明外部參數(shù)檢查要求。?
?2) 底層調(diào)用頻度比較高的方法。畢竟是像純凈水過(guò)濾的最后一道,參數(shù)錯(cuò)誤不太可能到底
層才會(huì)暴露問(wèn)題。一般 DAO 層與 Service 層都在同一個(gè)應(yīng)用中,部署在同一臺(tái)服務(wù)器中,所
以 DAO 的參數(shù)校驗(yàn),可以省略。?
?3) 被聲明成 private 只會(huì)被自己代碼所調(diào)用的方法,如果能夠確定調(diào)用方法的代碼傳入?yún)?br /> 數(shù)已經(jīng)做過(guò)檢查或者肯定不會(huì)有問(wèn)題,此時(shí)可以不校驗(yàn)參數(shù)。?
?
(八) 注釋規(guī)約?
1. 【強(qiáng)制】類、類屬性、類方法的注釋必須使用 Javadoc 規(guī)范,使用/**內(nèi)容*/格式,不得使用
//xxx 方式。?
說(shuō)明:在 IDE 編輯窗口中,Javadoc 方式會(huì)提示相關(guān)注釋,生成 Javadoc 可以正確輸出相應(yīng)注
釋;在 IDE 中,工程調(diào)用方法時(shí),不進(jìn)入方法即可懸浮提示方法、參數(shù)、返回值的意義,提高
?
2. 【強(qiáng)制】所有的抽象方法(包括接口中的方法)必須要用 Javadoc 注釋、除了返回值、參數(shù)、
異常說(shuō)明外,還必須指出該方法做什么事情,實(shí)現(xiàn)什么功能。?
說(shuō)明:對(duì)子類的實(shí)現(xiàn)要求,或者調(diào)用注意事項(xiàng),請(qǐng)一并說(shuō)明。?
3. 【強(qiáng)制】所有的類都必須添加創(chuàng)建者和創(chuàng)建日期。?
4. 【強(qiáng)制】方法內(nèi)部單行注釋,在被注釋語(yǔ)句上方另起一行,使用//注釋。方法內(nèi)部多行注釋
使用/* */注釋,注意與代碼對(duì)齊。?
5. 【強(qiáng)制】所有的枚舉類型字段必須要有注釋,說(shuō)明每個(gè)數(shù)據(jù)項(xiàng)的用途。?
6. 【推薦】與其“半吊子”英文來(lái)注釋,不如用中文注釋把問(wèn)題說(shuō)清楚。專有名詞與關(guān)鍵字保持
英文原文即可。?
反例:“TCP 連接超時(shí)”解釋成“傳輸控制協(xié)議連接超時(shí)”,理解反而費(fèi)腦筋。?
7. 【推薦】代碼修改的同時(shí),注釋也要進(jìn)行相應(yīng)的修改,尤其是參數(shù)、返回值、異常、核心邏輯
等的修改。?
說(shuō)明:代碼與注釋更新不同步,就像路網(wǎng)與導(dǎo)航軟件更新不同步一樣,如果導(dǎo)航軟件嚴(yán)重滯后,
就失去了導(dǎo)航的意義。?
8. 【參考】合理處理注釋掉的代碼。在上方詳細(xì)說(shuō)明,而不是簡(jiǎn)單的注釋掉。如果無(wú)用,則刪除。?
說(shuō)明:代碼被注釋掉有兩種可能性:1)后續(xù)會(huì)恢復(fù)此段代碼邏輯。2)永久不用。前者如果沒(méi)
有備注信息,難以知曉注釋動(dòng)機(jī)。后者建議直接刪掉(代碼倉(cāng)庫(kù)保存了歷史代碼)。?
9. 【參考】對(duì)于注釋的要求:
第一、能夠準(zhǔn)確反應(yīng)設(shè)計(jì)思想和代碼邏輯;第二、能夠描述業(yè)務(wù)含
義,使別的程序員能夠迅速了解到代碼背后的信息。完全沒(méi)有注釋的大段代碼對(duì)于閱讀者形同
天書(shū),注釋是給自己看的,即使隔很長(zhǎng)時(shí)間,也能清晰理解當(dāng)時(shí)的思路;注釋也是給繼任者看
的,使其能夠快速接替自己的工作。?
10. 【參考】好的命名、代碼結(jié)構(gòu)是自解釋的,注釋力求精簡(jiǎn)準(zhǔn)確、表達(dá)到位。避免出現(xiàn)注釋的
一個(gè)極端:過(guò)多過(guò)濫的注釋,代碼的邏輯一旦修改,修改注釋是相當(dāng)大的負(fù)擔(dān)。?
反例:?
// put elephant into fridge?
put(elephant, fridge);?
?方法名 put,加上兩個(gè)有意義的變量名 elephant 和 fridge,已經(jīng)說(shuō)明了這是在干什么,語(yǔ)
義清晰的代碼不需要額外的注釋。?
11. 【參考】特殊注釋標(biāo)記,請(qǐng)注明標(biāo)記人與標(biāo)記時(shí)間。注意及時(shí)處理這些標(biāo)記,通過(guò)標(biāo)記掃描,
經(jīng)常清理此類標(biāo)記。線上故障有時(shí)候就是來(lái)源于這些標(biāo)記處的代碼。?
?1) 待辦事宜(TODO):( 標(biāo)記人,標(biāo)記時(shí)間,[預(yù)計(jì)處理時(shí)間])?
?表示需要實(shí)現(xiàn),但目前還未實(shí)現(xiàn)的功能。這實(shí)際上是一個(gè) Javadoc 的標(biāo)簽,目前的 Javadoc
還沒(méi)有實(shí)現(xiàn),但已經(jīng)被廣泛使用。只能應(yīng)用于類,接口和方法(因?yàn)樗且粋€(gè) Javadoc 標(biāo)簽)。
?
?2) 錯(cuò)誤,不能工作(FIXME):(標(biāo)記人,標(biāo)記時(shí)間,[預(yù)計(jì)處理時(shí)間])?
?在注釋中用 FIXME 標(biāo)記某代碼是錯(cuò)誤的,而且不能工作,需要及時(shí)糾正的情況。?
?
(九) 其它?
1. 【強(qiáng)制】在使用正則表達(dá)式時(shí),利用好其預(yù)編譯功能,可以有效加快正則匹配速度。?
說(shuō)明:不要在方法體內(nèi)定義:Pattern pattern = Pattern.compile(規(guī)則);?
2. 【強(qiáng)制】velocity 調(diào)用 POJO 類的屬性時(shí),建議直接使用屬性名取值即可,模板引擎會(huì)自動(dòng)按
規(guī)范調(diào)用 POJO 的 getXxx(),如果是 boolean 基本數(shù)據(jù)類型變量(boolean 命名不需要加 is
前綴),會(huì)自動(dòng)調(diào)用 isXxx()方法。?
說(shuō)明:注意如果是 Boolean 包裝類對(duì)象,優(yōu)先調(diào)用 getXxx()的方法。?
3. 【強(qiáng)制】后臺(tái)輸送給頁(yè)面的變量必須加$!{var}——中間的感嘆號(hào)。?
說(shuō)明:如果 var=null 或者不存在,那么${var}會(huì)直接顯示在頁(yè)面上。?
4. 【強(qiáng)制】注意 Math.random() 這個(gè)方法返回是 double 類型,注意取值的范圍 0≤x<1(能夠
取到零值,注意除零異常),如果想獲取整數(shù)類型的隨機(jī)數(shù),不要將 x 放大 10 的若干倍然后
取整,直接使用 Random 對(duì)象的 nextInt 或者 nextLong 方法。?
5. 【強(qiáng)制】獲取當(dāng)前毫秒數(shù) System.currentTimeMillis(); 而不是 new Date().getTime();?
說(shuō)明:如果想獲取更加精確的納秒級(jí)時(shí)間值,使用 System.nanoTime()的方式。在 JDK8 中,
針對(duì)統(tǒng)計(jì)時(shí)間等場(chǎng)景,推薦使用 Instant 類。?
6. 【推薦】不要在視圖模板中加入任何復(fù)雜的邏輯。?
說(shuō)明:根據(jù) MVC 理論,視圖的職責(zé)是展示,不要搶模型和控制器的活。?
7. 【推薦】任何數(shù)據(jù)結(jié)構(gòu)的構(gòu)造或初始化,都應(yīng)指定大小,避免數(shù)據(jù)結(jié)構(gòu)無(wú)限增長(zhǎng)吃光內(nèi)存。?
8. 【推薦】對(duì)于“明確停止使用的代碼和配置”,如方法、變量、類、配置文件、動(dòng)態(tài)配置屬性
等要堅(jiān)決從程序中清理出去,避免造成過(guò)多垃圾。?
?
總結(jié)
- 上一篇: Mysql字符串截取函数SUBSTRIN
- 下一篇: 易语言多线程许可证的几点理解