zoukankan      html  css  js  c++  java
  • Python中使用@的理解

    Python函数中使用@

    稍提一下的基础

    funfun()的区别

    以一段代码为例:

    def fun():
        print('fun')
        return None
    
    a = fun() #fun函数并将返回值给a
    print('a的值为',a)
    b = fun #将fun函数地址赋给b
    b() #调用b,b和fun指向的地址相同
    print('b的值为',b)
    '''输出
    fun
    a的值为 None
    fun
    b的值为 <function fun at 0x00000248E1EBE0D0>
    '''
    

    根据输出可以看出,a=fun()是将函数fun的返回值(None)赋给a,而b=fun是将函数的地址赋给b,如果调用函数,需要b()

    函数返回值和函数地址

    类似的,其他内置函数也可以通过这种方法,相当于起了一个同名的函数

    >>>a = abs
    >>>a(-1)
    1
    

    除此之外,原来的函数名也被覆盖为其他函数,例如

    def fun():
        print('fun')
    abs = fun
    abs() #输出fun
    
    

    综上,可以理解为fun,abs在不带括号时为变量,该变量包含了函数在内容的地址

    函数名指向的地址被替换

    返回函数

    参考链接:https://www.liaoxuefeng.com/wiki/1016959663602400/1017434209254976

    以廖老师的教程为例

    def lazy_sum(*args):
        def sum():
            ax = 0
            for n in args:
                ax = ax + n
            return ax
        return sum
    >>> f = lazy_sum(1, 3, 5, 7, 9)
    >>> f
    <function lazy_sum.<locals>.sum at 0x101c6ed90>
    >>>f()
    25
    

    返回函数

    在单步调试中可以发现,当读到def sum():时,解释器会直接跳到return sumsum函数的地址返回给f,因此f()即为执行sum() (不是非常准确,但先如此理解)

    如果对返回函数还是有些不理解的话,可以假设lazy_sum()的返回值改为1

    def lazy_sum(*args):
        def sum():
            ax = 0
            for n in args:
                ax = ax + n
            return ax
        return 1
    
    f = lazy_sum(1,3,5,7,9)
    print(f)#Q输出1
    
    print(f())#报错'int' object is not callable
    
    

    此时无论lazy_sum()的参数如何修改,都会把1赋给f,而1()是不可调用的,因此会报错

    ⭐️返回函数中的闭包问题也要了解一下,内嵌函数可以访问外层函数的变量

    参数的嵌套调用

    仍然上述例子,此时将lazy_sum()改为空函数,内嵌的sum()需要参数:

    def lazy_sum():
        def sum(*args):
            ax = 0
            for n in args:
                ax = ax + n
            return ax
        return sum
    
    f = lazy_sum()(1,3,5,7,9)
    print(f)#输出25
    

    按照运算的优先级,可以理解为:

    1. 执行 lazy_sum(),返回sum;
    2. 执行sum(1,3,5,7,9),返回25;
    3. 25赋给f

    如果有了以上基础,再来看@的用法就会觉得很容易了

    @的使用

    如果需要具体理解装饰器,可以参考廖老师的博客,本文仅介绍@的执行流程

    本文参考了 Python @函数装饰器及用法(超级详细)Python中的注解“@”

    不带参数的单一使用(一个@修饰)

    def spamrun(fn):
        def sayspam():
            print("spam,spam,spam")
            fn()
        return sayspam
    @spamrun
    def useful():
        print('useful')
       
    useful() 
    '''
    输出:
    spam,spam,spam
    useful
    '''
    

    修饰效果相当于useful = spamrun(useful),具体步骤如下:

    1. 在初始化时,解释器读到@spamrun,此时将下方的useful作为参数传入到spamrun
    2. spamrun(useful)中,由于是返回函数,直接将sayspam()的内存地址赋给useful
    3. 执行useful(),此时useful指向了sayspam,因此打印spam,spam,spam。然后执行fn(),此时的fn才指向原来useful()的地址,开始执行print('useful')

    执行流程可以在下图了解一下,可以理解为经过@后,useful已经不直接指向函数useful()的地址了,而是sayspam。再调用useful()时,执行sayspam(),由于fn保存原先useful()函数的地址,因此可以执行useful()的功能,即可以打印出'useful'。如果‘使坏’把fn()去掉的话,相当于useful()再也不会执行了

    @的原理

    一般情况下,使用@时不会改变函数原先的执行逻辑,而只是增加功能,因此成为装饰器,如廖老师教程中可以使原函数打印日志

    def log(func):
        def wrapper(*args, **kw):
            print('call %s():' % func.__name__)
            return func(*args, **kw)
        return wrapper
    @log
    def now():
        print('2015-3-25')
    
    now()
    '''
    call now():
    2015-3-25
    '''
    

    不带参数的多次使用(两个@)

    def spamrun(fn):
        def sayspam():
            print("spam,spam,spam")
            fn()
        return sayspam
    
    
    def spamrun1(fn):
        def sayspam1():
            print("spam1,spam1,spam1")
            fn()
        return sayspam1
            
    @spamrun
    @spamrun1
    def useful():
        print('useful')
    
    useful()
    '''
    spam,spam,spam
    spam1,spam1,spam1
    useful
    '''
    

    修饰效果相当于useful = spamrun(spamrun1(useful))

    叠加使用时,装饰器的调用顺序和声明顺序是相反的,可以理解成是一个递归的过程。

    1. 遇到@spamrun,开始向下寻找def 函数名
    2. 结果第二行也是一个@。@spamrun1继续向下找
    3. 遇到了def useful,执行useful = spamrun1(useful)
    4. 回归。@spamrun1返回useful@spamrun,执行useful=spamrun(useful)

    带参数的单次使用

    以廖老师教程中的举例,简化一些,先不考虑*args,**kw,因为涉及到返回函数的闭包问题

    def log(text):
        def decorator(func):
            def wrapper():
                print('%s %s():' % (text, func.__name__))
                return func()
            return wrapper
        return decorator
    
    @log('execute')
    def now():
        print('2015-3-25')
    
    now()
    

    修饰效果相当于now=log('execute')(now)

    ​ 1. 解释器读到@log('execute'),先执行了log('execute'),返回函数decorator

    ​ 2. 将now作为decorator(func)的形参,返回warpper

        3. 将`warpper`的内存地址赋给变量`now`
    

    此时调用now(),先执行完print(...),然后return func()。注意此处是带括号的,因此执行了真正的now函数,最终return的为None

    带参数的多次调用可以将之前的情况组合即可

    总结

    1. @行不带参数
    @XXX
    def funXXX():
    

    会被解释成funXXX = XXX(funXXX)

    1. 如果@那行中带参数,则被解释成funXXX = XXX(@行的参数)(funXXX)
    2. 要深刻理解返回函数以及funfun()的区别
    3. 函数的内存地址,函数变量,函数的名称的区别。默认情况下,函数变量指向函数的内存地址,但也可以被改变

    初学Python,学识短浅,希望多多交流

  • 相关阅读:
    Python数值计算:一 使用Pylab绘图(2)
    Python数值计算:一 使用Pylab绘图(1)
    Gromacs处理amino acid residues小结
    实用正则表达式匹配和替换
    将Word转为带书签的PDF
    QT在windows下实现截屏操作并保存为png图片
    VMware虚拟机无法识别U盘解决方案
    C/C++/Qt 统计运行时间
    ffmpeg 中 swscale 的用法
    用qt代码怎样编写图片保存格式[qt4.6]
  • 原文地址:https://www.cnblogs.com/TianYuanSX/p/11170360.html
Copyright © 2011-2022 走看看