Python教程(26)——Python迭代器和生成器详解

这篇具有很好参考价值的文章主要介绍了Python教程(26)——Python迭代器和生成器详解。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

迭代器

Python中的迭代器是一种对象,它可以迭代(遍历)一个可迭代对象(比如列表、元组或字符串)的元素。迭代器用于实现迭代器协议,即包含 __iter__() 方法和 __next__() 方法。

迭代器的工作原理是每次调用 __next__() 方法时返回可迭代对象的下一个元素,当没有元素可迭代时,抛出 StopIteration 异常。

class MyIterator:
    def __init__(self, iterable):
        self.iterable = iterable
        self.index = 0

    def __iter__(self):
        return self

    def __next__(self):
        if self.index < len(self.iterable):
            result = self.iterable[self.index]
            self.index +=1
            return result
        else:
            raise StopIteration

# 创建一个可迭代对象
my_list = [1, 2, 3, 4, 5]

# 创建一个迭代器
my_iterator = MyIterator(my_list)

# 使用迭代器遍历元素
for item in my_iterator:
    print(item)

值得注意的是,当迭代器耗尽后,如果再次使用迭代器来便利,将不会得到任何输出。
所以总的来说,迭代器是用于遍历可迭代对象的对象,它实现了迭代器协议,具有 __iter__()__next__() 方法。

可迭代对象

我们看迭代器好像和平时我们使用的列表、字典等数据结构一样,都可以遍历,那么列表、字典等数据结构是迭代器吗?

不好意思,他们不是迭代器,而是可迭代对象(iterable)。

可迭代对象(iterable)是指具有迭代行为的对象。当我们希望能够按照一定方式遍历对象中的元素时,我们可以将该对象称为可迭代对象。换句话说,可迭代对象是一种提供迭代能力的容器。
可迭代对象的特点是实现了 __iter__() 方法,这个方法返回一个迭代器(iterator)。迭代器是能够按照一定顺序生成下一个元素的对象。

在 Python 中,许多数据结构都是可迭代对象,比如列表、元组、集合、字典等。我们可以使用for循环对这些对象进行遍历。

同时,也可以使用内置的 iter() 函数将可迭代对象转换为迭代器。迭代器是可迭代对象的一种特殊形式,实现了 __iter__()__next__() 方法。迭代器可以使用 next() 函数来获取下一个元素,并且在没有元素可返回时引发 StopIteration 异常。

my_list = [1, 2, 3, 4, 5]  # 列表是可迭代对象
for item in my_list:
    print(item)

my_iterator = iter(my_list)  # 使用iter()函数将列表这个可迭代对象转换为迭代器
print(next(my_iterator))  # 输出第一个元素
print(next(my_iterator))  # 输出第二个元素

在这个示例中,列表 my_list 是可迭代对象,它可以被 for 循环遍历。另外,我们还使用 iter() 函数将 my_list 转换为迭代器 my_iterator,并使用 next() 函数逐个访问其中的元素。

所以总的来说,可迭代对象是指具有迭行为的对象,它们实现了 __iter__() 方法。通过for循环或 iter() 函数,我们可以遍历这些对象的元素。

可迭代对象是指实现了 __iter__() 方法的对象,而迭代器是实现了 __iter__()__next__() 方法的对象,这个可以说是它们比较明显的区别。

for循环机制

从上面我们指定,列表、元组、集合、字典等数据结构是可迭代对象,并不是迭代器。而可迭代对象只实现了__iter__()方法,并不具有迭代(也就是返回下一个元素)的功能。那么很多同学可能就比较奇怪了,我们平时使用for循环遍历这个数据结构的时候,内部是怎么遍历的呢?

实际上在 Python 中,for循环在内部自动会调用 __iter__() 函数将可迭代对象转换为迭代器。用for循环遍历可迭代对象的实现机制为:

  • for循环首先会调用 __iter__() 函数,该函数会将可迭代对象转换为一个迭代器对象(如果对象本身就是迭代器,则不作转换)。
  • 接下来,for循环会调用迭代器对象的 __next__() 方法来获取下一个元素。如果迭代器对象没有下一个元素,会抛出 StopIteration 异常。
  • for循环会自动捕捉 StopIteration 异常,表示已经迭代完所有元素,循环将结束。

所以以下两个方法实际上是等价的。

my_string = "Hello"

for char in my_string:
    print(char)

实际上完全等价于:

my_string = "Hello"
my_iterator = iter(my_string)

print(next(my_iterator))  # 输出:H
print(next(my_iterator))  # 输出:e
print(next(my_iterator))  # 输出:l
print(next(my_iterator))  # 输出:l
print(next(my_iterator))  # 输出:o

生成器

生成器(Generator)是一种特殊的迭代器,它可以在迭代过程中动态地生成值,而不是一次性地将所有元素放在内存中。生成器使用 yield 关键字来定义,当生成器的代码块执行到 yield 语句时,就会暂停执行并返回一个值,下次调用时会从上次暂停的位置继续执行。这样可以在需要的时候生成值,而不是一次性生成所有的值。

def my_generator():
    yield 1
    yield 2
    yield 3

# 使用生成器
gen = my_generator()

print(next(gen))  # 输出:1
print(next(gen))  # 输出:2
print(next(gen))  # 输出:3
print(next(gen))  # 抛出 StopIteration 异常

在上面的示例中,我们定义了一个名为 my_generator 的生成器函数,它通过使用 yield 关键字来产生值。当我们调用生成器函数时,它返回一个生成器对象 gen。我们可以使用 next() 函数来逐个获取生成器的值。每次调用 next() 时,生成器函数会从上次暂停的位置继续执行,并返回 yield 语句的值。当生成器函数执行完毕或没有更多的值可生成时,调用 next() 会抛出 StopIteration 异常。

生成器的一个重要特点是它们可以节省内存,尤其在处理大量数据时非常有用。由于生成器是按需生成值,只有在需要时才会在迭代过程中生成值,不会一次性占用大量的内存。所在遍历大批量数据的时候,非常有用,因为如果将大批量的数据直接加载到内存中在遍历,肯定会消耗很多内存,而利用生成器就可以做到需要哪些遍历哪些。

如果用简单一句话来说就是,我既想大量的数据,又想让它占用空间少,实现鱼和熊掌的兼得,那么就用生成器!

def read_large_file(file_path):
    with open(file_path, 'r') as file:
        for line in file:
            yield line.rstrip()

# 使用生成器遍历大文件
file_generator = read_large_file('large_file.txt')
print(file_generator) # 输出<generator object fibonacci_generator at 0x0000017362DCFED0>
print(type(file_generator)) # 输出<class 'generator'>
for line in file_generator:
    # 处理每一行数据
    print(line)

从上面可以看出,read_large_file() 函数是一个生成器函数,它按行从一个大文件中读取数据。通过使用 yield 关键字,在每次迭代时逐行生成文件的内容,并将其作为生成器的值返回。然后,我们可以使用 for 循环逐行处理大文件。可以看到,我们打印的file_generator类型是一个生成器。

当然,以上的例子并不一定需要采用生成器才能处理,我们直接在第4行进行处理也是可以的,生成器更多的只是提供一种思路,当你用常规方法不能解决问题的时候,可以试试用生成器。

生成器原理

生成器的原理基于迭代器(iterators)和生成器函数(generator functions)。

def even_numbers(n):
    for i in range(1, n+1):
        if i % 2 == 0:
            yield i

# 创建生成器对象
even_generator = even_numbers(10)

# 打印生成的偶数
for number in even_generator:
    print(number)

生成器函数

生成器通过生成器函数创建,生成器函数是一种特殊类型的函数,使用 yield 语句来生成值。像上面的例子even_numbers函数就是要给生成器函数,当调用生成器函数时,它返回一个生成器对象,而不是立即执行函数体内的代码,even_generator就是一个生成器对象。

而之前我们说过生成器是一种特殊类型的迭代器,它可以在迭代中生成值。迭代器是一个实现了 __iter__()__next__() 方法的对象。__iter__() 方法返回迭代器本身,而for循环内部会自动调用 __next__() 方法用于获取下一个值。每次调用 __next__() 方法时,生成器会从上一次暂停的位置继续执行,直到遇到下一个 yield 语句,然后将 yield 后面的值返回给调用者。

逐个生成值

生成器在调用 __next__() 方法时逐个生成值,并且每次在生成一个值后会暂停执行。这种延迟生成的机制使得生成器能够处理大量数据或无限序列,而不需要一次性加载或计算所有值。

状态保存

生成器在暂停执行时会保存其状态,包括局部变量、指令指针等信息。下一次调用 __next__() 方法时,生成器会从上一次暂停的地方恢复执行,并继续执行剩余部分的代码。文章来源地址https://www.toymoban.com/news/detail-825116.html

到了这里,关于Python教程(26)——Python迭代器和生成器详解的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 一文详解列表,元组,字典,集合,生成器,迭代器,可迭代对象,zip,enumerate

      列表,元组,字典,集合,生成器都是 python 中的可迭代对象,使用的时候经常忘记,通过这篇博文总结一下。   列表( list )是 Python 中的一种数据结构,它可以存储不同类型的数据。不同元素以逗号分隔。 使用方括号[]表示开始和结束。 不同元素以逗号分隔。 每个

    2024年02月07日
    浏览(61)
  • python中函数,装饰器,迭代器,生成器

    1.函数可以作为参数进行传递 2.函数可以作为返回值进行返回 3.函数名称可以当成变量一样进行赋值操作 作用:在不改变原有函数调用的情况下,给函数增加新的功能          即可以在函数前面增加新的功能,但不改变原来的代码 可迭代的数据类型都会提供迭代器,即可以

    2024年02月07日
    浏览(35)
  • Python迭代器与生成器研究记录

    迭代器肯定是可迭代对象,但是可迭代对象不一定是迭代器,生成器一定是迭代器,但是迭代器不一定是生成器 生成器是特殊的迭代器,所以生成器一定是迭代器,迭代器一定是可迭代对象 我们平常接触最多的对象中,字符串,字典,列表,集合,元组和open打开的文件对象

    2024年02月05日
    浏览(40)
  • 3.0 Python 迭代器与生成器

    当我们需要处理一个大量的数据集合时,一次性将其全部读入内存并处理可能会导致内存溢出。此时,我们可以采用迭代器 Iterator 和生成器 Generator 的方法,逐个地处理数据,从而避免内存溢出的问题。 迭代器是一个可以逐个访问元素的对象,它实现了 python 的迭代协议,即

    2024年02月13日
    浏览(34)
  • Python中的迭代器与生成器

    在Python中,迭代器(Iterator)和生成器(Generator)是两种用于处理可迭代对象的重要工具。而可迭代对象包括列表,元组,字典,字符串等。 迭代器和生成器只能迭代一次,通常用于处理大型数据集,因为它们不会一次性加载所有数据到内存中,而是根据需要逐个生成值。

    2024年02月10日
    浏览(37)
  • Python基础篇(十):迭代器与生成器

    迭代器和生成器是Python中用于 处理可迭代对象 的重要概念。它们提供了一种有效的方式来 遍历和访问 集合中的元素,同时具有 节省内存和惰性计算 的特点。下面是关于迭代器和生成器的详细介绍和示例: 迭代器是一种实现了迭代协议的对象,它可以用于遍历集合中的元素

    2024年02月10日
    浏览(33)
  • 【python高级用法】迭代器、生成器、装饰器、闭包

    可迭代对象:可以使用for循环来遍历的,可以使用isinstance()来测试。 迭代器:同时实现了__iter__()方法和__next__()方法,可以使用isinstance()方法来测试是否是迭代器对象 使用类实现迭代器 两个类实现一个迭代器 一个类实现迭代器 可迭代对象与迭代器的总结 一个具备了__iter_

    2024年02月03日
    浏览(29)
  • Python小姿势 - ## Python中的迭代器与生成器

    Python中的迭代器与生成器 在Python中,迭代是一个非常重要的概念,迭代器和生成器是迭代的两种最常见的形式。那么,迭代器与生成器有何不同呢? 首先,我们先来了解一下迭代器。 迭代器是一种对象,它可以记住遍历的位置,并在每次访问时返回下一个元素。迭代器只能

    2024年02月04日
    浏览(42)
  • 【Python 4】列表与元组slice切片 迭代 列表生成式 生成器generator 迭代器Iterator对象

    在Python中,代码不是越多越好,而是越少越好 取一个list或tuple的部分元素是非常常见的操作 对这种经常取指定索引范围的操作,用循环十分繁琐,因此,Python提供了切片(Slice)操作符,能大大简化这种操作 L[0:3]表示,从索引0开始取,直到索引3为止,但不包括索引3 如果第

    2024年02月07日
    浏览(41)
  • python使用迭代生成器yield减少内存占用的方法

    在python编码中for循环处理任务时,会将所有的待遍历参量加载到内存中。 其实这本没有必要,因为这些参量很有可能是一次性使用的,甚至很多场景下这些参量是不需要同时存储在内存中的,这时候就会用到本文所介绍的迭代生成器yield。 首先我们用一个例子来演示一下迭代

    2024年04月28日
    浏览(24)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包