Java開發(fā)人員最常犯的5個錯誤總結(jié)
前言
作為一名java開發(fā)程序員,不知道大家有沒有遇到過一些匪夷所思的bug。這些錯誤通常需要您幾個小時才能解決。當(dāng)你找到它們的時候,你可能會默默地罵自己是個傻瓜。是的,這些可笑的bug基本上都是你忽略了一些基礎(chǔ)知識造成的。其實都是很低級的錯誤。今天,我總結(jié)一些常見的編碼錯誤,然后給出解決方案。希望大家在日常編碼中能夠避免這樣的問題。
1. 使用Objects.equals比較對象
這種方法相信大家并不陌生,甚至很多人都經(jīng)常使用。是JDK7提供的一種方法,可以快速實現(xiàn)對象的比較,有效避免煩人的空指針檢查。但是這種方法很容易用錯,例如:
Long longValue = 123L; System.out.println(longValue==123); //true System.out.println(Objects.equals(longValue,123)); //false
為什么替換==
為Objects.equals()
會導(dǎo)致不同的結(jié)果?這是因為使用==
編譯器會得到封裝類型對應(yīng)的基本數(shù)據(jù)類型longValue
,然后與這個基本數(shù)據(jù)類型進(jìn)行比較,相當(dāng)于編譯器會自動將常量轉(zhuǎn)換為比較基本數(shù)據(jù)類型, 而不是包裝類型。
使用該Objects.equals()
方法后,編譯器默認(rèn)常量的基本數(shù)據(jù)類型為int
。下面是源碼Objects.equals()
,其中a.equals(b)
使用的是Long.equals()
會判斷對象類型,因為編譯器已經(jīng)認(rèn)為常量是int
類型,所以比較結(jié)果一定是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; }
知道了原因,解決方法就很簡單了。直接聲明常量的數(shù)據(jù)類型,如Objects.equals(longValue,123L)
。其實如果邏輯嚴(yán)密,就不會出現(xiàn)上面的問題。我們需要做的是保持良好的編碼習(xí)慣。
2. 日期格式錯誤
在我們?nèi)粘5拈_發(fā)中,經(jīng)常需要對日期進(jìn)行格式化,但是很多人使用的格式不對,導(dǎo)致出現(xiàn)意想不到的情況。請看下面的例子。
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
之間存在細(xì)微的差異。它們都代表一年,但是yyyy
代表日歷年,而YYYY
代表星期。這是一個細(xì)微的差異,僅會導(dǎo)致一年左右的變更問題,因此您的代碼本可以一直正常運(yùn)行,而僅在新的一年中引發(fā)問題。12月31日按周計算的年份是2022年,正確的方式應(yīng)該是使用yyyy-MM-dd
格式化日期。
這個bug
特別隱蔽。這在平時不會有問題。它只會在新的一年到來時觸發(fā)。我公司就因為這個bug造成了生產(chǎn)事故。
3. 在 ThreadPool 中使用 ThreadLocal
如果創(chuàng)建一個ThreadLocal
變量,訪問該變量的線程將創(chuàng)建一個線程局部變量。合理使用ThreadLocal
可以避免線程安全問題。
但是,如果在線程池中使用ThreadLocal
,就要小心了。您的代碼可能會產(chǎn)生意想不到的結(jié)果。舉個很簡單的例子,假設(shè)我們有一個電商平臺,用戶購買商品后需要發(fā)郵件確認(rèn)。
private ThreadLocal<User> 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。因為使用了線程池,線程是可以復(fù)用的,所以在使用ThreadLocal
獲取用戶信息的時候,很可能會誤獲取到別人的信息。您可以使用會話來解決這個問題。
4. 使用HashSet去除重復(fù)數(shù)據(jù)
在編碼的時候,我們經(jīng)常會有去重的需求。一想到去重,很多人首先想到的就是用HashSet
去重。但是,不小心使用 HashSet
可能會導(dǎo)致去重失敗。
User user1 = new User(); user1.setUsername("test"); User user2 = new User(); user2.setUsername("test"); List<User> users = Arrays.asList(user1, user2); HashSet<User> sets = new HashSet<>(users); System.out.println(sets.size());// the size is 2
細(xì)心的讀者應(yīng)該已經(jīng)猜到失敗的原因了。HashSet
使用hashcode
對哈希表進(jìn)行尋址,使用equals
方法判斷對象是否相等。如果自定義對象沒有重寫hashcode
方法和equals方法,則默認(rèn)使用父對象的hashcode
方法和equals
方法。所以HashSet
會認(rèn)為這是兩個不同的對象,所以導(dǎo)致去重失敗。
5. 線程池中的異常被吃掉
ExecutorService executorService = Executors.newFixedThreadPool(1); executorService.submit(()->{ //do something double result = 10/0; });
上面的代碼模擬了一個線程池拋出異常的場景。我們真正的業(yè)務(wù)代碼要處理各種可能出現(xiàn)的情況,所以很有可能因為某些特定的原因而觸發(fā)RuntimeException
。
但是如果沒有特殊處理,這個異常就會被線程池吃掉。這樣就會導(dǎo)出出現(xiàn)問題你都不知道,這是很嚴(yán)重的后果。因此,最好在線程池中try catch
捕獲異常。
到此這篇關(guān)于Java開發(fā)人員最常犯的5個錯誤總結(jié)的文章就介紹到這了,更多相關(guān)Java開發(fā)人員常犯的錯誤內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
SpringBoot項目獲取統(tǒng)一前綴配置及獲取非確定名稱配置方法
在SpringBoot項目中,使用@ConfigurationProperties注解可獲取統(tǒng)一前綴的配置,具體做法是創(chuàng)建配置類,使用prefix屬性指定配置的前綴,本文給大家介紹SpringBoot項目獲取統(tǒng)一前綴配置以及獲取非確定名稱配置方法,感興趣的朋友跟隨小編一起看看吧2024-09-09java如何將Object數(shù)組轉(zhuǎn)換為指定類型數(shù)組
這篇文章主要介紹了java如何將Object數(shù)組轉(zhuǎn)換為指定類型數(shù)組,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-08-08SpringBoot應(yīng)用部署于外置Tomcat容器的方法
這篇文章主要介紹了SpringBoot應(yīng)用部署于外置Tomcat容器的方法,本文分步驟給大家介紹的非常詳細(xì),具有一定的參考借鑒價值,需要的朋友可以參考下2018-06-06mybatis框架order by作為參數(shù)傳入時失效的解決
這篇文章主要介紹了mybatis框架order by作為參數(shù)傳入時失效的解決方案,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-06-06gradle使用maven-publish發(fā)布jar包上傳到私有maven配置
這篇文章主要介紹了gradle使用maven-publish發(fā)布jar包上傳到私有maven的配置示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-03-03詳解Java編程中包package的內(nèi)容與包對象的規(guī)范
這篇文章主要介紹了Java編程中包package的內(nèi)容與包對象的規(guī)范,是Java入門學(xué)習(xí)中的基礎(chǔ)知識,需要的朋友可以參考下2015-12-12