我爱吃爆米花 发表于 2020-8-24 10:38:42

第19讲课后习题关于global var的疑惑

var = ' Hi '

def fun1():
    global var
    var = ' Baby '
    return fun2(var)

def fun2(var):
    var += 'I love you'
    fun3(var)
    return var

def fun3(var):
    var = ' 小甲鱼 '

print(fun1())

疑惑:
(1)在定义fun1函数过程中的使用global var,那么全局变量var的应用范围不应该是此份代码里的所有var吗,为什么fun2里的var适用为全局变量,而fun3里面的var不适用呢

sunrise085 发表于 2020-8-24 10:41:08

本帖最后由 sunrise085 于 2020-8-24 10:42 编辑

在 fun1中的global值对fun1有效,对fun2无效
写了注释,你看一下。这是变量作用域的问题。局部变量和全局变量其实是不同的变量,你可以用print(id(var))看一下就知道了,可以分别在三个函数和主程序中输出var的id就可以看出来了。
var = ' Hi '            #全局变量var,称之为var0,变量var0=' Hi '

def fun1():
    global var          #全局变量var0,变量var0=' Hi '
    var = ' Baby '      #全局变量var0,变量var0=' Baby '
    return fun2(var)    #全局变量var0,实参var0=' Baby '。得到的返回值在第11行

def fun2(var):          #局部变量var,称之为var1,变量var1=' Baby '
    var += 'I love you' #局部变量var1=' Baby I love you'
    fun3(var)         #局部变量var1,实参var1=' Baby I love you'。fun3()不影响var1的值
    return var          #将var1作为返回值,var1=' Baby I love you'。fun2()的返回值不影响var0的值

def fun3(var):          #局部变量var,称之为var2,变量var2=' Baby I love you'
    var = ' 小甲鱼 '    #局部变量var2=' 小甲鱼 '

print(fun1())         #该语句打印的是fun1()的返回值,即第6行的return内容,也即第11行的返回值' Baby I love you'
print(var)            #全局变量var0=' Baby '

Twilight6 发表于 2020-8-24 10:42:08


(1)在定义fun1函数过程中的使用global var,那么全局变量var的应用范围不应该是此份代码里的所有var吗,为什么fun2里的var适用为全局变量,而fun3里面的var不适用呢

只对于当前函数内声明是全局,但是到另一个函数要重新声明,否则就会触发屏蔽机制

看看这个解析整个题目,应该对你有所帮助:



先了解 Python 的屏蔽机制(Ps:都是指没有 global 和 nonlocal声明下,因为如果声明了就不是局部变量了):

当函数内部的变量和函数外部全局作用域的变量名相同时,会触发 Python 的屏蔽机制,函数内部会创建一个同名的局部变量

来暂时覆盖同名的全局变量,此时在函数内部的值都为这个局部变量,只要一出此函数,那么该局部变量就不复存在,变会原来的全局变量

了解了上面的屏蔽机制我们在来看看代码:
var = ' Hi '    # 这个是全局变量
先调用的是 fun1():
def fun1():
    global var      # 声明变量是全局变量
    var = ' Baby '# 对全局变量重新赋值
    return fun2(var)
这边用global 声明了 var 函数是全局变量,所以就不会触发屏蔽机制,给 var 重新赋值 'Baby' 导致覆盖之前的 var ,var = 'Baby' 返回 fun2
然后运行带 fun2(var) 此时传入的var = 'Baby':
def fun2(var):
    var += 'I love you'   # 没有声明全局变量 则这可以看成var = var + 'I love you'    这里等号右边的 var 是全局变量 等号左边的 var 是局部变量
    fun3(var)
    return var
因为 var += 'I love you'可以等价看成var = var + 'I love you'   所以 var 拼接上'I love you' 时 重新赋值给 var 触发屏蔽机制

此时暂时屏蔽全局变量的 var = ' Baby ' ,而为 var = ' Baby I love you ' 局部变量 ,然后传入fun3函数
def fun3(var):
    var = ' 小甲鱼 '   # 没有声明,是局部变量
这边再次屏蔽机制,因为对于fun3来说 fun2函数内以及fun2函数外的都算是全局作用域,所以这里赋值 var = ' 小甲鱼 ' 时

暂时屏蔽了 var = ' Baby I love you ' 但是一出 fun3就不复存在 变全局变量 var = ' Baby I love you '

所以最后回到 fun2 返回结果为 var局部变量的结果,即是 Baby I love you

uranometria 发表于 2020-9-3 11:33:57

本帖最后由 uranometria 于 2020-9-3 11:35 编辑

sunrise085 发表于 2020-8-24 10:41
在 fun1中的global值对fun1有效,对fun2无效
写了注释,你看一下。这是变量作用域的问题。局部变量和全局 ...

大佬您好,fun3处用的是全局变量baby吗?var2为什么是'Baby I love you'呢?


var = ' Hi '            #全局变量var,称之为var0,变量var0=' Hi '

def fun1():
    global var          #全局变量var0,变量var0=' Hi '
    var = ' Baby '      #全局变量var0,变量var0=' Baby '
    return fun2(var)    #全局变量var0,实参var0=' Baby '。得到的返回值在第11行

def fun2(var):          #局部变量var,称之为var1,变量var1=' Baby '
    #var += 'I love you' #局部变量var1=' Baby I love you'
    #fun3(var)         #局部变量var1,实参var1=' Baby I love you'。fun3()不影响var1的值
    return var          #将var1作为返回值,var1=' Baby I love you'。fun2()的返回值不影响var0的值

def fun3(var):          #局部变量var,称之为var2,变量var2=' Baby I love you'
    #var = ' 小甲鱼 '    #局部变量var2=' 小甲鱼 '####大佬您好,请问将这里注释之后,用的是全局变量baby吗?var2为什么是'Baby I love you'呢?####
    return var

print(fun1())         #该语句打印的是fun1()的返回值,即第6行的return内容,也即第11行的返回值' Baby I love you'
print(var)            #全局变量var0=' Baby '


sunrise085 发表于 2020-9-3 11:45:20

uranometria 发表于 2020-9-3 11:33
大佬您好,fun3处用的是全局变量baby吗?var2为什么是'Baby I love you'呢?

你说的是13行吧?
不是全局变量,是局部变量
第13行的var是在第10行调用fun3的时候传递进来的,实参是' Baby I love you',传递进来之后,var2也就是' Baby I love you'了

uranometria 发表于 2020-9-3 11:57:06

sunrise085 发表于 2020-9-3 11:45
你说的是13行吧?
不是全局变量,是局部变量
第13行的var是在第10行调用fun3的时候传递进来的,实参 ...

好的,在fun3处添加return var返回是baby,可以解释一下吗{:10_266:}

sunrise085 发表于 2020-9-3 13:04:04

uranometria 发表于 2020-9-3 11:57
好的,在fun3处添加return var返回是baby,可以解释一下吗

你在四楼的程序,fun3根本没执行,因为没有调用过fun3
在fun2中没有修改过局部变量var1,因此第11行返回的var1还是' Baby ' ,fun2执行完之后返回调用fun2 的位置,即fun1的最后一句。
然后fun1就返回了
在17行print输出的就依然还是' Baby '

uranometria 发表于 2020-9-3 13:38:51

sunrise085 发表于 2020-9-3 13:04
你在四楼的程序,fun3根本没执行,因为没有调用过fun3
在fun2中没有修改过局部变量var1,因此第11行返 ...

{:10_302:}谢谢大佬,我自己再思考看看!

uranometria 发表于 2020-9-3 16:05:12

本帖最后由 uranometria 于 2020-9-3 16:06 编辑

sunrise085 发表于 2020-9-3 13:04
你在四楼的程序,fun3根本没执行,因为没有调用过fun3
在fun2中没有修改过局部变量var1,因此第11行返 ...

提问:第13行中,局部变量是可以传递下来的吗?(我的理解是fun2内的fun3的参数仅用于fun2内,出了fun2,var的变量应该还是baby),如果是' Baby I love you'可以打出这个变量吗?
(可能觉得我有点在死磕,但是真的想弄懂,再次期待和谢谢大佬的指导了!{:10_266:} )

var = ' Hi '            #全局变量var,称之为var0,变量var0=' Hi '

def fun1():
    global var          #全局变量var0,变量var0=' Hi '
    var = ' Baby '      #全局变量var0,变量var0=' Baby '
    return fun2(var)    #全局变量var0,实参var0=' Baby '。得到的返回值在第11行

def fun2(var):          #局部变量var,称之为var1,变量var1=' Baby '
    var += 'I love you' #局部变量var1=' Baby I love you'
    fun3(var)         #局部变量var1,实参var1=' Baby I love you'。fun3()不影响var1的值
    return var          #将var1作为返回值,var1=' Baby I love you'。fun2()的返回值不影响var0的值

def fun3(var):          #局部变量var,称之为var2,变量var2=' Baby I love you'【局部变量是可以传递下来的吗?(我的理解是fun2内的fun3的参数仅用于fun2内,出了fun2,var的变量应该还是baby),如果是' Baby I love you'可以打出这个变量吗?】
    #var = ' 小甲鱼 '    #局部变量var2=' 小甲鱼 '
    return var
   
print(fun1())         #该语句打印的是fun1()的返回值,即第6行的return内容,也即第11行的返回值' Baby I love you'
print(var)            #全局变量var0=' Baby '

print(fun3(var))

sunrise085 发表于 2020-9-3 16:15:23

uranometria 发表于 2020-9-3 16:05
提问:第13行中,局部变量是可以传递下来的吗?(我的理解是fun2内的fun3的参数仅用于fun2内,出了fun2 ...

{:5_104:}
你这不是全局变量和局部变量的问题,是形参和实参的问题。。。
你看下面的例子,你认为fun3中的x是什么?
def fun3(x):
    print(3*x)

a='FishC'
fun3(a)

难道你会认为x没有定义吗?

uranometria 发表于 2020-9-3 17:51:42

本帖最后由 uranometria 于 2020-9-3 17:52 编辑

sunrise085 发表于 2020-9-3 16:15
你这不是全局变量和局部变量的问题,是形参和实参的问题。。。
你看下面的例子,你认为fun3 ...

x是形参 a= 'fishc'是实参...对吗?{:10_266:}

第13行的fun3的实参不是baby吗?(捂脸哭{:10_269:} )
页: [1]
查看完整版本: 第19讲课后习题关于global var的疑惑