8、numpy——數組的迭代


1、單數組的迭代

NumPy 迭代器對象 numpy.nditer 提供了一種靈活訪問一個或者多個數組元素的方式。

迭代器最基本的任務的可以完成對數組元素的訪問。

1.1 默認迭代順序

1 import numpy as np
2 a = np.arange(6).reshape(2, 3)
3 b = a.T  # b為a的裝置
4 print('a=', a)
5 for i in np.nditer(a):
6     print(i)
7 print('b=', b)
8 for i in np.nditer(b):
9     print(i)

 

 執行結果:

a= [[0 1 2]
 [3 4 5]]
0
1
2
3
4
5
b= [[0 3]
 [1 4]
 [2 5]]
0
1
2
3
4
5

 

注意:通過該種方式迭代輸出的是以元素在存儲器中的布局順序輸出的,無論其視圖做何種改變(轉置,變換shape),其輸出結果是一致的,該中方式可以提高迭代效率

a 和 a.T 的遍歷順序是一樣的,也就是他們在內存中的存儲順序也是一樣的,但是 a.T.copy(order = 'C') 的遍歷結果是不同的,那是因為它和前兩種的存儲方式是不一樣的,默認是按行訪問。

1.2 控制迭代的順序

  • for x in np.nditer(a, order='F'):Fortran order,即是列序優先;

  • for x in np.nditer(a.T, order='C'):C order,即是行序優先;

 1 import numpy as np
 2  
 3 a = np.arange(0,60,5) 
 4 a = a.reshape(3,4)  
 5 print ('原始數組是:') 
 6 print (a) 
 7 print ('\n') 
 8 print ('原始數組的轉置是:') 
 9 b = a.T 
10 print (b) 
11 print ('\n') 
12 print ('以 C 風格順序排序:') 
13 c = b.copy(order='C')  
14 print (c)
15 for x in np.nditer(c):  
16     print (x, end=", " )
17 print  ('\n') 
18 print  ('以 F 風格順序排序:')
19 c = b.copy(order='F')  
20 print (c)
21 for x in np.nditer(c):  
22     print (x, end=", " )

 

執行結果:

原始數組是:
[[ 0  5 10 15]
 [20 25 30 35]
 [40 45 50 55]]


原始數組的轉置是:
[[ 0 20 40]
 [ 5 25 45]
 [10 30 50]
 [15 35 55]]


以 C 風格順序排序:
[[ 0 20 40]
 [ 5 25 45]
 [10 30 50]
 [15 35 55]]
0, 20, 40, 5, 25, 45, 10, 30, 50, 15, 35, 55, 

以 F 風格順序排序:
[[ 0 20 40]
 [ 5 25 45]
 [10 30 50]
 [15 35 55]]
0, 5, 10, 15, 20, 25, 30, 35, 40, 45, 50, 55,

 

 

可以通過顯式設置,來強制 nditer 對象使用某種順序:

 1 import numpy as np 
 2  
 3 a = np.arange(0,60,5) 
 4 a = a.reshape(3,4)  
 5 print ('原始數組是:')
 6 print (a)
 7 print ('\n')
 8 print ('以 C 風格順序排序:')
 9 for x in np.nditer(a, order =  'C'):  
10     print (x, end=", " )
11 print ('\n')
12 print ('以 F 風格順序排序:')
13 for x in np.nditer(a, order =  'F'):  
14     print (x, end=", " )

 

執行結果:

 1 原始數組是:
 2 [[ 0  5 10 15]
 3  [20 25 30 35]
 4  [40 45 50 55]]
 5 
 6 
 7 以 C 風格順序排序:
 8 0, 5, 10, 15, 20, 25, 30, 35, 40, 45, 50, 55, 
 9 
10 以 F 風格順序排序:
11 0, 20, 40, 5, 25, 45, 10, 30, 50, 15, 35, 5

 

1.3 修改數組中的元素

nditer 對象有另一個可選參數 op_flags。 默認情況下,nditer 將視待迭代遍歷的數組為只讀對象(read-only),為了在遍歷數組的同時,實現對數組元素值得修改,必須指定 read-write 或者 write-only 的模式。

 1 import numpy as np
 2  
 3 a = np.arange(0,60,5) 
 4 a = a.reshape(3,4)  
 5 print ('原始數組是:')
 6 print (a)
 7 print ('\n')
 8 for x in np.nditer(a, op_flags=['readwrite']): 
 9     x[...]=2*x 
10 print ('修改后的數組是:')
11 print (a)

 

執行結果:

原始數組是:
[[ 0  5 10 15]
 [20 25 30 35]
 [40 45 50 55]]


修改后的數組是:
[[  0  10  20  30]
 [ 40  50  60  70]
 [ 80  90 100 110]]

 

2、數組迭代(Iterating over arrays)

該部分內容位於numpy-ref-1.14.5中的3.15.4 Iterating over arrays 章節

numpy.nditer 為高效多維迭代器對象,用於對數組的迭代。

flags:sequence of str ,optional

用於控制迭代器行為的標志(flags)

buffrered - 再需要時可以緩沖

c_index - 跟蹤C順序的索引

f_index - 跟蹤F順序的索引

multi_index - 跟蹤 多指標,或每個迭代維度的一組指數

external_loop - 外部循環,將一維的最內層的循環轉移到外部循環迭代器,使得numpy的矢量操作在處理更大規模數據時變得更有效率。

 

op_flags:list of list of str , optional

這是每個操作數的標志列表。至少必須指定一個“只讀”、“讀寫”或“寫”。

readonly - 該操作數表明可以讀取

readwrite - 該操作數表明可以讀取和寫入

writeonly - 該操作數表明僅寫入

no_broadcast - 該操作數可以防止被廣播

copy - 該操作數表明允許臨時只讀拷貝

 

op_dtypes:dtype or tuple of dtype(s), optional

操作數所需的數據類型(s)。

 

order:{‘C’, ‘F’, ‘A’, ‘K’}, optional

控制迭代順序(Controls the iteration order)

 2.1 使用外部循環 external_loop

將一維的最內層的循環轉移到外部循環迭代器,使得numpy的矢量化操作在處理更大規模數據時變得更有效率。

 1 a = np.arange(6).reshape(2,3)
 2 print(a)
 3 # [[0 1 2]
 4 #  [3 4 5]]
 5 for x in np.nditer(a, flags = ['external_loop']):
 6     print(x)
 7 # [0 1 2 3 4 5]
 8 
 9 for x in np.nditer(a, flags = ['external_loop'],order = 'C'):
10     print(x)
11 # [0 1 2 3 4 5]
12 
13 for x in np.nditer(a, flags = ['external_loop'],order = 'F'):
14     print(x)
15 # [0 3]
16 # [1 4]
17 # [2 5]

2.2 追蹤單個索引或多重索引

 1 a = np.arange(6).reshape(2,3)
 2 print(a)
 3 # [[0 1 2]
 4 #  [3 4 5]]
 5 it = np.nditer(a,flags = ['f_index'])
 6 
 7 while not it.finished:
 8     print("%d <%d>" % (it[0], it.index))
 9     it.iternext()
10 
11 # 0 <0>
12 # 1 <2>
13 # 2 <4>
14 # 3 <1>
15 # 4 <3>
16 # 5 <5>

為了更清楚地表述,可以直觀地看下表

flags = multi_index

 1 a = np.arange(6).reshape(2,3)
 2 print(a)
 3 # [[0 1 2]
 4 #  [3 4 5]]
 5 it = np.nditer(a,flags = ['multi_index'])
 6 
 7 while not it.finished:
 8     print("%d <%s>" % (it[0], it.multi_index))
 9     it.iternext()
10 # 0 <(0, 0)>
11 # 1 <(0, 1)>
12 # 2 <(0, 2)>
13 # 3 <(1, 0)>
14 # 4 <(1, 1)>
15 # 5 <(1, 2)>

multi_index是將元素的行列位置以元組方式打印出來,但元組形式不是整型,所以要將 %d 變為 %s,

若不改,則會報錯 TypeError: %d format: a number is required, not tuple 

 1 import numpy as np
 2 a = np.arange(6).reshape(1,2,3)
 3 print(a)
 4 # [[[0 1 2]
 5 #   [3 4 5]]]
 6 it = np.nditer(a,flags = ['multi_index'])
 7 
 8 while not it.finished:
 9     print("%d <%s>" % (it[0], it.multi_index))
10     it.iternext()
11 # 0 <(0, 0, 0)>
12 # 1 <(0, 0, 1)>
13 # 2 <(0, 0, 2)>
14 # 3 <(0, 1, 0)>
15 # 4 <(0, 1, 1)>
16 # 5 <(0, 1, 2)>

2.3 廣播迭代

如果兩個數組是 可廣播的,nditer組合對象能夠同時迭代它們,假設數組 a 具有維度 3*4 ,並且存在維度為 1*4的另一個數組b,則使用以下類型的迭代器(數組b被廣播到a的大小)

 

 1 import numpy as np 
 2  
 3 a = np.arange(0,60,5) 
 4 a = a.reshape(3,4)  
 5 print  ('第一個數組為:')
 6 print (a)
 7 print  ('\n')
 8 print ('第二個數組為:')
 9 b = np.array([1,  2,  3,  4], dtype =  int)  
10 print (b)
11 print ('\n')
12 print ('修改后的數組為:')
13 for x,y in np.nditer([a,b]):  
14     print ("%d:%d"  %  (x,y), end=", " )

執行結果:

 1 第一個數組為:
 2 [[ 0  5 10 15]
 3  [20 25 30 35]
 4  [40 45 50 55]]
 5 
 6 
 7 第二個數組為:
 8 [1 2 3 4]
 9 
10 
11 修改后的數組為:
12 0:1, 5:2, 10:3, 15:4, 20:1, 25:2, 30:3, 35:4, 40:1, 45:2, 50:3, 55:4,

 

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM