解析java稀疏數(shù)組如何幫助我們節(jié)省內存提升性能
什么是稀疏矩陣
稀疏矩陣是指矩陣中大部分元素為零的矩陣。在實際應用中,很多矩陣都是稀疏的,比如網(wǎng)絡圖、文本數(shù)據(jù)等。由于矩陣中存在大量的零元素,因此稀疏矩陣的存儲和計算都具有一定的特殊性。
一般來說,在矩陣中,若數(shù)值為0的元素數(shù)目遠遠多于非0元素的數(shù)目,并且非0元素分布沒有規(guī)律時,則稱該矩陣為稀疏矩陣;與之相反,若非0元素數(shù)目占大多數(shù)時,則稱該矩陣為稠密矩陣。下面的矩陣就是一個典型的稀疏矩陣:
優(yōu)化稀疏矩陣數(shù)據(jù)存儲的方法
1.直接存儲為二維矩陣
使用二維矩陣作為電子表格的存儲方法具有簡單直接的優(yōu)點,可以避免頻繁地創(chuàng)建或刪除內存段。然而,需要指出的是,這種方式在存儲值時可能會有一些不太高效的方面,因為它會占用大量的存儲空間來保存沒有實際內容的單元格。
在實際應用中通常使用三元組表示稀疏矩陣:
三元組的表示方法是:對于一個 m×n 的稀疏矩陣 A,我們只存儲矩陣中非零元素的信息,具體來說,將每個非零元素的行下標、列下標和值存儲下來,得到一個三元組(i,j,Ai,j),其中 i 是行下標,j 是列下標,Ai,j 是 A 中對應位置的值。
以前面舉的稀疏矩陣為例,其三元組表示如下:
(1, 4, 6)
(2, 2, 5)
(3, 3, 4)
直接存儲為二維矩陣的復雜度:
- 占用空間:O(N2) 。
- 插入數(shù)據(jù):需要破壞矩陣。
- 刪除數(shù)據(jù):需要破壞矩陣。
- 搜索數(shù)據(jù):O(N2)。
- 訪問數(shù)據(jù):O(1)。
N是假設行和列具有相同長度并形成正方形矩陣的行/列數(shù)。
2.通過鍵值對(Map, Dictionary)優(yōu)化
通過鍵值對(Map, Dictionary)來優(yōu)化,主要是利用哈希表的特性來快速查找元素。具體來說,可以將需要查找的元素作為鍵,將存儲這些元素的數(shù)據(jù)結構作為值,然后將它們存儲在一個哈希表中。這樣,當需要查找某個元素時,只需要使用該元素作為鍵,通過哈希表的查找操作即可快速找到對應的值。
在實際應用中,常見的情況包括:
- 緩存數(shù)據(jù):在需要頻繁訪問數(shù)據(jù)的場景中,通過建立一個緩存,將數(shù)據(jù)存儲在一個鍵值對的數(shù)據(jù)結構中,可以顯著提高數(shù)據(jù)的訪問效率。
- 字符串處理:在需要對字符串進行匹配、查找等操作的場景中,可以將字符串作為鍵,將相應的處理結果作為值,存儲在一個鍵值對的數(shù)據(jù)結構中,可以大幅提高字符串處理的效率。
- 數(shù)據(jù)庫操作:在需要對數(shù)據(jù)庫進行訪問的場景中,可以使用鍵值對數(shù)據(jù)結構來存儲查詢結果,避免重復執(zhí)行查詢操作,減輕數(shù)據(jù)庫的負載。
在下圖中,將單元格位置和對應的單元格值以鍵值對的形式進行了存儲。
通過鍵值對(Map, Dictionary)優(yōu)化稀疏數(shù)組的復雜度:
- 空間:O(N)。
- 插入:O(1)。
- 刪除:O(1)。
- 搜索:O(N)。
- 訪問:O(1)。
N為所記錄的條目數(shù)。
3.通過數(shù)組存儲方式優(yōu)化
在稀疏矩陣中,我們可以使用三個不同的數(shù)組來存儲行索引、列偏移、和其中的值,而不是直接在二維矩陣中存儲值。
存儲的三個數(shù)組:
- 值 =\>單元格中的值。
- 行索引=\>單元格的行索引。
- 列偏移=\>這里每個索引都代表列,并且該數(shù)組將行開始的索引值存儲在 Row 數(shù)組中。
下圖為將稀疏數(shù)組轉化為數(shù)組的形式:
稀疏矩陣具體的插入,刪除,搜索,訪問的代碼:
import java.util.HashMap; import java.util.Map; class SparseMatrix { private int rows; private int cols; private Map<String, Integer> matrix; public SparseMatrix(int rows, int cols) { this.rows = rows; this.cols = cols; this.matrix = new HashMap<>(); } public void insert(int row, int col, int value) { if (row < 0 || row >= rows || col < 0 || col >= cols) { throw new IndexOutOfBoundsException("Invalid matrix index"); } if (value != 0) { String key = row + "," + col; matrix.put(key, value); } } public void delete(int row, int col) { String key = row + "," + col; matrix.remove(key); } public int search(int row, int col) { String key = row + "," + col; return matrix.getOrDefault(key, 0); } public int access(int row, int col) { if (row < 0 || row >= rows || col < 0 || col >= cols) { throw new IndexOutOfBoundsException("Invalid matrix index"); } String key = row + "," + col; return matrix.getOrDefault(key, 0); } }
在上述代碼中,定義了一個 SparseMatrix 類來表示稀疏矩陣。在構造函數(shù)中,我們傳入矩陣的行數(shù)和列數(shù),并創(chuàng)建了一個 HashMap 對象 matrix 來存儲非零元素。insert 方法用于向矩陣中插入元素,如果插入的值不為零,則將其加入 matrix 中,其中鍵為字符串形式的 row,col。delete 方法用于刪除指定位置的元素,通過 remove 方法從 matrix 中移除對應的鍵值對。search 方法用于搜索指定位置的元素,通過調用 getOrDefault 方法從 matrix 中獲取對應的值,如果不存在則返回默認值 0。access 方法用于訪問指定位置的元素,如果超出矩陣邊界則拋出異常,通過調用 getOrDefault 方法從 matrix 中獲取對應的值。
通過稀疏矩陣存儲方式優(yōu)化的復雜度:
- 空間:O(N)。
- 插入:O(N)。
- 刪除:O(N)。
- 搜索:O(N)。
- 訪問:O(1)。
總結
相較于傳統(tǒng)的數(shù)組存儲或鍵值對存儲,稀疏矩陣存儲采用一種基于行索引的數(shù)據(jù)字典存儲方法,這種方法在處理松散布局的表格數(shù)據(jù)時表現(xiàn)出色。與其他存儲方式不同,稀疏矩陣只存儲非空數(shù)據(jù),無需額外開辟內存空間來存儲空數(shù)據(jù)。這種特殊存儲策略使得數(shù)據(jù)片段化變得容易,可以隨時框取整個數(shù)據(jù)層中的一片數(shù)據(jù)進行序列化或反序列化。如果在項目開發(fā)中需要存儲類似結構的數(shù)據(jù),使用稀疏矩陣存儲方式能夠顯著提升性能,無論從時間還是空間上都有很大的優(yōu)勢,葡萄城公司的純前端表格控件——SpreadJS正是借助此功能實現(xiàn)了高性能渲染能力(100 毫秒內加載 10 萬行數(shù)據(jù))。
以上就是解析java稀疏數(shù)組如何幫助我們節(jié)省內存提升性能的詳細內容,更多關于java稀疏數(shù)組提升內存的資料請關注腳本之家其它相關文章!
相關文章
JAVA Netty實現(xiàn)聊天室+私聊功能的示例代碼
這篇文章主要介紹了JAVA Netty實現(xiàn)聊天室+私聊功能的示例代碼,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2020-08-08SpringBoot實現(xiàn)文件斷點續(xù)傳功能詳解
在處理大文件傳輸或網(wǎng)絡不穩(wěn)定的情況下,文件斷點續(xù)傳功能顯得尤為重要,本文將詳細介紹如何使用Spring Boot實現(xiàn)文件的斷點續(xù)傳功能,需要的可以了解下2025-04-04Java簡單實現(xiàn)對一串數(shù)字采用相應的加密策略后傳輸
下面小編就為大家?guī)硪黄狫ava簡單實現(xiàn)對一串數(shù)字采用相應的加密策略后傳輸。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2016-09-09SSM使用mybatis分頁插件pagehepler實現(xiàn)分頁示例
本篇文章主要介紹了SSM使用mybatis分頁插件pagehepler實現(xiàn)分頁示例,使用分頁插件的原因,簡化了sql代碼的寫法,實現(xiàn)較好的物理分頁,非常具有實用價值,需要的朋友可以參考下2018-03-03