python 中迭代多个序列

来源:互联网 发布:源码本地放到php打不开 编辑:程序博客网 时间:2024/04/30 03:46

一、多个序列迭代

有时候我们希望能够同时遍历多个序列,比如有序列a = [1, 2, 3, 4, 5], b = ['a', 'b', 'c', 'd', 'e'],我们如果要同时遍历的话,可以采用如下的方式:

  1. >>> xpts = [1542107]  
  2. >>> ypts = [1017837156299]  
  3. >>> for x, y in zip(xpts, ypts):  
  4. ...     print(x, y)  
  5. ...   
  6. 1 101  
  7. 5 78  
  8. 4 37  
  9. 2 15  
  10. 10 62  
  11. 7 99 
  因为使用了zip()方法,我们将两个集合里的内容都同时取出来,按照tuple的方式一个个的组织起来。所以我们访问的时候也是通过一个个tuple的方式来读取。这里我们提供的两个list是长度一致的,如果不一致会怎么样呢?我们再来试试另外两个序列:

  1. >>> a = [1234]  
  2. >>> b = ['a''b''c']  
  3. >>> for x, y in zip(a, b):  
  4. ...     print(x, y)         
  5. ...   
  6. 1 a  
  7. 2 b  
  8. 3 c 
从代码运行的结果来看,默认是遍历到短的那个序列结束。如果我们需要到那个长的序列结束呢?这里有另外一种办法:

 

 

Python代码  收藏代码
  1. >>> from itertools import zip_longest  
  2. >>> for i in zip_longest(a, b):  
  3. ...     print(i)  
  4. ...   
  5. (1'a')  
  6. (2'b')  
  7. (3'c')  
  8. (4None)  
    这里引用了zip_longest方法,它可以将两个序列组合起来,不过对于短的那个序列,用None来补齐。

将几个序列串在一起

    我们可以直接看如下的代码:

 

Python代码  收藏代码
  1. >>> from itertools import chain  
  2. >>> a = [1234]  
  3. >>> b = ['a''b''c']  
  4. >>> for x in chain(a, b):  
  5. ...     print(x)  
  6. ...   
  7. 1  
  8. 2  
  9. 3  
  10. 4  
  11. a  
  12. b  
  13. c  
    一个chain方法就解决了大部分问题了。和我们默认想到的方法比起来,chain方法效率更加高。因为我们最开始会考虑将两个或者多个序列连在一起,比如a + b,这样会创造一个新的序列出来,这样带来的成本开销明显偏大了。

 

 

将嵌套的序列变平

    这是一个有意思的问题,因为一般来说当我们需要访问一个数组的时候,比如说a = [1, 2, [3, 4, [5, 6], 7, 8], 9, 10],我们希望能够将他们所有的元素都输出,并使得他们看起来像就是一个一维数组那样,如a = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]。我们默认的思路该怎么办呢?只怕一开始就是碰到一个元素的时候会判断它是否为数组,如果是的则递归的去输出它的元素。

    Python里面有一个很强大的特性可以很好的实现这个方法:

 

Python代码  收藏代码
  1. from collections import Iterable  
  2.   
  3. def flatten(items, ignore_types=(str, bytes)):  
  4.     for x in items:  
  5.         if isinstance(x, Iterable) and not isinstance(x, ignore_types):  
  6.             yield from flatten(x)  
  7.         else:  
  8.             yield x  
    这种实现里面有一个额外的ignore_types,里面列举了一些类型我们可以不需要进一步的去遍历。比如说str,我们一般碰到一个字符串可以直接将他们作为一个整的对象输出而不是再对它们进一步拆分的遍历。最有意思的地方在yield from这个部分。yield from这个部分的意思是将后续的值作为它本身的一个subroutine。所以它们就会被当作一个拉平的数组。关于yield from这部分我们在后面的文章中会专门讲述。

 

    按照这个方式,我们使用它们的代码如下:

 

 

Python代码  收藏代码
  1. >>> from nested import flatten  
  2. >>> items = [12, [34, [56], 7], 8]  
  3. >>> for x in flatten(items):  
  4. ...     print(x)  
  5. ...   
  6. 1  
  7. 2  
  8. 3  
  9. 4  
  10. 5  
  11. 6  
  12. 7  
  13. 8  

 

 

迭代多个有序排列数组

    这个问题不太好用一句话描述,就是说假定我们有若干个已经排序的数组了。当我们希望能够去遍历这所有的序列,但是保证我们每次都取出他们中间最小的元素,保证所有输出还是一个严格排序的结果,我们该怎么办呢?实际上,这是一个多路归并排序的问题。在前面的一些文章里有过讨论,不过要做一个好的java实现我们可是费了一番功夫。这里有什么好的招呢?

Python代码  收藏代码
  1. >>> import heapq  
  2. >>> a = [14710]  
  3. >>> b = [25611]  
  4. >>> for c in heapq.merge(a, b):  
  5. ...     print(c)  
  6. ...   
  7. 1  
  8. 2  
  9. 4  
  10. 5  
  11. 6  
  12. 7  
  13. 10  
  14. 11  

    这里是归并两路的数据结果。在一些我们如果要归并多个文件的情况下,也可以这样来做。因为这里heapq.merge不是一次将所有的数据都装载到内存里,它只是每次取很小的一部分,像generator一样。所以对于大文件的合并用这种方式来做。呵呵,寥寥几行代码就解决了问题,不能不说,很好很强大啊。 

 

 

总结

    Iterator的定义方法虽然看起来很简单,但是它的使用也可以非常的复杂和灵活。通过结合一些库的支持,我们可以实现非常强大的计算效果。当然,前提是我们需要知道去哪里找到这些库和知道这些用法。


1 0