Java語(yǔ)言最開(kāi)始是為了交互電視機(jī)而開(kāi)發(fā)的,隨著時(shí)間的推移,他已經(jīng)廣泛應(yīng)用各種軟件開(kāi)發(fā)領(lǐng)域?;诿嫦?qū)ο蟮脑O(shè)計(jì),屏蔽了諸如C,C++等語(yǔ)言的一些復(fù)雜性,提供了垃圾回收機(jī)制,平臺(tái)無(wú)關(guān)的虛擬機(jī)技術(shù),Java創(chuàng)造了一種前所未有的開(kāi)發(fā)方式。另一方面,得益于Java提出的“一次編碼,到處運(yùn)行”的口號(hào),讓Java更加出名。但是Java中的異常也是處處發(fā)生,下面我就列出了我認(rèn)為的Java開(kāi)發(fā)最容易出現(xiàn)的5個(gè)錯(cuò)誤。
創(chuàng)新互聯(lián)專注于樂(lè)業(yè)企業(yè)網(wǎng)站建設(shè),響應(yīng)式網(wǎng)站建設(shè),商城開(kāi)發(fā)。樂(lè)業(yè)網(wǎng)站建設(shè)公司,為樂(lè)業(yè)等地區(qū)提供建站服務(wù)。全流程按需求定制制作,專業(yè)設(shè)計(jì),全程項(xiàng)目跟蹤,創(chuàng)新互聯(lián)專業(yè)和態(tài)度為您提供的服務(wù)
1、重復(fù)造輪子
一個(gè)明顯的錯(cuò)誤就是Java程序員習(xí)慣性的忽略已經(jīng)存在的大量的庫(kù)。在你決定造一個(gè)輪子之間,我建議你試著先搜一下是否有已經(jīng)存在庫(kù)。例如日志方面,有l(wèi)ogback,新log4j,網(wǎng)絡(luò)方面,有Netty或者Akka。有一些庫(kù),已經(jīng)逐步變成了標(biāo)準(zhǔn),比如Java8中加入的Joda-Time。
下面講述的是我上一個(gè)項(xiàng)目中的個(gè)人經(jīng)歷。有一部分用于HTML轉(zhuǎn)義的代碼是一個(gè)開(kāi)發(fā)自己完成的。這個(gè)代碼正常工作了多年,但是又一次遇到了一個(gè)用戶輸入,代碼陷入了死循環(huán)。這個(gè)用戶發(fā)現(xiàn)應(yīng)用沒(méi)有反應(yīng),又重新輸入了一遍,服務(wù)器因?yàn)檫@個(gè)死循環(huán)掛了。如果這個(gè)開(kāi)發(fā)使用已有的HTML轉(zhuǎn)義工具,比如Google Guava項(xiàng)目提供的HtmlEscaper,這個(gè)嚴(yán)重的問(wèn)題可能就不會(huì)出現(xiàn)。并且現(xiàn)在市面上流行的大部分的開(kāi)源庫(kù),背后都有團(tuán)隊(duì)和社區(qū)在支持,類似這樣的錯(cuò)誤,都能夠及時(shí)的被修復(fù)。
2、在Switch-Case中錯(cuò)誤的使用break
這是一個(gè)很尷尬的問(wèn)題,但是仍然在實(shí)際開(kāi)發(fā)中經(jīng)常出現(xiàn)。瀑布特性在switch語(yǔ)句中有時(shí)會(huì)非常有用,但是必要的break關(guān)鍵字的缺失,有時(shí)會(huì)帶來(lái)災(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ò)誤,也可以通過(guò)類似FindBugs或者PMD這樣的工具檢查出來(lái)。
3、忘記釋放資源
一旦打開(kāi)一個(gè)文件,或者建立一個(gè)網(wǎng)絡(luò)連接,一個(gè)非常重要的習(xí)慣是記得關(guān)閉資源。并且一定記得,如果在使用類似這樣的資源過(guò)程中出現(xiàn)了錯(cuò)誤,在異常處理中,也需要做對(duì)應(yīng)的關(guān)閉操作??赡苡腥藭?huì)說(shuō),F(xiàn)ileInputStream對(duì)象在GC的時(shí)候,Java終結(jié)器(finalizer)會(huì)自動(dòng)調(diào)用其close()方法,但是我們知道,我們無(wú)法預(yù)知GC在什么時(shí)候開(kāi)始,所以我們無(wú)法預(yù)知在執(zhí)行GC之前,會(huì)有多少資源無(wú)法及時(shí)關(guān)閉。為了避免這種情況,Java7推出的try-with-resources語(yǔ)法,是值得每個(gè)開(kāi)發(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語(yǔ)法適用于所有實(shí)現(xiàn)了AutoClosable接口的類。它能保證每一個(gè)資源及時(shí)的關(guān)閉。
4、內(nèi)存泄露
Java使用自動(dòng)內(nèi)存管理,所以大部分時(shí)間,我們都不會(huì)去關(guān)心內(nèi)存的分配和釋放,但是,這并不意味著Java開(kāi)發(fā)人員需要忽略內(nèi)存。在Java應(yīng)用中,內(nèi)存的問(wèn)題也經(jīng)常出現(xiàn)。我們知道,對(duì)象如果沒(méi)有被引用了,這個(gè)對(duì)象就會(huì)被釋放,但是并不意味著,就不會(huì)出現(xiàn)內(nèi)存泄露的問(wèn)題。在Java中,造成內(nèi)存泄露的原因有很多,但最容易出現(xiàn)的情況就是對(duì)象引用無(wú)法釋放,因?yàn)镚C在回收堆內(nèi)存的時(shí)候,如果一個(gè)對(duì)象仍然被其他對(duì)象引用,這個(gè)對(duì)象空間是不會(huì)被回收的,舉個(gè)例子,如果在類中,有一個(gè)靜態(tài)字段引用到一個(gè)集合,假如我們沒(méi)有手動(dòng)的在使用完成這個(gè)集合之后,將他設(shè)置為null,那么這個(gè)集合及這個(gè)集合中的對(duì)象,是永遠(yuǎn)不會(huì)被回收的,因?yàn)轭愳o態(tài)字段是不會(huì)被GC的。
比如還有一種造成內(nèi)存泄露的原因,就是一組對(duì)象互相引用對(duì)方,就是我們經(jīng)常說(shuō)的循環(huán)引用,因?yàn)檠h(huán)引用,所以GC不能確定這些互相引用的對(duì)象是否還有繼續(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方法來(lái)代替peekLast方法,因?yàn)閜ollLast方法會(huì)在拿到最后一個(gè)元素之后,把這個(gè)元素從deque中移除。
5、過(guò)度產(chǎn)生垃圾數(shù)據(jù)
過(guò)度產(chǎn)生垃圾數(shù)據(jù)的意思,是程序運(yùn)行中大量產(chǎn)生短聲明周期的對(duì)象。這回導(dǎo)致GC頻繁的執(zhí)行,從內(nèi)存中回收空間,GC的執(zhí)行是需要完成堆掃描的,這對(duì)系統(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è)新的字符串對(duì)象。為了改進(jìn)這種代碼,我們可以使用StringBuilder來(lái)代替:
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ì)提高不少的性能。
以上總結(jié)了5個(gè)Java開(kāi)發(fā)人員最常犯的錯(cuò)誤,是我基于大量的github上的開(kāi)源項(xiàng)目,Stack overflow上的問(wèn)題,還有一些流行的google搜索的分析,希望分享能給你帶來(lái)幫助。
當(dāng)前標(biāo)題:程序員最容易犯的錯(cuò),你中了幾個(gè)?
標(biāo)題來(lái)源:http://m.rwnh.cn/article24/gdisce.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供ChatGPT、網(wǎng)站維護(hù)、標(biāo)簽優(yōu)化、商城網(wǎng)站、、電子商務(wù)
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來(lái)源: 創(chuàng)新互聯(lián)