迭代器(Iterator)与生成器(Generator)的区别

迭代器(Iterator)与生成器(Generator)的区别

迭代器是一个更抽象的概念,任何对象,如果它的类有next方法(next python3)和__iter__方法返回自己本身。

每个生成器都是一个迭代器,但是反过来不行。通常生成器是通过调用一个或多个yield表达式构成的函数s生成的。同时满足迭代器的定义。

当你需要一个类除了有生成器的特性之外还要有一些自定义的方法时,可以使用自定义的迭代器,一般来说生成器更方便,更简单。

def squares(start, stop):
    for i in xrange(start, stop):
        yield i*i

等同于生成器表达式:

(i*i for i in xrange(start, stop))

列表推倒式是:

[i*i for i in xrange(start, stop)]

如果是构建一个自定义的迭代器:

class Squares(object):
    def __init__(self, start, stop):
        self.start = start
        self.stop = stop
    def __iter__(self):
        return self
    def next(self):
        if self.start >= self.stop:
            raise StopIteration
        current = self.start * self.start
        self.start += 1
        return current

此时,你还可以定义自己的方法如:

def current(self):
    return self.start

两者的相同点:对象迭代完后就不能重写迭代了。

Iterables, Iterators, Genrators

热身一下

如果你是来自其它语言比如c,很自然想到的方式是创建一个计数器,然后以自增的方式迭代list。

my_list = [17  23  47  51  101  173  999  1001]

i = 0
while i < len(my_list):
    v = my_list[i]
    print v,
    i += 1

输出:

17 23 47 51 101 173 999 1001

也有可能会借用range,写一个类C语言的风格的for循环:

for i in range(len(my_list)):
    v = my_list[i]
    print v,

输出:

17 23 47 51 101 173 999 1001

上面两种方法都不是Pythonic方式,取而代之的是:

for v in my_list:
    print v,

输出:

17 23 47 51 101 173 999 1001

很多类型的对象都能通过这种方式来迭代,迭代字符串会生成单个字符:

for v in "Hello":
    print v,

输出:

H e l l o

迭代字典,生成字典的key(以无序的方式):

d = {
    'a': 1,
    'b': 2,
    'c': 3,
    }

for v in d:
    print v,
# 注意这里是无序的

输出:

a c b

迭代文件对象,产生字符串行,包括换行符:

f = open("suzuki.txt")
for line in f:
    print ">", line

输出:

> On education

> "Education has failed in a very serious way to convey the most important lesson science can teach: skepticism."

> "An educational system isn't worth a great deal if it teaches young people how to make a living but doesn't teach them how to make a life."

以上可以看出列表、元祖、字符串、字典、文件都可以迭代,能被迭代的对象都称为可迭代对象(Iteratbles),for循环不是唯一接收Iteratbles的东东,还有:

list构造器接收任何类型的Iteratbles,可以使用list()接收字典对象返回只有key的列表:

list(d)

输出:

['a', 'c', 'b']

还可以:

list("Hello")

输出:

['H', 'e', 'l', 'l', 'o']

还可以用在列表推倒式中:

ascii = [ord(x) for x in "Hello"]
ascii

输出:

[72, 101, 108, 108, 111]

sum()函数接收任何数字类型的可迭代对象:

sum(ascii)

输出:

500

str.join()方法接收任何字符类型的可迭代对象 (这里的说法不严谨,总之原则是迭代的元素必须是str类型的):

"-".join(d)

输出:

‘a-c-b'

http://stackoverflow.com/questions/2776829/difference-between-python-generators-vs-iterators
http://excess.org/article/2013/02/itergen1/

  • 2
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
Python中,迭代器iterator)是一个对象,它能够被迭代,即能够被用在循环语句中。迭代器必须实现两个方法:\_\_iter\_\_() 和 \_\_next\_\_()。其中,\_\_iter\_\_() 返回迭代器对象本身,\_\_next\_\_() 返回迭代器的下一个值。如果没有更多的值可供返回,那么就应该抛出 StopIteration 异常。 例如,我们可以使用一个迭代器来遍历一个列表: ```python my_list = [1, 2, 3] my_iterator = iter(my_list) print(next(my_iterator)) # 输出 1 print(next(my_iterator)) # 输出 2 print(next(my_iterator)) # 输出 3 ``` 生成器generator)是一种特殊的迭代器,它是通过函数来实现的。生成器函数使用关键字 yield 来返回一个值,而不是使用 return。当函数被调用时,它并不会立即执行,而是返回一个生成器对象。当调用生成器对象的 \_\_next\_\_() 方法时,函数才会执行,并且执行到 yield 关键字时会返回一个值。 例如,下面的代码是一个简单的生成器函数,它生成斐波那契数列: ```python def fibonacci(): a, b = 0, 1 while True: yield a a, b = b, a + b # 使用生成器函数来生成斐波那契数列 f = fibonacci() for i in range(10): print(next(f)) ``` 输出结果为: ``` 0 1 1 2 3 5 8 13 21 34 ``` 在这个例子中,我们定义了一个无限循环的生成器函数 fibonacci(),它每次返回斐波那契数列中的下一个数。我们使用 for 循环来遍历生成器对象 f,并且调用 next(f) 来获取每个数。由于生成器函数是无限循环的,所以我们需要使用 range(10) 来限制循环次数。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

墨痕_777

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值