NumPy迭代陣列的實現示例

大雄45發表於2023-03-04
導讀 本文主要介紹了NumPy迭代陣列的實現,文中透過示例程式碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
迭代陣列

NumPy中引入了 nditer 物件來提供一種對於陣列元素的訪問方式。

一、單陣列迭代
1. 使用 nditer 訪問陣列的每個元素
>>>a = np.arange(12).reshape(3, 4)
>>>for x in np.nditer(a):
            print(x, end=' ')
0 1 2 3 4 5 6 7 8 9 10 11 
 
# 以上例項不是使用標準 C 或者 Fortran 順序,選擇的順序是和陣列記憶體佈局一致的,
# 這樣做是為了提升訪問的效率,預設是行序優先(row-major order,或者說是 C-order)。
# 這反映了預設情況下只需訪問每個元素,而無需考慮其特定順序。
# 我們可以透過迭代上述陣列的轉置來看到這一點,
# 並與以 C 順序訪問陣列轉置的 copy 方式做對比,如下例項:
>>>for x in np.nditer(a.T):
            print(x, end=' ')
0 1 2 3 4 5 6 7 8 9 10 11 
 
>>>for x in np.nditer(a.T.copy(order='C')):
            print(x, end=' ')
0 4 8 1 5 9 2 6 10 3 7 11
2. 控制陣列元素的迭代順序

使用引數 order 控制元素的訪問順序,引數的可選值有:

  • ‘C’:C order,即是行序優先;
  • ‘F’:Fortran order,即是列序優先;
  • ’K’:參考陣列元素在記憶體中的順序;
  • ‘A’:表示’F’順序;
>>>a = np.arange(12).reshape(3, 4)
>>>for x in np.nditer(a, order='C'):
        print(x, end=' ')
0 1 2 3 4 5 6 7 8 9 10 11 
 
>>>a = np.arange(12).reshape(3, 4)
>>>for x in np.nditer(a, order='F'):
        print(x, end=' ')
0 4 8 1 5 9 2 6 10 3 7 11 
 
>>>a = np.arange(12).reshape(3, 4)
>>>for x in np.nditer(a, order='K'):
        print(x, end=' ')
0 1 2 3 4 5 6 7 8 9 10 11 
 
>>>a = np.arange(12).reshape(3, 4)
>>>for x in np.nditer(a, order='A'):
        print(x, end=' ')
0 1 2 3 4 5 6 7 8 9 10 11
3. 修改陣列值

在使用 nditer 物件迭代陣列時,預設情況下是隻讀狀態。因此,如果需要修改陣列,可以使用引數 op_flags = 'readwrite' or 'writeonly' 來標誌為讀寫或只讀模式。

此時,nditer 在迭代時將生成可寫的緩衝區陣列,可以在此進行修改。為了在修改後,可以將修改的資料回寫到原始位置,需要在迭代結束後,丟擲迭代結束訊號,有兩種方式:

  • 使用 with 上下文管理器;
  • 在迭代結束後,呼叫迭代器的close方法;
  • >>>a = np.arange(12).reshape(3, 4)
    >>>print(a)
    >>>with np.nditer(a, op_flags=['readwrite']) as it:
            for x in it:
                x += 10
    >>>print(a)
    [[ 0  1  2  3]
     [ 4  5  6  7]
     [ 8  9 10 11]]
    [[10 11 12 13]
     [14 15 16 17]
     [18 19 20 21]]
    4. 使用外部迴圈,跟蹤索引或多索引

    以上操作在迭代過程中,都是逐元素進行的,這雖然簡單,但是效率不高。可以使用引數 flags 讓 nditer 迭代時提供更大的塊。並可以透過強制設定 C 和 F 順序,得到不同的塊大小。

    # 預設情況下保持本機的記憶體順序,迭代器提供單一的一維陣列
    # 'external_loop' 給出的值是具有多個值的一維陣列,而不是零維陣列
    >>>a = np.arange(12).reshape(3, 4)
    >>>print(a)
    >>>for x in np.nditer(a, flags=['external_loop']):
            print(x, end=' ')
    [[ 0  1  2  3]
     [ 4  5  6  7]
     [ 8  9 10 11]]
    [ 0  1  2  3  4  5  6  7  8  9 10 11], 
     
    # 設定 'F' 順序
    >>>a = np.arange(12).reshape(3, 4)
    >>>print(a)
    >>>for x in np.nditer(a, flags=['external_loop'], order='F'):
            print(x, end=' ')
    [[ 0  1  2  3]
     [ 4  5  6  7]
     [ 8  9 10 11]]
    [0 4 8], [1 5 9], [ 2  6 10], [ 3  7 11], 
     
    # 'c_index' 可以透過 it.index 跟蹤 'C‘ 順序的索引
    >>>a = np.arange(12).reshape(3, 4)
    >>>print(a)
    >>>it = np.nditer(a, flags=['c_index'])
    >>>for x in it:
                print("{}: ({})".format(x, it.index))
    [[ 0  1  2  3]
     [ 4  5  6  7]
     [ 8  9 10 11]]
    0: (0)
    1: (1)
    2: (2)
    3: (3)
    4: (4)
    5: (5)
    6: (6)
    7: (7)
    8: (8)
    9: (9)
    10: (10)
    11: (11)
     
    # 'f_index' 可以透過 it.index 跟蹤 'F‘ 順序的索引
    >>>a = np.arange(12).reshape(3, 4)
    >>>print(a)
    >>>it = np.nditer(a, flags=['c_index'])
    >>>for x in it:
                print("{}: ({})".format(x, it.index))
    [[ 0  1  2  3]
     [ 4  5  6  7]
     [ 8  9 10 11]]
    0: (0)
    1: (3)
    2: (6)
    3: (9)
    4: (1)
    5: (4)
    6: (7)
    7: (10)
    8: (2)
    9: (5)
    10: (8)
    11: (11)
     
    # 'multi_index' 可以透過 it.multi_index 跟蹤陣列索引
    >>>a = np.arange(12).reshape(3, 4)
    >>>print(a)
    >>>it = np.nditer(a, flags=['multi_index'])
    >>>for x in it:
            print("{}: {}".format(x, it.multi_index))
    [[ 0  1  2  3]
     [ 4  5  6  7]
     [ 8  9 10 11]]
    0: (0, 0)
    1: (0, 1)
    2: (0, 2)
    3: (0, 3)
    4: (1, 0)
    5: (1, 1)
    6: (1, 2)
    7: (1, 3)
    8: (2, 0)
    9: (2, 1)
    10: (2, 2)
    11: (2, 3)
    external_loop 與 multi_index、c_index、c_index不可同時使用,否則將引發錯誤 ValueError: Iterator flag EXTERNAL_LOOP cannot be used if an index or multi-index is being tracked
    5. 以特定資料型別迭代

    當需要以其它的資料型別來迭代陣列時,有兩種方法:

  • 臨時副本:迭代時,會使用新的資料型別建立陣列的副本,然後在副本中完成迭代。但是,這種方法會消耗大量的記憶體空間。
  • 緩衝模式: 使用緩衝來支援靈活輸入,記憶體開銷最小。
  • # 臨時副本
    >>>a = np.arange(12).reshape(3, 4)
    >>>print(a.dtype)
    >>>it = np.nditer(a, op_flags=['readonly', 'copy'],op_dtypes=[np.float64])
    >>>for x in it:
            print("{}".format(x), end=', ')
    int32
    0.0, 1.0, 2.0, 3.0, 4.0, 5.0, 6.0, 7.0, 8.0, 9.0, 10.0, 11.0,
     
    # 緩衝模式
     
    >>>a = np.arange(12).reshape(3, 4)
    >>>print(a.dtype)
    >>>it = np.nditer(a, flags=['buffered'],op_dtypes=[np.float64])
    >>>for x in it:
            print("{}".format(x), end=', ')
    int32
    0.0, 1.0, 2.0, 3.0, 4.0, 5.0, 6.0, 7.0, 8.0, 9.0, 10.0, 11.0, 
    注意
    預設情況下,轉化會執行“安全”機制,如果不符合 NumPy 的轉換規則,會引發異常:TypeError: Iterator operand 0 dtype could not be cast from dtype('float64') to dtype('float32') according to the rule 'safe'
    二、廣播陣列迭代

    如果不同形狀的陣列是可廣播的,那麼 dtype 可以迭代多個陣列。

    >>> a = np.arange(3)
    >>> b = np.arange(6).reshape(2,3)
    >>> for x, y in np.nditer([a,b]):
            print("%d:%d" % (x,y), end=' ')
    0:0 1:1 2:2 0:3 1:4 2:5

    到此這篇關於NumPy迭代陣列的實現的文章就介紹到這了

    原文來自:


    來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/69955379/viewspace-2938000/,如需轉載,請註明出處,否則將追究法律責任。

    相關文章