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

发布于 2021-04-12 12:17:16 字数 3053 浏览 1202 评论 0

迭代器是一个更抽象的概念,任何对象,如果它的类有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

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


如果你是来自其它语言比如 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'

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据

关于作者

JSmiles

生命进入颠沛而奔忙的本质状态,并将以不断告别和相遇的陈旧方式继续下去。

0 文章
0 评论
84961 人气
更多

推荐作者

daid

文章 0 评论 0

我心依旧

文章 0 评论 0

晒暮凉

文章 0 评论 0

微信用户

文章 0 评论 0

DS

文章 0 评论 0

〆凄凉。

文章 0 评论 0

    我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
    原文