網站首頁 編程語言 正文
1. 前言
數組和矩陣是數值計算的基礎元素。目前為止,我們都是使用NumPy的ndarray數據結構來表示數組,這是一種同構的容器,用于存儲數組的所有元素。
有一種特殊情況,矩陣的大部分元素都為零,這種矩陣被稱為稀疏矩陣。對于稀疏矩陣,將所有零保存在計算機內存中的效率很低,更合適的方法是只保存非零元素以及位置信息。于是SciPy應運而生,為稀疏矩陣的表示及其線性代數運算提供了豐富易用的接口。
2. 導入包
SciPy中提供了稀疏矩陣模塊scipy.sparse,為稀疏矩陣的表示及其線性代數運算提供了豐富易用的接口。
import scipy.sparse as sp
import scipy.sparse.linalg
import scipy.linalg as la
3. 稀疏矩陣總覽
There are seven available sparse matrix types:
1. csc_matrix: Compressed Sparse Column format
2. csr_matrix: Compressed Sparse Row format
3. bsr_matrix: Block Sparse Row format
4. lil_matrix: List of Lists format
5. dok_matrix: Dictionary of Keys format
6. coo_matrix: COOrdinate format (aka IJV, triplet format)
7. dia_matrix: DIAgonal format
- sp.coo_matrix(坐標的列表)描述:將非零值及其行列信息保存在一個列表。構造簡單,添加元素方便,但訪問元素效率低下;
- sp.lil_matrix(列表的列表):將每行的非零元素列索引保存在一個列表,將對應值保存在另一個列表。支持切片操作,但不方便進行數學運算;
- sp.dok_matrix(值的字典):將非零值保存在字典,非零值的坐標元組作為字典的鍵。構造簡單,可快速添加刪除元素,但不方便進行數學運算;
- sp.dia_matrix(對角矩陣):矩陣的對角線列表 。對于對角矩陣非常有效 ,但不適用非對角矩陣;
- sp.csc_matrix 和 sp.csr_matrix(壓縮列格式和壓縮行格式):將值與行列索引的數組一起存儲。對于矩陣的向量乘法很高效,但構造相對復雜;
- sp.bsr_matrix(塊稀疏矩陣):與CSR類似,用于具有稠密子矩陣的稀疏矩陣。對于此類特殊矩陣很高效,但不適用一般矩陣。
注意:
- 為了有效地構建矩陣,使用dok_matrix或者lil_matrix,lil_matrix類支持基本的切片和索引操作,語法與NumPy的arrays相似。
- COO格式也能有效率地構建矩陣。盡管與NumPy有許多相似性,但是強烈不建議使用NumPy的函數直接對稀疏矩陣格式進行操作,因為可能導致不正確的結果。如果將NumPy的函數用在這些矩陣上,首先檢查SciPy在對應的稀疏矩陣類上有沒有已經實現的操作,或者使用toarray()方法將稀疏矩陣對象轉為NumPy的array。
- 實現乘法與轉置操作,則轉為CSC或CSR格式,lil_matrix格式是基于行的,所以轉為為CSR比CSC更有效率。所有的轉換在CSR,CSC和COO格式之間都是有效的,線性時間操作。
4. 稀疏矩陣詳細介紹
4.1 coo_matrix
coo_matrix是最簡單的存儲方式。采用三個數組row、col和data保存非零元素的行下標,列下標與值。這三個數組的長度相同一般來說,coo_matrix主要用來創建矩陣,因為coo_matrix無法對矩陣的元素進行增刪改等操作,一旦創建之后,除了將之轉換成其它格式的矩陣,幾乎無法對其做任何操作和矩陣運算。
為了創建sp.coo_matrix對象,需要創建非零值、行索引以及列索引的列表或數組,并將其傳遞給生成函數sp.coo_matrix。
values = [1, 2, 3, 4]
rows = [0, 1, 2, 3]
cols = [1, 3, 2, 0]
A = sp.coo_matrix((values, (rows, cols)), shape=[4, 4])
A
>>> A.toarray()
array([[1, 0, 0, 0],
[0, 0, 0, 2],
[0, 0, 3, 0],
[4, 0, 0, 0]])
>>> type(A)
<class 'scipy.sparse.coo.coo_matrix'>
>>> type(A.toarray())
<class 'numpy.ndarray'>
SciPy的sparse模塊中稀疏矩陣的屬性大部分派生自NumPy的ndarray對象,同時也包括nnz(非零元素數目)和data(非零值)等屬性。
A.shape, A.size, A.dtype, A.ndim
A.nnz, A.data
對于sp.coo_matrix對象,還可以使用row和col屬性來訪問底層的行列坐標數組。
A.row, A.col
4.2 dok_matrix
dok_matrix適用的場景是逐漸添加矩陣的元素。dok_matrix的策略是采用字典來記錄矩陣中不為0的元素。所以字典的key存的是記錄元素的位置信息的元祖,value是記錄元素的具體值。
>>> S = sparse.dok_matrix((5, 5), dtype=np.float32)
>>> for i in range(5):
for j in range(5):
S[i,j] = i+j # 更新元素
>>> S.toarray()
[[0. 1. 2. 3. 4.]
[1. 2. 3. 4. 5.]
[2. 3. 4. 5. 6.]
[3. 4. 5. 6. 7.]
[4. 5. 6. 7. 8.]]
優點:對于遞增的構建稀疏矩陣很高效,比如定義該矩陣后,想進行每行每列更新值,可用該矩陣。當訪問某個單元,只需要O(1)
缺點:不允許重復索引(coo中適用),但可以很高效的轉換成coo后進行重復索引。
4.3 lil_matrix
lil_matrix適用的場景也是逐漸添加矩陣的元素。與dok不同,lil_matrix則是使用兩個列表存儲非0元素。data保存每行中的非零元素,rows保存非零元素所在的列。這種格式也很適合逐個添加元素,并且能快速獲取行相關的數據。
>>> l = sparse.lil_matrix((4, 4))
>>> l[1, 1] = 1
>>> l[1, 3] =2
>>> l[2, 3] = 3
>>> l.toarray()
array([[0., 0., 0., 0.],
[0., 1., 0., 2.],
[0., 0., 0., 3.],
[0., 0., 0., 0.]])
>>> l.data
array([list([]), list([1.0, 2.0]), list([3.0]), list([])], dtype=object)
>>> l.rows
array([list([]), list([1, 3]), list([3]), list([])], dtype=object)
優點:適合遞增的構建成矩陣、轉換成其它存儲方式很高效、支持靈活的切片。
缺點:當矩陣很大時,考慮用coo、算術操作,列切片,矩陣向量內積操作慢。
4.4 dia_matrix
如果稀疏矩陣僅包含非0元素的對角線,則對角存儲格式(DIA)可以減少非0元素定位的信息量。這種存儲格式對有限元素或者有限差分離散化的矩陣尤其有效。dia_matrix通過兩個數組確定: data和offsets。其中data對角線元素的值;offsets:第i個offsets是當前第i個對角線和主對角線的距離。data[k:]存儲了offsets[k]對應的對角線的全部元素。例子如下:
>>> data = np.array([[1, 2, 3, 4], [5, 6, 0, 0], [0, 7, 8, 9]])
>>> offsets = np.array([0, -2, 1])
>>> sparse.dia_matrix((data, offsets), shape=(4, 4)).toarray()
array([[1, 7, 0, 0],
[0, 2, 8, 0],
[5, 0, 3, 9],
[0, 6, 0, 4]])
注意:offsets[0]=0表示第0個對角線與主對角線的距離為0,表示第0個對角線就是主對角線,data[0]就是第0個對角線的值。offsets[1]=-2表示第1個對角線與主對角線距離為-2,此時該對角線在主對角線的左下方,對角線上數值的數量為4-2=2,對應的值為data[1, :2+1],此時data[1, 3:]為無效的值,在構造對角稀疏矩陣時不起作用。offsets[2]=1表示第2個對角線與主對角線距離為1,此時該對角線在主對角線的右上方,對角線上數值的數量為4-1=3,對應的值為data[2, 1:],此時data[2, :1]為無效的值,在構造對角稀疏矩陣時不起作用。
4.5 csc_matrix & csr_matrix
csr_matrix是按行對矩陣進行壓縮的,csc_matrix則是按列對矩陣進行壓縮的。通過row_offsets,column_indices,data來確定矩陣。column_indices,data與coo格式的列索引與數值的含義完全相同,row_offsets表示元素的行偏移量。例子如下,
>>> indptr = np.array([0, 2, 3, 6])
>>> indices = np.array([0, 2, 2, 0, 1, 2])
>>> data = np.array([1, 2, 3, 4, 5, 6])
>>> csr_matrix((data, indices, indptr), shape=(3, 3)).toarray()
array([[1, 0, 2],
[0, 0, 3],
[4, 5, 6]])
注意:indices和data分別表示列索引和數據,第 i 行的列索引存儲在indices[indptr[i]:indptr[i+1]] 中,對應的值為data[indptr[i]:indptr[i+1]]。即第0行的列索引為indices[0:2]=[0,2],值為data[0:2]=[1,2];第1行的列索引為indices[2:3]=[2],值為data[2:3]=[3]…
CSR格式常用于讀入數據后進行稀疏矩陣計算。
兩者的優缺點互反:
- CSR優點:高效的稀疏矩陣算術操作、高效的行切片、快速地矩陣向量內積操作;
- CSR缺點:緩慢地列切片操作(可以考慮csc)、轉換到稀疏結構代價較高(可以考慮lil,dok)。
- CSC優點:高效的稀疏矩陣算術操作、高效的列切片、快速地矩陣向量內積操作(不如csr,bsr塊);
- CSC缺點:緩慢地行切片操作(可以考慮csr)、 轉換到稀疏結構代價較高(可以考慮lil,dok)。
4.6 bsr_matrix
基于行的塊壓縮,通過row_offsets,column_indices,data來確定矩陣。與csr相比,只是data中的元數據由0維的數變為了一個矩陣(塊),其余完全相同。
>>> indptr = np.array([0,2,3,6])
>>> indices = np.array([0,2,2,0,1,2])
>>> data = np.array([1,2,3,4,5,6]).repeat(4).reshape(6,2,2)
>>> bsr_matrix((data,indices,indptr), shape=(6,6)).todense()
matrix([[1, 1, 0, 0, 2, 2],
[1, 1, 0, 0, 2, 2],
[0, 0, 0, 0, 3, 3],
[0, 0, 0, 0, 3, 3],
[4, 4, 5, 5, 6, 6],
[4, 4, 5, 5, 6, 6]])
優點:很類似于csr,更適合于矩陣的某些子矩陣很多值,在某些情況下比csr和csc計算更高效。
5. 稀疏矩陣的存取
5.1 用save_npz保存單個稀疏矩陣
>>> scipy.sparse.save_npz('sparse_matrix.npz', sparse_matrix)
>>> sparse_matrix = scipy.sparse.load_npz('sparse_matrix.npz')
稀疏矩陣存儲大小比較:
a = np.arange(100000).reshape(1000,100)
a[10: 300] = 0
b = sparse.csr_matrix(a)
# 稀疏矩陣壓縮存儲到npz文件
sparse.save_npz('b_compressed.npz', b, True) # 文件大小:100KB
# 稀疏矩陣不壓縮存儲到npz文件
sparse.save_npz('b_uncompressed.npz', b, False) # 文件大小:560KB
# 存儲到普通的npy文件
np.save('a.npy', a) # 文件大?。?91KB
# 存儲到壓縮的npz文件
np.savez_compressed('a_compressed.npz', a=a) # 文件大?。?7KB
對于存儲到npz文件中的CSR格式的稀疏矩陣,內容為:
data.npy
format.npy
indices.npy
indptr.npy
shape.npy
6. 總結
加載數據文件時使用coo_matrix快速構建稀疏矩陣,然后調用to_csr()、to_csc()、to_dense()把它轉換成CSR或稠密矩陣(numpy.matrix)。
coo_matrix格式常用于從文件中進行稀疏矩陣的讀寫,而csr_matrix格式常用于讀入數據后進行稀疏矩陣計算。
7. 參考
【1】https://blog.csdn.net/winycg/article/details/80967112
【2】https://blog.csdn.net/vor234/article/details/124935384
原文鏈接:https://blog.csdn.net/qq_51392112/article/details/129014167
相關推薦
- 2022-06-19 selenium?IDE自動化測試腳本的實現_其它綜合
- 2022-03-14 Failed to load ApplicationContext異常的解決思路
- 2023-12-09 SpringBoot自定義異常處理機制
- 2023-04-02 linux?top命令基本實戰_linux shell
- 2023-03-05 Suspend函數與回調的互相轉換示例詳解_Android
- 2022-04-09 Nginx1.21.6生產環境升級步驟_nginx
- 2022-07-29 PyTorch實現手寫數字的識別入門小白教程_python
- 2022-12-15 redis分布式鎖與zk分布式鎖的對比分析_Redis
- 最近更新
-
- window11 系統安裝 yarn
- 超詳細win安裝深度學習環境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優雅實現加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發現-Nac
- Spring Security之基于HttpR
- Redis 底層數據結構-簡單動態字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支