python中奇怪的符号 @ __xxx__ __xxx

python中奇怪的符号

@

python中的@即代表装饰器,为了书写简单故采用了@符号。
本质上是一个带有返回函数的高阶函数(记住这几个名词,看起来很专(zhuang)业(bi))
作用:在不改变原先函数的代码的情况下扩充函数的功能
在这里插入图片描述
一个简单的例子

# test 1 注释掉@add后运行
def add(func):    # 接收一个函数作为参数,将原来函数的结果加1
    def wrapper(*args, **kw): 
        return func(*args, **kw) + 1
    return wrapper			# 返回函数wrapper


# @add   
def fun(a, b):
    return a + b


f = fun
print(f(1, 2))
print(f.__name__)

"""
结果
3
fun
"""

由于注释掉了@add,故变量 f 指向 函数fun (函数也是一种特殊的变量,所以才可以作为参数传递给另外的函数,或作为返回值返回) 结果即为 1+ 2 = 3,函数名字为fun



# test 2  加上@add后运行
def add(func):    # 接收一个函数作为参数,将原来函数的结果加1
    def wrapper(*args, **kw): 
        return func(*args, **kw) + 1
    return wrapper			# 返回函数wrapper


@add   
def fun(a, b):
    return a + b


f = fun
print(f(1, 2))
print(f.__name__)

"""
结果
4
wrapper
"""

此时 变量 f 虽然看上去指向了函数fun,但由f__name__可知 f 指向的其实是wrapper函数,即add函数的返回值 , 故此时f(1,2)为1+2+1=4



# test 3  装饰器的本质
def add(func):    # 接收一个函数作为参数,将原来函数的结果加1
    def wrapper(*args, **kw): 
        return func(*args, **kw) + 1
    return wrapper			# 返回函数wrapper



def fun(a, b):
    return a + b


f = fun
print(f(1, 2))
print(f.__name__)
f1 = add(fun)   
print(f1(1, 2))
print(f1.__name__)

"""
结果
3
fun
4
wrapper
"""

这个例子即用一种粗糙的方式实现了装饰器的功能,在这个例子中高阶函数即为add函数,返回函数即为wrapper函数


另外,装饰器改变了函数的__name__属性(虽然我觉得没什么影响),有可能会影响到依赖函数签名的代码,故提供一种使原先函数的__name__属性不改变的方法(实际上是使得返回函数的__name__属性等于原先函数的__name__属性,即 wrapper.__name__ = fun.__name__)
# test 4  完善装饰器的功能
import functools  # 导入functools模块


def add(func):
    @functools.wraps(func)  # 另一个装饰器完成wrapper.__name__ =  fun.__name__功能
    def wrapper(*args, **kw):
        return func(*args, **kw) + 1
    return wrapper


@add
def fun(a, b):
    return a + b


f = fun
print(f(1, 2))
print(f.__name__)
"""
结果
4
fun
"""
# functools.wraps 函数源码如下
def wraps(wrapped,
          assigned = WRAPPER_ASSIGNMENTS,
          updated = WRAPPER_UPDATES):
    """Decorator factory to apply update_wrapper() to a wrapper function

       Returns a decorator that invokes update_wrapper() with the decorated
       function as the wrapper argument and the arguments to wraps() as the
       remaining arguments. Default arguments are as for update_wrapper().
       This is a convenience function to simplify applying partial() to
       update_wrapper().
    """
    return partial(update_wrapper, wrapped=wrapped,
                   assigned=assigned, updated=updated)

此时__name__属性等于原先的函数__name__




验证:在异步程序中经常使用@asyncio.coroutine将生成器标记为协程,asyncio.coroutine实际上也是一个函数

# test 4  完善装饰器的功能
import functools  # 导入functools模块


async def init():   # async即为@asyncio.coroutine
    app = web.Application()
    app.router.add_route('GET', '/', index)
    runner = web.AppRunner(app)
    await runner.setup()
    site = web.TCPSite(runner, 'localhost', 9000)
    await site.start()
    logging.info('server started at http://127.0.0.1:9000...')
#asyncio中的coroutines.pyi文件	
from typing import Any, Callable, Generator, List, TypeVar

__all__: List[str]

_F = TypeVar('_F', bound=Callable[..., Any])

def coroutine(func: _F) -> _F: ...
def iscoroutinefunction(func: Callable[..., Any]) -> bool: ...
def iscoroutine(obj: Any) -> bool: ...


下划线

在这里插入图片描述

图片盗自:
python中各种下划线的含义
.

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

最佳损友1020

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值