一、装饰器

装饰器:本质就是函数,功能是为其它函数添加附加功能

装饰器的原则:

  • 不修改被修饰函数的源代码
  • 不修改被修饰函数的调用方式

装饰器的知识储备:

  装饰器 = 高阶函数 + 函数嵌套 + 闭包

二、高阶函数

高阶函数的定义:

  • 函数的接收参数是一个函数名
  • 函数的返回值是一个函数名
  • 满足上述条件任意一个都可以是高阶函数
import time

def fun1():
    time.sleep(0.5)
    print("hello")

def computing_run_time(fun):
    """
    计算函数运行时间
    :param fun:
    :return:
    """
    start_time = time.time()
    fun()
    end_time = time.time()
    print("运行时间%s" % (end_time - start_time))

computing_run_time(fun1)

"""
优点:在不修改函数源代码的前提下,给函数添加了额外的功能
缺点:改变了调用方式
"""

函数的接收参数是函数名

import time

def fun1():
    time.sleep(0.5)
    print("hello")

def computing_run_time(fun):
    """
    计算函数运行时间
    :param fun:
    :return:
    """
    fun()
    return fun

fun1 = computing_run_time(fun1)
fun1()   

"""
优点:没有改变函数的调用方式
缺点:不能为函数添加新的功能
"""

函数的返回值是函数名

注:仅仅是高阶函数不能满足装饰器的需求

三、函数嵌套和闭包

"""
闭包:首先必须是内部定义的函数,该函数包含对外部作用域而不是全局作用域名字的引用

定义:内部函数的代码包含对外部函数的代码的引用,但一定不是对全局作用域的引用
"""

def fun1():

    print("fun1")
    name = 1

    def fun2():
        print("fun2")
        print(name)

        def fun3():
            print("fun3")
            print(name)
        fun3()
    fun2()

fun1()

  

四、装饰器示例

无参装饰器

def outer(fun):

    def wrapper():
        fun()

    return wrapper

加上参数

def outer(fun):

    def wrapper(*args, **kwargs):
        fun(*args, **kwargs)

    return wrapper

  

简单装饰器

import time

def outer(func):
    """
    计算程序的运行时间
    :param func:
    :return:
    """

    def wrapper(*args, **kwargs):
        start_time = time.time()
        result = func(*args, **kwargs)
        stop_time = time.time()
        print("函数运行的时间为:%s" % (stop_time-start_time))
        return result
    return wrapper

@outer   # 语法糖  等价于 : cal = outer(cal)  把wrapper函数的函数地址赋值给cal
def cal(list1):
    result = 0
    for i in list1:
        time.sleep(0.1)
        result += i

    return result

res = cal(range(20))
print(res)

  

注:

  函数cal的执行流程,先执行装饰器的outer函数,outer将函数的返回值赋值给变量cal,所以调用cal()  等价于  cal = outer(cal),   cal()  等价于这两步

多层装饰器

装饰器中含有函数参数

def before(name):
    print("before:%s" % name)
    return "before"

def after(name):
    print("after:%s" % name)
    return "after"

# 外层的参数只是为了将值传递到里层
def outer(before_func, after_func):

    def middle(main_func):

        def wrapper(*args, **kwargs):

            before_result = before_func(*args, **kwargs)
            print("before_result:%s" % before_result)

            main_result = main_func(*args, **kwargs)
            print("main_result: %s" % main_result)

            after_result = after_func(*args, **kwargs)
            print("after_result: %s" % after_result)

            return main_result

        return wrapper

    return middle

@outer(before, after)
def index(name):
    print("index")
    return name

result = index("aaa")
print(result)

  

最新文章

  1. AC日记——忠诚 洛谷 P1816
  2. cocos2dx UI总结
  3. IE8和W3C标准下IFRAME刷新和URL的区别
  4. 济南学习D2T2__数学分析题
  5. SMON功能(一):清理临时段
  6. MATLAB学习笔记(十)——MATLAB图形句柄
  7. Django 1.6 最佳实践: 如何设置和使用 Log(转)
  8. Solr配置与简单Demo
  9. Counting Lines, Words, and Characters with wc
  10. 4.1. 如何在Windows环境下开发Python
  11. 解决VS2015中没有报表项(ReportViewer)的方法
  12. 深入理解JSP
  13. February 12th, 2018 Week 7th Monday
  14. BZOJ.5249.[九省联考2018]iiidx(贪心 线段树)
  15. Redis入门到高可用(二十一)——缓存的使用和设计
  16. CSS--点击改变样式
  17. Using the G711 standard
  18. win7 远程桌面连接centos 6.5
  19. webstorm 重置所有设置
  20. IDEA/Git 设置多个push远程仓库或者同时提交多个push仓库

热门文章

  1. C# 多线程控制 通讯
  2. DevExpress 15.1.sln
  3. ssh2学习-applicationContext.xml文件配置-----<context:annotation-config/>详解
  4. Server 非阻塞
  5. iOS 核心动画 Core Animation浅谈
  6. codeforces #550D Regular Bridge 构造
  7. java之super关键字
  8. nginx频率限制
  9. windows lua 多线程 线程同步
  10. MaterialUp 官方client源代码