以前用装饰器,都是定义好了装饰器后,使用@装饰器名的方法写入被装饰函数的正上方

在这里,定义的装饰器都是没有参数的

在定义装饰器的函数的时候,没有在括号里定义参数,这就叫做无参装饰器

既然有无参装饰器,那么当然也就会有有参装饰器

有参装饰器的定义和使用

定义一个普通的装饰器

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

用装饰器make_func_dic装饰函数

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对输入的字符串进行一条一条的判断

这就是使用装饰器在代码级别达到函数的路由功能

python函数式编程之装饰器(二)的更多相关文章

  1. Python函数式编程之装饰器

    原则:对修改是封闭的,对扩展是开放的,方法:一般不修改函数或者类,而是扩展函数或者类 一:装饰器 允许我们将一个提供核心功能的对象和其他可以改变这个功能的对象’包裹‘在一起, 使用装饰对象的任何对象与 ...

  2. python函数式编程之装饰器(一)

    1.开放封闭原则 简单来说,就是对扩展开放,对修改封闭 在面向对象的编程方式中,经常会定义各种函数. 一个函数的使用分为定义阶段和使用阶段,一个函数定义完成以后,可能会在很多位置被调用 这意味着如果函 ...

  3. Python模块化编程与装饰器

    Python的模块化编程 我们首先以一个例子来介绍模块化编程的应用场景,有这样一个名为requirements.py的python3文件,其中两个函数的作用是分别以不同的顺序来打印一个字符串: # r ...

  4. python高级编程之装饰器04

    from __future__ import with_statement # -*- coding: utf-8 -*- # python:2.x __author__ = 'Administrat ...

  5. python函数式编程之返回函数、匿名函数、装饰器、偏函数学习

    python函数式编程之返回函数 高阶函数处理可以接受函数作为参数外,还可以把函数作为结果值返回. 函数作为返回值 def laxy_sum(*args): def sum(): ax = 0; fo ...

  6. Python编程举例-装饰器

    装饰器的通常用途是扩展已定义好的函数的功能 一个浅显的装饰器编程例子 #装饰器函数 def outer(fun): def wrapper(): #添加新的功能 print('验证') fun() r ...

  7. python函数知识七 闭包、装饰器一(入门)、装饰器二(进阶)

    21.闭包 闭包:在嵌套函数内,使用非全局变量(且不使用本层变量) 闭包的作用:1.保证数据的安全性(纯洁度).2.装饰器使用 .__closure__判断是否是闭包 def func(): a = ...

  8. Python函数式编程(进阶2)

    转载请标明出处: http://www.cnblogs.com/why168888/p/6411915.html 本文出自:[Edwin博客园] Python函数式编程(进阶2) 1. python把 ...

  9. Python中利用函数装饰器实现备忘功能

    Python中利用函数装饰器实现备忘功能 这篇文章主要介绍了Python中利用函数装饰器实现备忘功能,同时还降到了利用装饰器来检查函数的递归.确保参数传递的正确,需要的朋友可以参考下   " ...

随机推荐

  1. jQuery的Nicescroll滚动条插件使用方法

    Nicescroll滚动条插件是一个非常强大的基于jQuery的滚动条插件,不需要增加额外的css,几乎全浏览器兼容.ie6+,实现只需要一段代码,侵入性非常小,样式可完全自定义,支持触摸事件,可在触 ...

  2. Java Cookie和Session

    */ .hljs { display: block; overflow-x: auto; padding: 0.5em; color: #333; background: #f8f8f8; } .hl ...

  3. HTML5 拖放(Drag 和 Drop)详解与实例

    简介 拖放是一种常见的特性,即抓取对象以后拖到另一个位置. 在 HTML5 中,拖放是标准的一部分,任何元素都能够拖放. 先点击一个小例子:在用户开始拖动 <p> 元素时执行 JavaSc ...

  4. JavaSE基础篇—MySQL三大范式—数据库设计规范

    1.概   念     范式是一种符合设计要求的总结,要想设计一个结构合理的关系型数据库,必须满足一定的范式.各个范式是以此嵌套包含的,范式越高,设计等级越高,在现实设计中也越难实现,一般数据库只要打 ...

  5. Array.prototype鲜为人知的事实

    // constructor 属性是每个具有原型的对象的原型成员. // 这包括除 Global 和 Math 对象之外的所有内部 JavaScript 对象. // constructor 属性包含 ...

  6. 动态库Link error 一种可能

    一般出现link error都是链接时找不到对应函数. 前些天出现这种错误,我反复地检查,都发现动态库里明确的是有对应函数的. 代码里有这样一句#pragma comment(lib,"li ...

  7. R语言-图形初阶

    在本节中,主要目的是如何使用R语言做出简单的图形 案例1:做出wt和mpg之间的关系 attach(mtcars) plot(wt,mpg) abline(lm(mpg~wt)) title('Reg ...

  8. 02_Python基本数据类型

    一.什么是数据 数据是描述客观事物的字符(比如95,不同的语义可表示成绩或体重),是计算机可以操作的对象,能够被计算机识别并输入给计算机处理的符号集合. 数据不仅仅包含整形,还包括图像.音乐.视频等非 ...

  9. CSS3属性详解(图文教程)

    本文最初发表于博客园,并在GitHub上持续更新前端的系列文章.欢迎在GitHub上关注我,一起入门和进阶前端. 以下是正文. 前言 我们在上一篇文章中学习了CSS3的选择器,本文来学一下CSS3的一 ...

  10. 前段篇:HTML

    <!DOCTYPE html> 文件开头统一的标准! HTML包含了两部分: head与body  固定的格式. 一.head部分: head部分分为两部分:meta标签与非meta标签: ...