怎么实现NumPy迭代数组

其他教程   发布日期:2023年07月11日   浏览次数:419

这篇文章主要介绍“怎么实现NumPy迭代数组”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“怎么实现NumPy迭代数组”文章能帮助大家解决问题。

    迭代数组

    NumPy中引入了 nditer 对象来提供一种对于数组元素的访问方式。

    一、单数组迭代

    1. 使用 nditer 访问数组的每个元素

    1. >>>a = np.arange(12).reshape(3, 4)
    2. >>>for x in np.nditer(a):
    3. print(x, end=' ')
    4. 0 1 2 3 4 5 6 7 8 9 10 11
    5. # 以上实例不是使用标准 C 或者 Fortran 顺序,选择的顺序是和数组内存布局一致的,
    6. # 这样做是为了提升访问的效率,默认是行序优先(row-major order,或者说是 C-order)。
    7. # 这反映了默认情况下只需访问每个元素,而无需考虑其特定顺序。
    8. # 我们可以通过迭代上述数组的转置来看到这一点,
    9. # 并与以 C 顺序访问数组转置的 copy 方式做对比,如下实例:
    10. >>>for x in np.nditer(a.T):
    11. print(x, end=' ')
    12. 0 1 2 3 4 5 6 7 8 9 10 11
    13. >>>for x in np.nditer(a.T.copy(order='C')):
    14. print(x, end=' ')
    15. 0 4 8 1 5 9 2 6 10 3 7 11

    2. 控制数组元素的迭代顺序

    使用参数 order 控制元素的访问顺序,参数的可选值有:

    • ‘C’:C order,即是行序优先;

    • ‘F’:Fortran order,即是列序优先;

    • ’K’:参考数组元素在内存中的顺序;

    • ‘A’:表示’F’顺序;

    1. >>>a = np.arange(12).reshape(3, 4)
    2. >>>for x in np.nditer(a, order='C'):
    3. print(x, end=' ')
    4. 0 1 2 3 4 5 6 7 8 9 10 11
    5. >>>a = np.arange(12).reshape(3, 4)
    6. >>>for x in np.nditer(a, order='F'):
    7. print(x, end=' ')
    8. 0 4 8 1 5 9 2 6 10 3 7 11
    9. >>>a = np.arange(12).reshape(3, 4)
    10. >>>for x in np.nditer(a, order='K'):
    11. print(x, end=' ')
    12. 0 1 2 3 4 5 6 7 8 9 10 11
    13. >>>a = np.arange(12).reshape(3, 4)
    14. >>>for x in np.nditer(a, order='A'):
    15. print(x, end=' ')
    16. 0 1 2 3 4 5 6 7 8 9 10 11

    3. 修改数组值

    在使用 nditer 对象迭代数组时,默认情况下是只读状态。因此,如果需要修改数组,可以使用参数 op_flags = 'readwrite' or 'writeonly' 来标志为读写或只读模式。

    此时,nditer 在迭代时将生成可写的缓冲区数组,可以在此进行修改。为了在修改后,可以将修改的数据回写到原始位置,需要在迭代结束后,抛出迭代结束信号,有两种方式:

    • 使用 with 上下文管理器;

    • 在迭代结束后,调用迭代器的close方法;

    1. >>>a = np.arange(12).reshape(3, 4)
    2. >>>print(a)
    3. >>>with np.nditer(a, op_flags=['readwrite']) as it:
    4. for x in it:
    5. x += 10
    6. >>>print(a)
    7. [[ 0 1 2 3]
    8. [ 4 5 6 7]
    9. [ 8 9 10 11]]
    10. [[10 11 12 13]
    11. [14 15 16 17]
    12. [18 19 20 21]]

    4. 使用外部循环,跟踪索引或多索引

    以上操作在迭代过程中,都是逐元素进行的,这虽然简单,但是效率不高。可以使用参数 flags 让 nditer 迭代时提供更大的块。并可以通过强制设定 C 和 F 顺序,得到不同的块大小。

    1. # 默认情况下保持本机的内存顺序,迭代器提供单一的一维数组
    2. # 'external_loop' 给出的值是具有多个值的一维数组,而不是零维数组
    3. >>>a = np.arange(12).reshape(3, 4)
    4. >>>print(a)
    5. >>>for x in np.nditer(a, flags=['external_loop']):
    6. print(x, end=' ')
    7. [[ 0 1 2 3]
    8. [ 4 5 6 7]
    9. [ 8 9 10 11]]
    10. [ 0 1 2 3 4 5 6 7 8 9 10 11],
    11. # 设定 'F' 顺序
    12. >>>a = np.arange(12).reshape(3, 4)
    13. >>>print(a)
    14. >>>for x in np.nditer(a, flags=['external_loop'], order='F'):
    15. print(x, end=' ')
    16. [[ 0 1 2 3]
    17. [ 4 5 6 7]
    18. [ 8 9 10 11]]
    19. [0 4 8], [1 5 9], [ 2 6 10], [ 3 7 11],
    20. # 'c_index' 可以通过 it.index 跟踪 'C‘ 顺序的索引
    21. >>>a = np.arange(12).reshape(3, 4)
    22. >>>print(a)
    23. >>>it = np.nditer(a, flags=['c_index'])
    24. >>>for x in it:
    25. print("{}: ({})".format(x, it.index))
    26. [[ 0 1 2 3]
    27. [ 4 5 6 7]
    28. [ 8 9 10 11]]
    29. 0: (0)
    30. 1: (1)
    31. 2: (2)
    32. 3: (3)
    33. 4: (4)
    34. 5: (5)
    35. 6: (6)
    36. 7: (7)
    37. 8: (8)
    38. 9: (9)
    39. 10: (10)
    40. 11: (11)
    41. # 'f_index' 可以通过 it.index 跟踪 'F‘ 顺序的索引
    42. >>>a = np.arange(12).reshape(3, 4)
    43. >>>print(a)
    44. >>>it = np.nditer(a, flags=['c_index'])
    45. >>>for x in it:
    46. print("{}: ({})".format(x, it.index))
    47. [[ 0 1 2 3]
    48. [ 4 5 6 7]
    49. [ 8 9 10 11]]
    50. 0: (0)
    51. 1: (3)
    52. 2: (6)
    53. 3: (9)
    54. 4: (1)
    55. 5: (4)
    56. 6: (7)
    57. 7: (10)
    58. 8: (2)
    59. 9: (5)
    60. 10: (8)
    61. 11: (11)
    62. # 'multi_index' 可以通过 it.multi_index 跟踪数组索引
    63. >>>a = np.arange(12).reshape(3, 4)
    64. >>>print(a)
    65. >>>it = np.nditer(a, flags=['multi_index'])
    66. >>>for x in it:
    67. print("{}: {}".format(x, it.multi_index))
    68. [[ 0 1 2 3]
    69. [ 4 5 6 7]
    70. [ 8 9 10 11]]
    71. 0: (0, 0)
    72. 1: (0, 1)
    73. 2: (0, 2)
    74. 3: (0, 3)
    75. 4: (1, 0)
    76. 5: (1, 1)
    77. 6: (1, 2)
    78. 7: (1, 3)
    79. 8: (2, 0)
    80. 9: (2, 1)
    81. 10: (2, 2)
    82. 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. 以特定数据类型迭代

    当需要以其它的数据类型来迭代数组时,有两种方法:

    • 临时副本:迭代时,会使用新的数据类型创建数组的副本,然后在副本中完成迭代。但是,这种方法会消耗大量的内存空间。

    • 缓冲模式: 使用缓冲来支持灵活输入,内存开销最小。

    1. # 临时副本
    2. >>>a = np.arange(12).reshape(3, 4)
    3. >>>print(a.dtype)
    4. >>>it = np.nditer(a, op_flags=['readonly', 'copy'],op_dtypes=[np.float64])
    5. >>>for x in it:
    6. print("{}".format(x), end=', ')
    7. int32
    8. 0.0, 1.0, 2.0, 3.0, 4.0, 5.0, 6.0, 7.0, 8.0, 9.0, 10.0, 11.0,
    9. # 缓冲模式
    10. >>>a = np.arange(12).reshape(3, 4)
    11. >>>print(a.dtype)
    12. >>>it = np.nditer(a, flags=['buffered'],op_dtypes=[np.float64])
    13. >>>for x in it:
    14. print("{}".format(x), end=', ')
    15. int32
    16. 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 可以迭代多个数组。

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

    以上就是怎么实现NumPy迭代数组的详细内容,更多关于怎么实现NumPy迭代数组的资料请关注九品源码其它相关文章!