java作為最受歡迎程度榜榜首語言,自然是廣大開發(fā)者使用最多的語言。正因?yàn)橛腥绱藦V泛的使用性,java開發(fā)中發(fā)生異常也比比皆是,接下來我們就來看看那些java開發(fā)中最容易出現(xiàn)的那些錯(cuò)誤。
目前成都創(chuàng)新互聯(lián)公司已為1000+的企業(yè)提供了網(wǎng)站建設(shè)、域名、網(wǎng)站空間、網(wǎng)站托管運(yùn)營、企業(yè)網(wǎng)站設(shè)計(jì)、玉田網(wǎng)站維護(hù)等服務(wù),公司將堅(jiān)持客戶導(dǎo)向、應(yīng)用為本的策略,正道將秉承"和諧、參與、激情"的文化,與客戶和合作伙伴齊心協(xié)力一起成長,共同發(fā)展。1、重復(fù)造輪子
一個(gè)明顯的錯(cuò)誤就是Java程序員習(xí)慣性的忽略已經(jīng)存在的大量的庫。在你決定造一個(gè)輪子之間,我建議你試著先搜一下是否有已經(jīng)存在庫。例如日志方面,有l(wèi)ogback,新log4j,網(wǎng)絡(luò)方面,有Netty或者Akka。有一些庫,已經(jīng)逐步變成了標(biāo)準(zhǔn),比如Java8中加入的Joda-Time。
下面講述的是我上一個(gè)項(xiàng)目中的個(gè)人經(jīng)歷。有一部分用于HTML轉(zhuǎn)義的代碼是一個(gè)開發(fā)自己完成的。這個(gè)代碼正常工作了多年,但是又一次遇到了一個(gè)用戶輸入,代碼陷入了死循環(huán)。這個(gè)用戶發(fā)現(xiàn)應(yīng)用沒有反應(yīng),又重新輸入了一遍,服務(wù)器因?yàn)檫@個(gè)死循環(huán)掛了。如果這個(gè)開發(fā)使用已有的HTML轉(zhuǎn)義工具,比如Google Guava項(xiàng)目提供的HtmlEscaper,這個(gè)嚴(yán)重的問題可能就不會(huì)出現(xiàn)。并且現(xiàn)在市面上流行的大部分的開源庫,背后都有團(tuán)隊(duì)和社區(qū)在支持,類似這樣的錯(cuò)誤,都能夠及時(shí)的被修復(fù)。
2、在Switch-Case中錯(cuò)誤的使用break
這是一個(gè)很尷尬的問題,但是仍然在實(shí)際開發(fā)中經(jīng)常出現(xiàn)。瀑布特性在switch語句中有時(shí)會(huì)非常有用,但是必要的break關(guān)鍵字的缺失,有時(shí)會(huì)帶來災(zāi)難性的后果。比如在下面的代碼中,如果在case 0中忘記放一個(gè)break關(guān)鍵字,代碼會(huì)繼續(xù)向下執(zhí)行,就會(huì)在Zero之后再輸出一個(gè)One:
public static void switchCasePrimer() {
int caseIndex = 0;
switch (caseIndex) {
case 0:
System.out.println("Zero");
case 1:
System.out.println("One");
break;
case 2:
System.out.println("Two");
break;
default:
System.out.println("Default");
}
}
最好的解決辦法是使用多態(tài),并把不同的處理代碼放到子類中。當(dāng)然,類似這樣的錯(cuò)誤,也可以通過類似FindBugs或者PMD這樣的工具檢查出來。
3、忘記釋放資源
一旦打開一個(gè)文件,或者建立一個(gè)網(wǎng)絡(luò)連接,一個(gè)非常重要的習(xí)慣是記得關(guān)閉資源。并且一定記得,如果在使用類似這樣的資源過程中出現(xiàn)了錯(cuò)誤,在異常處理中,也需要做對應(yīng)的關(guān)閉操作。可能有人會(huì)說,F(xiàn)ileInputStream對象在GC的時(shí)候,Java終結(jié)器(finalizer)會(huì)自動(dòng)調(diào)用其close()方法,但是我們知道,我們無法預(yù)知GC在什么時(shí)候開始,所以我們無法預(yù)知在執(zhí)行GC之前,會(huì)有多少資源無法及時(shí)關(guān)閉。為了避免這種情況,Java7推出的try-with-resources語法,是值得每個(gè)開發(fā)使用的。
private static void printFileJava7() throws IOException {
try(FileInputStream input = new FileInputStream("file.txt")) {
int data = input.read();
while(data != -1){
System.out.print((char) data);
data = input.read();
}
}
}
try-with-resources語法適用于所有實(shí)現(xiàn)了AutoClosable接口的類。它能保證每一個(gè)資源及時(shí)的關(guān)閉。
4、內(nèi)存泄露
Java使用自動(dòng)內(nèi)存管理,所以大部分時(shí)間,我們都不會(huì)去關(guān)心內(nèi)存的分配和釋放,但是,這并不意味著Java開發(fā)人員需要忽略內(nèi)存。在Java應(yīng)用中,內(nèi)存的問題也經(jīng)常出現(xiàn)。我們知道,對象如果沒有被引用了,這個(gè)對象就會(huì)被釋放,但是并不意味著,就不會(huì)出現(xiàn)內(nèi)存泄露的問題。在Java中,造成內(nèi)存泄露的原因有很多,但最容易出現(xiàn)的情況就是對象引用無法釋放,因?yàn)镚C在回收堆內(nèi)存的時(shí)候,如果一個(gè)對象仍然被其他對象引用,這個(gè)對象空間是不會(huì)被回收的,舉個(gè)例子,如果在類中,有一個(gè)靜態(tài)字段引用到一個(gè)集合,假如我們沒有手動(dòng)的在使用完成這個(gè)集合之后,將他設(shè)置為null,那么這個(gè)集合及這個(gè)集合中的對象,是永遠(yuǎn)不會(huì)被回收的,因?yàn)轭愳o態(tài)字段是不會(huì)被GC的。
比如還有一種造成內(nèi)存泄露的原因,就是一組對象互相引用對方,就是我們經(jīng)常說的循環(huán)引用,因?yàn)檠h(huán)引用,所以GC不能確定這些互相引用的對象是否還有繼續(xù)存活的必要。還有一種情況,就是使用JNI時(shí)的非堆內(nèi)存泄露。
一個(gè)典型的內(nèi)存泄露例子:
final ScheduledExecutorService scheduledExecutorService = Executors.newScheduledThreadPool(1);final Deque numbers = new LinkedBlockingDeque<>();final BigDecimal divisor = new BigDecimal(51);
scheduledExecutorService.scheduleAtFixedRate(() -> {
BigDecimal number = numbers.peekLast();
if (number != null && number.remainder(divisor).byteValue() == 0) {
System.out.println("Number: " + number);
System.out.println("Deque size: " + numbers.size());
}
}, 10, 10, TimeUnit.MILLISECONDS);
scheduledExecutorService.scheduleAtFixedRate(() -> {
numbers.add(new BigDecimal(System.currentTimeMillis()));
}, 10, 10, TimeUnit.MILLISECONDS);
try {
scheduledExecutorService.awaitTermination(1, TimeUnit.DAYS);
} catch (InterruptedException e) {
e.printStackTrace();
}
在上面的例子中,我們創(chuàng)建了兩個(gè)定時(shí)任務(wù)。第一個(gè)定時(shí)任務(wù),從deque中獲取了最后的一個(gè)數(shù)字”numbers”并判斷,如果這個(gè)數(shù)字能被51整除,則打印該數(shù)字和deque的大小。第二個(gè)定時(shí)任務(wù),不斷的向deque中添加數(shù)據(jù)。兩個(gè)任務(wù)都間隔10ms執(zhí)行。如果這個(gè)代碼執(zhí)行,你會(huì)發(fā)現(xiàn),deque的大小會(huì)持續(xù)的增加,直到deque中的數(shù)據(jù)占滿整個(gè)堆空間。為了阻止這種情況的發(fā)生,我們可以使用pollLast方法來代替peekLast方法,因?yàn)閜ollLast方法會(huì)在拿到最后一個(gè)元素之后,把這個(gè)元素從deque中移除。
5、過度產(chǎn)生垃圾數(shù)據(jù)
過度產(chǎn)生垃圾數(shù)據(jù)的意思,是程序運(yùn)行中大量產(chǎn)生短聲明周期的對象。這回導(dǎo)致GC頻繁的執(zhí)行,從內(nèi)存中回收空間,GC的執(zhí)行是需要完成堆掃描的,這對系統(tǒng)的性能影響是非常大的。下面是一個(gè)小例子:
String oneMillionHello = "";for (int i = 0; i < 1000000; i++) {
oneMillionHello = oneMillionHello + "Hello!";
}
System.out.println(oneMillionHello.substring(0, 6));
在Java中,字符串是不可變的,所以每一次循環(huán)都會(huì)創(chuàng)建一個(gè)新的字符串對象。為了改進(jìn)這種代碼,我們可以使用StringBuilder來代替:
StringBuilder oneMillionHelloSB = new StringBuilder();
for (int i = 0; i < 1000000; i++) {
oneMillionHelloSB.append("Hello!");
}
System.out.println(oneMillionHelloSB.toString().substring(0, 6));
第二個(gè)版本的代碼,在執(zhí)行的時(shí)候會(huì)提高不少的性能。
這些錯(cuò)誤每個(gè)都很經(jīng)典,解決它們對你的能力絕對是極大的提升。當(dāng)然這并不是全部,如果還想了解關(guān)于這方面的其他內(nèi)容,不妨關(guān)注我們,接下來,我還會(huì)為你帶來容易出現(xiàn)的其他問題。希望可以幫你在java之路上掃清障礙。
創(chuàng)新互聯(lián)www.cdcxhl.cn,專業(yè)提供香港、美國云服務(wù)器,動(dòng)態(tài)BGP最優(yōu)骨干路由自動(dòng)選擇,持續(xù)穩(wěn)定高效的網(wǎng)絡(luò)助力業(yè)務(wù)部署。公司持有工信部辦法的idc、isp許可證, 機(jī)房獨(dú)有T級流量清洗系統(tǒng)配攻擊溯源,準(zhǔn)確進(jìn)行流量調(diào)度,確保服務(wù)器高可用性。佳節(jié)活動(dòng)現(xiàn)已開啟,新人活動(dòng)云服務(wù)器買多久送多久。