Java中代碼Bug記錄--泛型失效、數組刪除、HashMap死迴圈

来源:https://www.cnblogs.com/songjiyang/archive/2023/07/26/17582799.html
-Advertisement-
Play Games

最近在工作的過程中,遇到了不少奇怪自己或者同事的Bug,都是一些出乎意料的,不太容易發現的,記錄一下來幫助可能也遇到了這些Bug的人 # 1. 編譯時泛型校驗失效 ```java Map nameToType = new HashMap(); nameToType.put( "testName", ...


最近在工作的過程中,遇到了不少奇怪自己或者同事的Bug,都是一些出乎意料的,不太容易發現的,記錄一下來幫助可能也遇到了這些Bug的人

1. 編譯時泛型校驗失效

Map<String, String> nameToType = new HashMap<>();
nameToType.put( "testName", 123 ); // java: 不相容的類型: int無法轉換為java.lang.String

上面的代碼,我們很容易看出來,無法通過編譯,因為Map的value需要的是一個String,但我們傳的是一個int。但我只要稍微改一下:

package generic;

import java.util.HashMap;
import java.util.Map;

public class RecContext<T>{

	private Map<String, String> nameToType = new HashMap<>();

	public Map<String, String> getNameToType(){

		return nameToType;
	}

	public void setNameToType( Map<String, String> nameToType ){

		this.nameToType = nameToType;
	}

	public static void main( String[] args ){

		RecContext recContextRaw = new RecContext();
		recContextRaw.getNameToType().put( "testName", 123 );

	}
}

同樣是一個value要求為String的Map, 放到一個對象裡面就可以通過編譯了

不過這不是一個普通的對象,這個Class本身帶泛型,但我們在使用的時候,沒有指定這個泛型,也就是IDEA中常常報的錯,說你在Raw Use這個類型

也正是因為Raw Use了這個類,所以導致它的泛型屬性也被類型擦除了,具體可以看StackOverflow-What is a raw type and why shouldn't we use it?

這篇文章裡面是這麼說的

In simpler terms, when a raw type is used, the constructors, instance methods and non-static fields are also erased

簡單地講,當使用了原始類型,構造器、實例方法和非靜態的欄位都會被擦除

我們有一個老的項目裡面有不少這樣的Raw use,也正好有另外一個同事把一個其他的類型插入了這個Map,於是就報了一個類型轉換錯誤,同事們都很震驚,認為這個Map不是有泛型嗎,怎麼可能插入別的類型,一番排查,才發現是這個問題

解決方法

  1. 不要Raw use類,會使編譯校驗失效,也有很多其他的理由,可以參考上面那篇文章,我不使用的原因是因為IDEA每次都會發出告警。還有如果發現這個類一直在Raw use也沒啥問題,說明這個類本身就不需要泛型,可以考慮是一下是不是需要重構一下

image.png

  • 我的告警配置的是黃色的背景,看起來很惹眼
  • IDEA的告警會是Raw use,不可能的條件判斷,可以更換寫法的代碼(完全不影響效果),甚至可能是bug(之前碰到過,修複的時候發現IDEA已經提示了,但是可能別人的告警不是很明顯,沒看到)
  1. 如果是真的Raw use了類,那檢查類型是否錯誤的責任就落到我們自己的頭上,確保不要傳進錯誤的類型,確保取出來的類型不要轉換錯誤

2. 數組刪除中的“刻舟求劍”

線上代碼中有這樣一個邏輯,想從帖子列表中篩選出一部分帖子然後從原列表中刪除,其代碼邏輯如下:

import java.util.ArrayList;
import java.util.List;

public class ListDelTest{

	public static void main( String[] args ){

		List<Integer> jobIds = new ArrayList<>();
		for( int i = 0; i < 10; i++ ){
			jobIds.add( i );
		}

		List<Integer> delIndex = new ArrayList<>();
		for( int i = 0; i < jobIds.size(); i++ ){

			// 線上是其他的篩選邏輯,在這我們用偶數代替
			if( jobIds.get( i ) % 2 == 0 ){

				delIndex.add( i );
			}
		}

		for( int i = 0; i < jobIds.size(); i++ ){
			if( delIndex.contains( i ) ){
				jobIds.remove( jobIds.get( i ) );
			}
		}

		System.out.println( jobIds );
		// [1, 2, 4, 5, 7, 8]
	}
}

可以看到輸出結果中,並不符合我們的預期,我們希望的是把所有的偶數刪除,結果中不但有偶數,而且一些奇數也不見了

這個其實很容易理解,因為我們記得位置是0,2,4,6,8

原始數據是0,1,2,3,4,5,6,7,8,9
當我們刪除了0時,數據變成了1,2,3,4,5,6,7,8,9

這時候我們再去刪除index是2的值,結果就把3這個值給刪除了

解決方法

  1. 使用stream filter collect,這種其實不是原地刪除,而是新建了一個List, 不過我們把這個List覆蓋原來的引用,效果一樣
  2. 常見的邊遍歷邊刪除的方法,使用Iterator,可以避免ConcurrentModificationException異常
		int i = 0;
		Iterator<Integer> iterator = jobIds.iterator();
		while( iterator.hasNext() ){
			iterator.next();
			if( delIndex.contains( i ) ){
				iterator.remove();
			}
			i++;
		}

		System.out.println( jobIds );
		// [1, 3, 5, 7, 9]
  1. 倒著刪除,這樣不會影響我們已經記錄過的index,我記得當時我在華為OD面試的時候一個面試官問的我的一個問題,我沒答出來,他告訴我的答案
		for( int i = jobIds.size() - 1; i > -1; i-- ){
			if( delIndex.contains( i ) ){
				jobIds.remove( jobIds.get( i ) );
			}
		}

3. Java8 HashMap死迴圈

線上同事上線了一個新的過濾器,我們的過濾器是併發執行的,比如,帖子敏感詞過濾,會將帖子分成10份,用10個線程分別執行,執行完了就把結果放到一個公共的map中

很明顯,這個map是有線程安全問題的,因為會有多個線程同時去put,然而,因為沒考慮到,同事使用了普通的HashMap

線上的現象就是,每過一段時間,某個機器的CPU使用率就到了90%以上,需要重啟

按照我們的理解,就算是有併發問題,怎麼會使CPU使用變高呢

我們都背過,在Java1.7中的HashMap會因為併發插入產生死循,1.8使用尾插法代替頭插法解決了死迴圈

可我們用的是Java1.8,看起來好像還是有死迴圈的問題

具體原因我就不仔細分析了,是在鏈表轉換樹或者對樹進行操作的時候會出現線程安全的問題

可以參考HashMap在jdk1.8也會出現死迴圈的問題

解決方法

  1. 多線程還是需要使用ConcurrentHashMap

參考

[1] StackOverflow-What is a raw type and why shouldn't we use it?

[2] HashMap在jdk1.8也會出現死迴圈的問題

本文來自博客園,作者:songtianer,轉載請註明原文鏈接:https://www.cnblogs.com/songjiyang/p/17582799.html


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • >我們是[袋鼠雲數棧 UED 團隊](http://ued.dtstack.cn/),致力於打造優秀的一站式數據中台產品。我們始終保持工匠精神,探索前端道路,為社區積累並傳播經驗價值。 >本文作者:空山 # 前言 > 由於筆者最近在開發中遇到了一個重覆渲染導致子組件狀態值丟失的問題,因此關於性能優化 ...
  • ### 寫在前面 前面的文章中提到過,自己開始在博客園上更新文章。 說也奇怪,自己博客園賬號註冊了好久,都沒在上面更新過博客。 直到前段時間博客園的求助信息火了,才對博客園有了全新的認知。 博客園一個最大的特點就是簡潔、乾凈,廣告少。 但也有一個個人認為很不好的地方就是界面太醜,容易勸退新人。 直到 ...
  • 沒有足夠的特征數據,安全策略將是"無根之木,無源之水"。微信安全數據倉庫應運而生,成為整個安全業務的特征數據存儲中心,每天服務了萬億級的特征數據讀寫請求,為整個微信安全策略提供了可靠的數據支撐,是微信安全基石之所在。然而,微信安全數據倉庫不僅僅是一個存儲中心,更是一個特征管理和數據質量管理的中心。在... ...
  • ## 3.1、創建module #### 3.1.1、右擊project,創建新module ![image](https://img2023.cnblogs.com/blog/2052479/202307/2052479-20230725081202352-22924479.png) ### 3. ...
  • # 包 go程式由一個個不同的包組成,程式的入口是名為main的包,比如我們創建一個main文件 main.go ```go package main import "fmt" func main(){ fmt.Println("hello") } ``` go要求每一個go文件的非註釋開頭必須是` ...
  • 到目前為止,我們只討論了使用Docker來部署應用程式。然而,Docker也是一個極好的用於開發應用程式的工具。可以採用一些不同的建議來改善開發體驗。 - 在應用程式中使用`docker-compose`以方便開發。 - 使用綁定掛載將本地代碼掛載到容器文件系統中,以避免每次更改都需要重新構建容器映 ...
  • # java線程詳解 ## 線程 ### 概念 說到線程,就不得不提進程,為什麼呢,因為進程是操作系統進行分配資源和調度的最小單位,比如windows系統安裝的應用軟體(office、qq、微信等)啟動時,由操作系統協調分配資源和調度執行稱之為一個進程,進程間是相互獨立和隔離的。而線程是進程最小執行 ...
  • # JDBC API ## 獲取資料庫連接5種方式 1. 通過new創建Driver對象; 2. 使用反射載入Driver類,動態載入,減少依賴性,更加靈活; 3. 使用DriverManager 替代 Driver 進行統一管理,有了更好的擴展性; 4. 使用 Class.forName 自動完成 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...