鱼C论坛

 找回密码
 立即注册
查看: 110|回复: 5

[技术交流] 迭代魔法方法问题

[复制链接]
发表于 2024-10-27 22:56:54 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能^_^

您需要 登录 才可以下载或查看,没有账号?立即注册

x
本帖最后由 鬼少 于 2024-10-27 23:03 编辑

有如下一段代码:
class Fibs:
    def __init__(self,n = 20):
        self.a = 0
        self.b = 1
        self.n = n

    def __iter__(self):
        return self

    def __next__(self):
        self.a,self.b = self.b,self.a + self.b
        if self.a > self.n:
            raise StopIteration
        
        return self.a

当执行:
fibs = Fibs()
for each in fibs:
    print(each)

打印结果:
1
1
2
3
5
8
13

为何打印结果不是这样的?
1
1
2
3
5
8
13
21

为何不输出21这个值呢?

想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

发表于 2024-10-27 23:03:13 | 显示全部楼层
在 Python 中,`StopIteration` 异常用于标识迭代的结束。具体到您的代码,`StopIteration` 异常是在 `if self.a > self.n:` 这行代码之后引发的,当 `self.a` 达到或超过 `self.n` 时,迭代器就会停止工作,不再返回任何值,也就不会执行 `return self.a`。

以下是逐步解释:

1. 在 `__next__` 方法中,`self.a` 和 `self.b` 的值不断更新。
2. 每次执行 `__next__` 时,先会更新 `self.a` 和 `self.b`,然后判断 `self.a` 是否大于 `self.n`。
3. 如果 `self.a` 大于 `self.n`(在您的例子中 `self.a` 达到 `21` 时),会引发 `StopIteration` 异常。
4. 迭代停止后,`for each in fibs:` 循环会捕获到 `StopIteration` 异常并停止迭代。
5. 因为一旦 `StopIteration` 异常引发,意味着迭代结束,Python 不会再有机会执行 `return self.a`。

所以,`21` 从未被返回而打印。正确行为是,迭代器在遇到终止条件后,立即退出循环,且不再返回任何新的结果。

总结一下,当 `StopIteration` 异常引发时,循环就会终止,`21` 不会被返回或打印。

以上回复来自 FishC AI 小助理,如未能正确解答您的问题,请继续追问。
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2024-10-27 23:07:06 | 显示全部楼层
FishC 发表于 2024-10-27 23:03
在 Python 中,`StopIteration` 异常用于标识迭代的结束。具体到您的代码,`StopIteration` 异常是在 `if s ...

代码__next__中的
        "self.a,self.b = self.b,self.a + self.b"
叫什么赋值方式?
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2024-10-27 23:07:13 | 显示全部楼层
鬼少 发表于 2024-10-27 23:07
代码__next__中的
        "self.a,self.b = self.b, ...


这行代码 `self.a, self.b = self.b, self.a + self.b` 采用的是Python中的一种**多重赋值**(也叫**多元赋值**)方式。

在这行代码中,`self.b` 和 `self.a + self.b` 都被计算出来并同时赋值给 `self.a` 和 `self.b`。其工作原理如下:

1. **计算右边的表达式**:首先计算表达式 `self.b` 和 `self.a + self.b`,并将它们的值存储在临时变量中。
2. **同时赋值**:将右边的值同时赋给左边对应的变量,即 `self.a` 接受 `self.b` 的值,`self.b` 接受 `self.a + self.b` 的值。

这种方式的优点是简洁和高效,避免了在更新变量值时需要使用临时变量或者多行代码。

例如,下面的代码:
self.a, self.b = self.b, self.a + self.b
等价于下面的分步操作,但更加紧凑和直观:
temp_a = self.a
    temp_b = self.b
    self.a = temp_b
    self.b = temp_a + temp_b

所以,这种赋值方式叫做**多重赋值**(Multiple Assignment)。

以上回复来自 FishC AI 小助理,如未能正确解答您的问题,请继续追问。
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2024-10-27 23:22:33 | 显示全部楼层
          因为在 __next__() 中,当 self . a > self . n 的时候,会抛出 StopIteration 异常,停止迭代,无法同时返回 self . a,所以,不会打印 21。  
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2024-10-28 19:59:59 | 显示全部楼层
jackz007 发表于 2024-10-27 23:22
因为在 __next__() 中,当 self . a > self . n 的时候,会抛出 StopIteration 异常,停止迭代, ...

嗯嗯
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|鱼C工作室 ( 粤ICP备18085999号-1 | 粤公网安备 44051102000585号)

GMT+8, 2024-11-21 19:28

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表