鱼C论坛

 找回密码
 立即注册
查看: 1889|回复: 10

[已解决]第19讲课后习题关于global var的疑惑

[复制链接]
发表于 2020-8-24 10:38:42 | 显示全部楼层 |阅读模式

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

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

x
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不适用呢
最佳答案
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 '
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

发表于 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 '
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 1 反对 0

使用道具 举报

发表于 2020-8-24 10:42:08 | 显示全部楼层
[b]
(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

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

使用道具 举报

发表于 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 '

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

使用道具 举报

发表于 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'了
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

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

好的,在fun3处添加return var返回是baby,可以解释一下吗
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 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 '
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

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

谢谢大佬,我自己再思考看看!
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 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'可以打出这个变量吗?
(可能觉得我有点在死磕,但是真的想弄懂,再次期待和谢谢大佬的指导了!
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))
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

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


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

a='FishC'
fun3(a)

难道你会认为x没有定义吗?
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2020-9-3 17:51:42 | 显示全部楼层
本帖最后由 uranometria 于 2020-9-3 17:52 编辑
sunrise085 发表于 2020-9-3 16:15
你这不是全局变量和局部变量的问题,是形参和实参的问题。。。
你看下面的例子,你认为fun3 ...


x是形参 a= 'fishc'是实参...对吗?

第13行的fun3的实参不是baby吗?(捂脸哭
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

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

本版积分规则

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

GMT+8, 2025-1-18 20:20

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

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