为了账号安全,请及时绑定邮箱和手机立即绑定

小白求解答

为什么这样会出现变量未定义:

def calc_prod(list):

    s=1.0 在python的函数中和全局变量同名的变量,如果有修改变量的值就会变成局部变量,在修改之前对该变量的引用自然就会出现没有定义的错误

    def lazy_prod():

        for i in list:

            s=s*i

        return s

    return lazy_prod

f=calc_prod([1,2,3,4])

f

f()

而这样却不会呢?

def count():

    fs=[]

    for i in range(1,4):

        def f():

            return i*i

        fs.append(f)

    return fs

f1,f2,f3=count()

f1(),f2(),f3()


正在回答

1 回答

Python中的数据类型分为“不可变数据类型”和“可变数据类型”,不可变数据类型更改后地址发生改变,可变数据类型更改地址不发生改变。不可变数据类型,不允许变量的值发生变化,如果改变了变量的值,相当于是新建了一个对象。列表属于可变数据类型,浮点数属于不可变数据类型。

而 s = s*i    是先调用等号右边 s*i 中的s,此时系统会新建一个名为s的对象,s将不再指向原来的1,后面也没有对s的赋值操作,因此此时s并没有值,所以会报未定义的错误。列表是可变数据类型,运用append之后并不会创建新的对象,因此还是原来的那个对象。

至于你说的“在python的函数中和全局变量同名的变量,如果有修改变量的值就会变成局部变量,在修改之前对该变量的引用自然就会出现没有定义的错误”,这句话应该是有问题的,如果有修改,则会创建新的对象指向新的地址,而不是简单地从全局变局部。下面这段代码就没有任何问题,因为它没有对变量进行修改,只是单纯的引用了变量:

x = 1
def function1():
    return x
print function1()


0 回复 有任何疑惑可以回复我~
#1

宝慕林6305465 提问者

谢谢你,我再好好参透一下
2019-05-13 回复 有任何疑惑可以回复我~
#2

宝慕林6305465 提问者

我想再问一下‘而 s = s*i 是先调用等号右边 s*i 中的s,此时系统会新建一个名为s的对象,s将不再指向原来的1,后面也没有对s的赋值操作,因此此时s并没有值,所以会报未定义的错误。’这句话不是已经对新建的s赋值s=s*i了吗?为什么还是未赋值呢?难道s=s*i只是创建一个s变量吗?s变量并没有指向具体的值吗?
2019-05-13 回复 有任何疑惑可以回复我~
#3

荼灬 回复 宝慕林6305465 提问者

s = s*i,你企图改变s的值,编译器先创建一个名为s 的对象,并且将s指空,也就是等号右边的s指空。实际上这个时候数值1的地址已经被回收(你再也找不到1了),也就是说左边的s在赋值给右边的s之前,自己就已经指空了,然后这个等号的作用就是将一个空值赋给另一个空值,所以系统会认为你没有给s变量赋值
2019-05-13 回复 有任何疑惑可以回复我~
#4

宝慕林6305465 提问者 回复 荼灬

谢谢你,你好厉害,学到了
2019-05-13 回复 有任何疑惑可以回复我~
查看1条回复

举报

0/150
提交
取消
python进阶
  • 参与学习       255665    人
  • 解答问题       2949    个

学习函数式、模块和面向对象编程,掌握Python高级程序设计

进入课程

小白求解答

我要回答 关注问题
意见反馈 帮助中心 APP下载
官方微信