zoukankan      html  css  js  c++  java
  • Python语法糖

    这个应该是基础吧,回头好好的补一下python基础 [TOC] ###作用:

    • python函数修饰符@的作用是为现有函数增加额外的功能,常用于插入日志、性能测试、事务处理等等。菜鸟教程
    • 用于拓展原来函数功能的一种函数,目的是在不改变原函数名(或类名)的情况下,给函数增加新的功能。 ref
    • 这个函数的特殊之处在于它的返回值也是一个函数,这个函数是内嵌“原”函数的函数
    • 在 Python 中, 装饰器 一般用来修饰函数,实现公共功能,达到代码复用的目的。 在函数定义前加上 @xxxx ,然后函数就注入了某些功能,很神奇! 然而,这只是 语法糖 而已,起决定性作用的其实是 闭包 。

    层层嵌套 ###用法:

    • 修饰符是一个函数
    • 修饰符取被修饰函数为参数
    • 修饰符返回一个新函数
    • 修饰符维护被维护函数的签名

    ###demo 这个demo是完全fork大神博客的,这里面闭包和语法糖讲的很清晰 简书 ####1.假设有不同的工作函数来完成一些任务

    def work_bar(data):
        pass
    
    def work_foo(data):
        pass
    

    如果想在函数调用前后输出日志的话怎么办呢?

    ####2.傻瓜式解法

    logging.info('begin call work_bar')
    work_bar(1)
    logging.info('call work_bar done')
    

    如果有多处代码调用呢?想想就怕! 因此函数包装安排上

    傻瓜解法无非是有太多代码冗余,每次函数调用都要写一遍 logging 。 可以把这部分冗余逻辑封装到一个新函数里:

    ####3.定义函数,缩减代码

    def smart_work_bar(data):
        logging.info('begin call: work_bar')
        work_bar(data)
        logging.info('call doen: work_bar')
    

    ####4.这样,每次调用 smart_work_bar 即可:

    smart_work_bar(1)
    # others...
    smart_work_bar(some_data)
    

    ####5.通用闭包 看上去挺完美…… 然而,当 work_foo 也有同样的需要时,还要再实现一遍 smart_work_foo 吗? 这样显然不科学呀!

    因此,我们可以用 闭包 :

    def log_call(func):
        def proxy(*args, **kwargs):
            logging.info('begin call: {name}'.format(name=func.func_name))
            result = func(*args, **kwargs)
            logging.info('call done: {name}'.format(name=func.func_name))
            return result
        return proxy
    

    这个函数接收一个函数对象(被代理函数)作为参数,返回一个代理函数。 调用代理函数时,先输出日志,然后调用被代理函数,调用完成后再输出日志,最后返回调用结果。 这样,不就达到通用化的目的了吗?——对于任意被代理函数 func , log_call 均可轻松应对。 代理函数应该就是被修饰的函数

    smart_work_bar = log_call(work_bar)
    smart_work_foo = log_call(work_foo)
    
    smart_work_bar(1)
    smart_work_foo(1)
    
    # others...
    
    smart_work_bar(some_data)
    smart_work_foo(some_data)
    

    第 1 行中, log_call 接收参数 work_bar ,返回一个代理函数 proxy ,并赋给 smart_work_bar 。 第 4 行中,调用 smart_work_bar ,也就是代理函数 proxy ,先输出日志,然后调用 func 也就是 work_bar ,最后再输出日志。 注意到,代理函数中, func 与传进去的 work_bar 对象紧紧关联在一起了,这就是 闭包 。

    再提一下,可以覆盖被代理函数名,以 smart_ 为前缀取新名字还是显得有些累赘:

    work_bar = log_call(work_bar)
    work_foo = log_call(work_foo)
    

    ####6.语法糖 先来看看以下代码:

    def work_bar(data):
        pass
    work_bar = log_call(work_bar)
    
    def work_foo(data):
        pass
    work_foo = log_call(work_foo)
    

    虽然代码没有什么冗余了,但是看是去还是不够直观。这时候,语法糖来了~~~

    @log_call
    def work_bar(data):
        pass
    

    因此,注意一点( 划重点啦 ),这里 @log_call 的作用只是: 告诉 Python 编译器插入代码 work_bar = log_call(work_bar)

    ###参考文献

    参考1

  • 相关阅读:
    UVW源码漫谈(四)
    UVW源码漫谈(三)
    UVW源码漫谈(番外篇)—— Emitter
    UVW源码漫谈(一)
    UVW源码漫谈(二)
    如何利用GDI图像来制作验证码(在winform上面制作的当然也可以在web上面制作)
    利用C语言版本的数据库制作一个学生成绩管理系统
    行内元素与块级元素的区别
    filter的两种使用方法
    JavaScript随机数类型
  • 原文地址:https://www.cnblogs.com/gaowenxingxing/p/12233381.html
Copyright © 2011-2022 走看看