Pugss_风 发表于 2020-5-27 10:40:58

课后动动手习题答案存疑

各位大佬,想请教一下,在零基础入门课程23-34的课后拆分多位数数字的动动手练习中,当程序为:
num_list = []
def get_digits(n):
   
    if (n > 0):
      num_list.insert(0, n % 10)
      get_digits(n // 10)
      return num_list


n = int(input('请输入一个整数:'))
result = get_digits(n)
print(result)      
时可以输出正确结果,但是当把全局变量num_list放在方法中定义时候(程序如下),却只能得到个位数,请问这是为什么?
def get_digits(n):
    num_list = []
    if (n > 0):
      num_list.insert(0, n % 10)
      get_digits(n // 10)
      return num_list


n = int(input('请输入一个整数:'))
result = get_digits(n)
print(result)      

Twilight6 发表于 2020-5-27 10:43:37

本帖最后由 Twilight6 于 2020-5-27 10:48 编辑

因为对于这个函数命名空间里 第一次赋值的 num_list 是全局变量 每次递归的num_list 都是重新赋值了 [] 空列表,会触发屏蔽效果

当函数内赋值的变量名和函数外变量名相同时,会触发python 的屏蔽效果,会暂时屏蔽外部的全局变量,创建一个同名的局部变量来覆盖全局变量

只要一出了这个函数,局部变量就不复存在

所以说 递归后的 num_list 是一个局部变量 而不是原来的列表,而你每次都是向局部变量 num_list 添加元素

而递归最后结果只返回第一次加入列表的结果

sunrise085 发表于 2020-5-27 10:45:23

因为,放到函数里面的话,每次递归调用的时候,num_list都会被重新赋值为空列表
放到外面,则不会被重新置空,而是逐个添加元素

Pugss_风 发表于 2020-5-27 10:55:08

Twilight6 发表于 2020-5-27 10:43
因为对于这个函数命名空间里 第一次赋值的 num_list 是全局变量 每次递归的num_list 都是重新赋值了 [] 空 ...

这种的话,似乎不太能够通过在调用自身前加入nonlocal关键字声明非局部变量的方式来解决?

Twilight6 发表于 2020-5-27 10:56:03

Pugss_风 发表于 2020-5-27 10:55
这种的话,似乎不太能够通过在调用自身前加入nonlocal关键字声明非局部变量的方式来解决?

嗯不能 因为递归过程 你函数本身就算 '内部函数'   而你第一次赋值的 num_list 本来就不需要声明 会报错的
页: [1]
查看完整版本: 课后动动手习题答案存疑