日韩无码专区无码一级三级片|91人人爱网站中日韩无码电影|厨房大战丰满熟妇|AV高清无码在线免费观看|另类AV日韩少妇熟女|中文日本大黄一级黄色片|色情在线视频免费|亚洲成人特黄a片|黄片wwwav色图欧美|欧亚乱色一区二区三区

RELATEED CONSULTING
相關咨詢
選擇下列產品馬上在線溝通
服務時間:8:30-17:00
你可能遇到了下面的問題
關閉右側工具欄

新聞中心

這里有您想知道的互聯(lián)網營銷解決方案
90%的Java開發(fā)人員都會犯的五個錯誤

?前言

作為一名java開發(fā)程序員,不知道大家有沒有遇到過一些匪夷所思的bug。這些錯誤通常需要您幾個小時才能解決。當你找到它們的時候,你可能會默默地罵自己是個傻瓜。是的,這些可笑的bug基本上都是你忽略了一些基礎知識造成的。其實都是很低級的錯誤。今天,我總結一些常見的編碼錯誤,然后給出解決方案。希望大家在日常編碼中能夠避免這樣的問題。

創(chuàng)新互聯(lián)長期為成百上千客戶提供的網站建設服務,團隊從業(yè)經驗10年,關注不同地域、不同群體,并針對不同對象提供差異化的產品和服務;打造開放共贏平臺,與合作伙伴共同營造健康的互聯(lián)網生態(tài)環(huán)境。為秀英企業(yè)提供專業(yè)的成都網站制作、網站設計,秀英網站改版等技術服務。擁有十余年豐富建站經驗和眾多成功案例,為您定制開發(fā)。

1. 使用Objects.equals比較對象

這種方法相信大家并不陌生,甚至很多人都經常使用。是JDK7提供的一種方法,可以快速實現對象的比較,有效避免煩人的空指針檢查。但是這種方法很容易用錯,例如:

Long longValue = 123L;
System.out.println(longValue==123); //true
System.out.println(Objects.equals(longValue,123)); //false

為什么替換==為Objects.equals()?會導致不同的結果?這是因為使用==?編譯器會得到封裝類型對應的基本數據類型longValue,然后與這個基本數據類型進行比較,相當于編譯器會自動將常量轉換為比較基本數據類型, 而不是包裝類型。

使用該Objects.equals()?方法后,編譯器默認常量的基本數據類型為int?。下面是源碼Objects.equals()?,其中a.equals(b)?使用的是Long.equals()?會判斷對象類型,因為編譯器已經認為常量是int?類型,所以比較結果一定是false。

public static boolean equals(Object a, Object b) {
return (a == b) || (a != null && a.equals(b));
}

public boolean equals(Object obj) {
if (obj instanceof Long) {
return value == ((Long)obj).longValue();
}
return false;
}

知道了原因,解決方法就很簡單了。直接聲明常量的數據類型,如Objects.equals(longValue,123L)。其實如果邏輯嚴密,就不會出現上面的問題。我們需要做的是保持良好的編碼習慣。

2. 日期格式錯誤

在我們日常的開發(fā)中,經常需要對日期進行格式化,但是很多人使用的格式不對,導致出現意想不到的情況。請看下面的例子。

Instant instant = Instant.parse("2021-12-31T00:00:00.00Z");
DateTimeFormatter formatter = DateTimeFormatter.ofPattern("YYYY-MM-dd HH:mm:ss")
.withZone(ZoneId.systemDefault());
System.out.println(formatter.format(instant));//2022-12-31 08:00:00

以上用于YYYY-MM-dd?格式化, 年從2021? 變成了 2022?。為什么?這是因為 java? 的DateTimeFormatter? 模式YYYY和yyyy?之間存在細微的差異。它們都代表一年,但是yyyy?代表日歷年,而YYYY?代表星期。這是一個細微的差異,僅會導致一年左右的變更問題,因此您的代碼本可以一直正常運行,而僅在新的一年中引發(fā)問題。12月31日按周計算的年份是2022年,正確的方式應該是使用yyyy-MM-dd格式化日期。

這個bug特別隱蔽。這在平時不會有問題。它只會在新的一年到來時觸發(fā)。我公司就因為這個bug造成了生產事故。

3. 在 ThreadPool 中使用 ThreadLocal

如果創(chuàng)建一個ThreadLocal? 變量,訪問該變量的線程將創(chuàng)建一個線程局部變量。合理使用ThreadLocal可以避免線程安全問題。

但是,如果在線程池中使用ThreadLocal ,就要小心了。您的代碼可能會產生意想不到的結果。舉個很簡單的例子,假設我們有一個電商平臺,用戶購買商品后需要發(fā)郵件確認。

private ThreadLocal currentUser = ThreadLocal.withInitial(() -> null);

private ExecutorService executorService = Executors.newFixedThreadPool(4);

public void executor() {
executorService.submit(()->{
User user = currentUser.get();
Integer userId = user.getId();
sendEmail(userId);
});
}

如果我們使用ThreadLocal?來保存用戶信息,這里就會有一個隱藏的bug。因為使用了線程池,線程是可以復用的,所以在使用ThreadLocal獲取用戶信息的時候,很可能會誤獲取到別人的信息。您可以使用會話來解決這個問題。

4. 使用HashSet去除重復數據

在編碼的時候,我們經常會有去重的需求。一想到去重,很多人首先想到的就是用HashSet?去重。但是,不小心使用 HashSet 可能會導致去重失敗。

User user1 = new User();
user1.setUsername("test");

User user2 = new User();
user2.setUsername("test");

List users = Arrays.asList(user1, user2);
HashSet sets = new HashSet<>(users);
System.out.println(sets.size());// the size is 2

細心的讀者應該已經猜到失敗的原因了。HashSet?使用hashcode?對哈希表進行尋址,使用equals?方法判斷對象是否相等。如果自定義對象沒有重寫hashcode?方法和equals方法,則默認使用父對象的hashcode?方法和equals?方法。所以HashSet會認為這是兩個不同的對象,所以導致去重失敗。

5. 線程池中的異常被吃掉

ExecutorService executorService = Executors.newFixedThreadPool(1);
executorService.submit(()->{
//do something
double result = 10/0;
});

上面的代碼模擬了一個線程池拋出異常的場景。我們真正的業(yè)務代碼要處理各種可能出現的情況,所以很有可能因為某些特定的原因而觸發(fā)RuntimeException 。

但是如果沒有特殊處理,這個異常就會被線程池吃掉。這樣就會導出出現問題你都不知道,這是很嚴重的后果。因此,最好在線程池中try catch捕獲異常。

總結

本文總結了在開發(fā)過程中很容易犯的5個錯誤,希望大家養(yǎng)成良好的編碼習慣。


本文名稱:90%的Java開發(fā)人員都會犯的五個錯誤
文章URL:http://www.5511xx.com/article/coeihpi.html