在学习装饰器的时候看到了这样的两段代码
代码段1
def login(func):#为多个函数加上的认证功能
def inner(*args,**kwargs):#加上装饰器
global FLAG#给函数内的全局变量赋值
if FLAG:
ret = func(*args, **kwargs) # func就是被装饰的函数
return ret
#'''写登录程序'''
else:
username=input('亲输入你的名字:')
password=input('亲输入你的密码:')
if username=='lxx' and password=='123':
FLAG=True
ret=func(*args,**kwargs) #func就是被装饰的函数
return ret
else:
print('错误,登录失败')
return inner
@login
def shoplist_add(): #两个函数有不同的功能
print('增加一件物品')
@login
def shoplist_del():
print('删除一件物品')
shoplist_add()
shoplist_del()
代码段2
user_list = [
{'name':'sb1','passwd':'123'},
{'name':'sb2','passwd':'123'},
{'name':'sb3','passwd':'123'},
{'name':'sb4','passwd':'123'}
]
#初始状态,用来保存登陆的用户,
client_dic = {'username':None,'login':False}
#添加新功能
def auth_func(func):
def wrapper(*args,**kwargs):
#参数检查,判断是否有用户登录,如果有,不用验证,直接执行函数的功能
if client_dic['username'] and client_dic['login']:
res = func(*args,**kwargs)
return res
#输入用户名和密码
username = input('用户名:').strip()
passwd = input('passwd:').strip()
#对比列表,检查用户名和密码是否正确
for user_dic in user_list:
if username == user_dic['name'] and passwd == user_dic['passwd']:
client_dic['username'] = user_dic['name']
client_dic['login'] = True
res = func(*args,**kwargs)
return res
else:
print('用户名或者密码错误!')
return wrapper
@auth_func
def index():
print("欢迎来到主页")
@auth_func
def home(name):
print("欢迎回家:%s"%name)
@auth_func
def shoppping_car():
print('购物车里有[%s,%s,%s]'%('奶茶','妹妹','娃娃'))
print(client_dic)
index()
print(client_dic)
home('root')
————————————————
版权声明:本文为CSDN博主「我是马克思小清新」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/qq_33531400/article/details/79324551
代码1无法正常运行,代码段2可以,疑惑的地方是代码1和代码2都引用了“外部变量”
同样没有提前声明,为什么代码1会报错
UnboundLocalError: local variable 'flag' referenced before assignment
代码2不会报错
上网查了才明白
首先要了解python的命名空间和作用域
但是python解释器在看到代码段1 FLAG=True 这一句之后,将FLAG识别为局部变量,此时不看外部的FLAG就会认为这句话就是一个变量的初始化,然而再看前面的 if FLAG: 因为FLAG是局部变量,但是使用的时候并未创建,所以会导致报错,此时只需要在掐面加一个global FLAG就可以解决问题
再说到代码段2中为什么字典没有使用globol声明也可以正常运行,是因为字典是可变变量,而布尔类型是不可变变量,在字典内容发生改变的时候,字典变量始终指向同一个内存空间,而布尔型变量进行改变的时候会指向不同的内存空间,所以python解释器会认为这是一次变量的初始化,而不是引用了原来的变量