First blood
System.out.println("hello world");
以前用装饰器,都是定义好了装饰器后,使用@装饰器名的方法写入被装饰函数的正上方
在这里,定义的装饰器都是没有参数的
在定义装饰器的函数的时候,没有在括号里定义参数,这就叫做无参装饰器
既然有无参装饰器,那么当然也就会有有参装饰器
db_path = "db.txt"
login_dic = {
"user": None,
"status": None,
}
def auth(func):
def wrapper(*args, **kwargs):
if login_dic['user'] and login_dic['status']:
res = func(*args, **kwargs)
return res
name = input("your name:").strip()
password = input("your password:").strip()
with open(db_path, "r", encoding="utf-8") as f:
user_dic = eval(f.read())
if name in user_dic and password == user_dic[name]:
print("login ok")
login_dic['user'] = name
login_dic['status'] = True
res = func(*args, **kwargs)
return res
else:
print("login error")
return wrapper
@auth
def index():
print("welcome to index page")
@auth
def home(name):
print("welcome to %s home page" % name)
index()
home()
上面的例子是一个典型的基于文件的认证方式的装饰器
在生产环境中,用户的认证方式可能有很多种,比如还有LDAP和数据库的认证方式
如果想把上面的代码修改,添加基于不同认证方式的用户认证
db_path = "db.txt"
login_dic = {
"user": None,
"status": None,
}
def deco(auth_type):
def auth(func):
def wrapper(*args, **kwargs):
if auth_type == "file":
if login_dic['user'] and login_dic['status']:
res = func(*args, **kwargs)
return res
name = input("your name:").strip()
password = input("your password:").strip()
with open(db_path, "r", encoding="utf-8") as f:
user_dic = eval(f.read())
if name in user_dic and password == user_dic[name]:
print("login ok")
login_dic['user'] = name
login_dic['status'] = True
res = func(*args, **kwargs)
return res
else:
print("login error")
elif auth_type == 'ldap':
print("ldap认证方式")
elif auth_type == "sql":
print("数据库的认证方式")
else:
print("不知道的认证方式")
return wrapper
return auth
@deco(auth_type="file")
def index():
print("welcome to index page")
@deco(auth_type="abc")
def home(name):
print("welcome to %s home page" % name)
index()
home()
在上面的例子里,deco(auth_type="file")这个函数的执行以后就得到了auth函数,但是现在得到的auth函数跟以前的auth函数是不一样的,现在的auth函数内部添加了认证方式这样一个参数,
这样添加auth_type参数后的结果就相当于加在index函数和home函数正上方的是auth函数,这样就为最开始时的auth函数添加了一个参数
先定义3个函数,现在想做的是只需要用户输入字符串,就执行对应的函数
可以通过为每个函数添加装饰器的功能,把普通函数添加到函数字典中,然后到函数字典中找到输入字符串对应的函数地址,加括号就可以运行了
类似于
func_dic = {"f1": f1, "f2": f2, "f3": f3}
def f1():
print("f1 func")
def f2():
print("f2 func")
def f3():
print("f3 func")
这里就要使用到有参装饰器
def make_func_dic(key):
def deco(func):
func_dic[key] = func
return deco
func_dic = {}
def make_func_dic(key):
def deco(func):
func_dic[key] = func
return deco
@make_func_dic("f1") # 等同于deco(f1)
def f1():
print("f1 func")
@make_func_dic("f2") # 等同于deco(f2)
def f2():
print("f2 func")
@make_func_dic("f3") # 等同于deco(f3)
def f3():
print("f3 func")
打印func_dic,得到结果
{'f1': <function f1 at 0x0000000002D2A510>, 'f2': <function f2 at 0x0000000002D2A598>, 'f3': <function f3 at 0x0000000002D2A620>}
可以看到func_dic字典中,每个字符串对应的是相应字符串的函数的内存地址
这样想运行f1函数,只需要把func_dic中的"f1"的值加括号运行就可以了
然后使用户输入字符串,运行字符串对应的函数
while True:
cmd=input(">>: ").strip()
if cmd in func_dic:
func_dic[cmd]()
这样就可以得到想要的结果了
这样做的好处:只需要输入对应的字符串,就可以执行对应的函数,不需要使用if和elif对输入的字符串进行一条一条的判断了
这就是使用装饰器在代码级别达到函数的路由功能
微信关注我们
转载内容版权归作者及来源网站所有!
低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
近一个月的开发和优化,本站点的第一个app全新上线。该app采用极致压缩,本体才4.36MB。系统里面做了大量数据访问、缓存优化。方便用户在手机上查看文章。后续会推出HarmonyOS的适配版本。
马里奥是站在游戏界顶峰的超人气多面角色。马里奥靠吃蘑菇成长,特征是大鼻子、头戴帽子、身穿背带裤,还留着胡子。与他的双胞胎兄弟路易基一起,长年担任任天堂的招牌角色。
为解决软件依赖安装时官方源访问速度慢的问题,腾讯云为一些软件搭建了缓存服务。您可以通过使用腾讯云软件源站来提升依赖包的安装速度。为了方便用户自由搭建服务架构,目前腾讯云软件源站支持公网访问和内网访问。
Sublime Text具有漂亮的用户界面和强大的功能,例如代码缩略图,Python的插件,代码段等。还可自定义键绑定,菜单和工具栏。Sublime Text 的主要功能包括:拼写检查,书签,完整的 Python API , Goto 功能,即时项目切换,多选择,多窗口等等。Sublime Text 是一个跨平台的编辑器,同时支持Windows、Linux、Mac OS X等操作系统。