手记

修改全局变量时,可变类型和不可变类型的区别

先说结论:

嵌套函数(如闭包,修饰器等),需要修改全局变量时,如果全局变量是不可变类型,那么在修改全局变量时要先写‘global 变量名‘,此时才可以修改;如果全局变量是可变类型,则可以直接修改。

底层机制为:函数内修改全局变量时,不可以更改全局变量的内存地址路径。调用的话是随便的,不需要进行其他特殊操作。

for一颗栗子

from functools import wraps
tag = False  #不可变类型的全局变量
def auth(name=None):
    def app(func):
        @wraps(func)
        def check(*args,**kwargs):
            global tag  #声明全局变量
            if tag == False:
                name = input('your name:').strip()
                pwd = input('your password:').strip()
                with open('c.txt',mode='rt',encoding='utf-8') as f:
                    dic = eval(f.read())  #eval可以理解为将内容原封不动的输出,原来是字典,输出还是字典
                    if name in dic and dic[name] == pwd:
                        print('login successful')
                        tag = True  #修改不可变类型的全局变量,使后续函数不用再次认证
                        res=func(*args,**kwargs)
                        return res

                    else:
                        print('error')
            else:
                res = func(*args,**kwargs)
                return res
        return check
    return app

@auth()
def index():
    print('from index')


index()

@auth()
def home(name):
    print('welcome %s to home' %name)

home('egon')

 

上下两个例子分别使用了三层和两层函数,注意两者修饰器引用上的区别,还有全局变量和函数内变量之间的关系(name)

from functools import wraps
tag = {'name':None,'pwd':False}  #字典是可变类型,修改字典时不会修改tag所对应的数据路径
def app(func):
        @wraps(func)
        def check(*args,**kwargs):
            if tag['name'] and tag['pwd']:
                res = func(*args,**kwargs)
                return res
            else:
                name = input('your name:').strip()
                pwd = input('your password:').strip()
                with open('c.txt',mode='rt',encoding='utf-8') as f:
                    dic = eval(f.read())  
                    if name in dic and dic[name] == pwd:
                        print('login successful')
                        tag['name']=True
                        tag['pwd']=True
                        res=func(*args,**kwargs)
                        return res

                    else:
                        print('error')
        return check

@app
def index():
    print('from index')


index()

@app
def home(name):
    print('welcome %s to home' %name)

home('egon')

作者:real-admin

原文:https://www.cnblogs.com/realadmin/p/10049224.html

0人推荐
随时随地看视频
慕课网APP