一隻太平洋睡鯊 发表于 2020-7-26 19:59:53

關於[第020讲:函数:内嵌函数和闭包 | 课后测试题及答案]

一、
def funX():
    x = 5
    def funY():
      nonlocal x
      x += 1
      return x
    return funY

a = funX()
print(a())
print(a())
print(a())

為什麼以上代碼是打印?
6
7
8

看了小甲魚的解釋,沒看懂,有人能更詳細解釋嗎?

二、
str1 = '''ABSaDKSbRIHcRHGcdDIF'''

countA = 0# 统计前边的大写字母
countB = 0# 统计小写字母
countC = 0# 统计后边的大写字母
length = len(str1)

for i in range(length):
    if str1 == '\n':
      continue

    """
    |如果str1是大写字母:
    |-- 如果已经出现小写字母:
    |-- -- 统计后边的大写字母
    |-- 如果未出现小写字母:
    |-- -- 清空后边大写字母的统计
    |-- -- 统计前边的大写字母
    """
    if str1.isupper():
      if countB:
            countC += 1
      else:
            countC = 0
            countA += 1

    """
    |如果str1是小写字母:
    |-- 如果小写字母前边不是三个大写字母(不符合条件):
    |-- -- 清空所有记录,重新统计
    |-- 如果小写字母前边是三个大写字母(符合条件):
    |-- -- 如果已经存在小写字母:
    |-- -- -- 清空所有记录,重新统计(出现两个小写字母)
    |-- -- 如果该小写字母是唯一的:
    |-- -- -- countB记录出现小写字母,准备开始统计countC
    """
    if str1.islower():
      if countA != 3:
            countA = 0
            countB = 0
            countC = 0
      else:
            if countB:
                countA = 0
                countB = 0
                countC = 0
            else:
                countB = 1
                countC = 0
                target = i

    """
    |如果前边和后边都是三个大写字母:
    |-- 如果后边第四个字母也是大写字母(不符合条件):
    |-- -- 清空记录B和C,重新统计
    |-- 如果后边仅有三个大写字母(符合所有条件):
    |-- -- 打印结果,并清空所有记录,进入下一轮统计
    """
    if countA == 3 and countC == 3:
      if i+1 != length and str1.isupper():
            countB = 0
            countC = 0
      else:
            print(str1, end='')
            countA = 3
            countB = 0
            countC = 0


我發現在最後面的地方
      else:
            print(str1, end='')
            countA = 3
            countB = 0
            countC = 0
如果把
            countA = 3
改成
            countA = 0

或是乾脆把
            countA = 3
            countB = 0
            countC = 0
都拿掉
也會得到一樣的結果,請問這麼做對代碼會有什麼影響呢?
而小甲魚加上那段代碼的理由是...?

xiaofeiyu 发表于 2020-7-27 10:14:51

本帖最后由 xiaofeiyu 于 2020-7-27 10:19 编辑

变量a的值funX()的返回值,因为funX加了括号,就会自动执行funX()的内容,而funX()的返回值是funY   (没有括号)
所以a就代表funY
执行a()就是执行funY()
执行三次a(),每次都把x+1,x最初是5,所以结果是5+1=6, 6+1=7, 7+1=8

而(2)中
最后面已经打印出了结果,countA countB countC重新赋值只是为了重置,没有其他用处
所以随便修改或删掉不影响结果。print语句在前面已经执行了

一隻太平洋睡鯊 发表于 2020-7-27 20:35:06

xiaofeiyu 发表于 2020-7-27 10:14
变量a的值funX()的返回值,因为funX加了括号,就会自动执行funX()的内容,而funX()的返回值是funY   (没有 ...

抱歉,關於第一個問題,我還是不懂為何不會被funX函數中的x = 5影響到?

xiaofeiyu 发表于 2020-7-28 09:07:50

一隻太平洋睡鯊 发表于 2020-7-27 20:35
抱歉,關於第一個問題,我還是不懂為何不會被funX函數中的x = 5影響到?

a=funx()就运行了funX(),所以x=5,这时a的值就变成了funY,和funX()没有了一点关系
其实就相当于:
x=5
def funY():
    x+=1
    reuturn x
def funX():
    return funY

a=funx() # 此时的a是返回值funY
a() # 运行的是funY,x没有被重置
a() # 同理
a() # 同理

一隻太平洋睡鯊 发表于 2020-7-28 22:38:08

xiaofeiyu 发表于 2020-7-28 09:07
a=funx()就运行了funX(),所以x=5,这时a的值就变成了funY,和funX()没有了一点关系
其实就相当于:

也就是說funX()在這個代碼裡面只運行了一次
導致x初始值為5
之後a被附值funY
print(a())
print(a())
print(a())
運行三次funY
我這樣理解對嗎?

可以如果是這樣的話
不是說內包含數不能在區域外使用
這樣不是等於在區域外使用了內包含數嗎?
還是說有想叫出內包含數時就這麼做的慣例?

xiaofeiyu 发表于 2020-7-29 21:00:47

一隻太平洋睡鯊 发表于 2020-7-28 22:38
也就是說funX()在這個代碼裡面只運行了一次
導致x初始值為5
之後a被附值funY


对的

xavier113 发表于 2020-11-8 00:04:33

xiaofeiyu 发表于 2020-7-27 10:14
变量a的值funX()的返回值,因为funX加了括号,就会自动执行funX()的内容,而funX()的返回值是funY   (没有 ...

茅塞頓開,感謝大神
页: [1]
查看完整版本: 關於[第020讲:函数:内嵌函数和闭包 | 课后测试题及答案]