python 装饰器有哪些_python装饰器有哪些作用?python特性你知道多少?

实际上,在日常的工作中,我们很多需求,无论是常见的、还是不常见的,Python 都为我们提供了一些独特的解决方案,既不需要自己造轮子,也不需要引入新的依赖(引入新的依赖势必会增加项目的复杂度)。

但是 Python 有太多功能和特性被我们忽略了,导致我们在遇到问题的时候,没法第一时间作出良好的决策。

所以,干脆来一起扫清这些被我们忽略的 Python 死角。

装饰器的妙用

我们经常会想完成一些注册&调用的功能,比如我们有四个函数:defadd(a: int, b: int) -> float:

returna + b

defsub(a: int, b: int) -> float:

returna - b

defmul(a: int, b: int) -> float:

returna * b

defdiv(a: int, b: int) -> float:

returna / b

现在我们想将这四个函数和 +、-、*、/ 四个操作符绑定,那么我们该怎么做?

可能我们第一反应是这样:operator_map= {}

defadd(a: int, b: int) -> float:

returna + b

defsub(a: int, b: int) -> float:

returna - b

defmul(a: int, b: int) -> float:

returna * b

defdiv(a: int, b: int) -> float:

returna / b

operator_map["+"] = add

operator_map["-"] = sub

operator_map["*"] = mul

operator_map["/"] = div

但这样写起来,有一个很大的问题就是太不美观了。因为直接对于dict的操作从实际上来讲可维护性是很差的,那么我们这个地方应该怎么做?

在改进这段代码之前,我们首先要明确 Python 中一个很重要的概念,即:函数/方法是:First Class Member 。用不精确的话来讲,就是函数/方法可以作为参数被传递、被使用。

举个例子:import typing

defexecute(func: typing.Callable, *args, **kwargs) -> typing.Any:

returnfunc(*args, **kwargs)

defprint_func(data: int) ->None:

print(data)

execute(print,2)

大家可以看到我们将print_func这个函数作为参数传递给execute函数并被调用。

那么我们来改造下之前的代码:importtyping

operator_map = {}

defadd(a: int, b: int) -> float:

returna + b

defsub(a: int, b: int) -> float:

returna - b

defmul(a: int, b: int) -> float:

returna * b

defdiv(a: int, b: int) -> float:

returna / b

defregister_operator(operator: str, func: typing.Callable) ->None:

operator_map[operator] = func

register_operator("+", add)

register_operator("-", sub)

register_operator("*", mul)

register_operator("/", div)

好了,大家看看,目前整体代码的可读性以及可维护性是不是改了很多?

但是我们现在的问题在于,每次都需要在单独调用一次register_operator函数,这样也太烦了吧!要不要再改进一下?要得。我们可以用装饰器来改进一下。

首先,看一个最简单的装饰器例子:importfunctools

importtyping

importtime

defexecute(func: typing.Callable) -> typing.Callable:

@functools.wraps(func)

defwraps(*args, **kwargs) -> typing.Any:

start_time = time.time()

result = func(*args, **kwargs)

print("{}".format(time.time() - start_time))

returnresult

returnwraps

@execute

defadd(a: int, b: int) -> float:

returna + b

我们能看到这段函数的意义是计算函数的执行时间。那么这个原理是什么?

实际上装饰器是一个语法糖,具体可以参见 PEP318Decorators for Functions and Methods。

简而言之,实际上是 Python 替我们做了一个替换过程。以上面的例子为例,这个替换过程就是add=execute(add) 。

好了,我们就用这个知识点来改进下之前的代码:importtyping

operator_map = {}

defregister_operator(operator: str) -> typing.Callable:

defwraps(func: typing.Callable) -> typing.Callable:

operator_map[operator] = func

returnfunc

returnwraps

@register_operator("+")

defadd(a: int, b: int) -> float:

returna + b

@register_operator("-")

defsub(a: int, b: int) -> float:

returna - b

@register_operator("*")

defmul(a: int, b: int) -> float:

returna * b

@register_operator("/")

defdiv(a: int, b: int) -> float:

returna / b

这样我们这段代码的注册过程是不是就显得更优雅了?嗯,是的!实际上 Python 中有很多特性会帮助我们的代码更简洁,更优美。接下来这个例子很可能帮我们减轻工作量。

聊聊 OrderedDict

dict是我们经常使用的一种数据解构。但是在 Python 3.6 之前dict都是无序的,即我插入的顺序,和数据在dict中存放的顺序并无关联(笔者注:Python 3.6 dict 有序只是新版实现的顺带产物,Python 3.7 正式作为 feature 被固定下来)。但是很多时候,比如在验签等场景,我们需要保证dict数据存放顺序,和我们插入顺序是一致的。那么我们该怎么办?老板有需求下来了,我们肯定不能告诉老板这个需求没法做。那我们就自己实现一个ordereddict吧。于是,想了想,写了如下的代码:

importtyping

classOrderedDict:

def__init__(self, *args, **kwargs):

self._data = {}

self._ordered_key = []

def__getitem__(self, key: typing.Any) -> typing.Any:

returnself._data[key]

def__setitem__(self, key: typing.Any, value: typing.Any) ->None:

ifkeynotinself._data:

return

self._data[key] = value

self._ordered_key.append(key)

def__delitem__(self, key: typing.Any):

delself._data[key]

self._ordered_key.remove(key)

通过额外维护一个list来维护key插入的顺序。这段代码,看似完成了我们的需求,但是实则存在很大问题。大家可以猜猜问题在哪?

3,2,1!

揭晓答案,这段代码利用list来保证key的有序性,在删除的时候,list的删除操作,是一个时间复杂度 O(n) 的操作。换句话说,我们的删除操作随着内部数据的增多,所需的删除时间也变得越长。这对于某些性能敏感的场景是无法接受的。

那要怎么办呢?事实上,Python 在很早之前就已经内置了有序字典,即很多人可能都用过的 collections.OrderedDict 。

在OrderedDict中,Python维护了一个双向链表解构,来保证插入的有序性,如下图所示:

8092f8816a5c73414a9c69240445a2b4.png

在最左侧维护一个卫兵节点,卫兵节点的next指针恒指向于数据中最后插入的节点。那么插入新的数据时,我们将新的数据插入到卫兵节点之后,从而达成维护插入顺序的目的。

在删除的时候,通过额外维护的一个字典找到待删除的key所对应的节点。这个操作是 O(1) 的复杂度,然后大家都知道,双向链表删除一个节点的时间复杂度也是 O(1) 。通过这样保证我们在即便有大量数据的情况下,也能保证相应的性能。好了,我们按照这个思路来做一个最简单的实现:

importtyping

classNode:

def__init__(self, key: typing.Any, value: typing.Any) ->None:

self.key = key

self.value = value

self.prev =None

self.next =None

classOrderedDict:

def__init__(self, *args, **kwargs):

self._data = {}

self._head =Node(None,None)

self._last = self._head

def__getitem__(self, key: typing.Any) -> typing.Any:

ifkeyinself._data:

returnself._data[key].value

raiseValueError

def__setitem__(self, key: typing.Any, value: typing.Any) ->None:

ifkeynotinself._data:

return

value_node =Node(key, value)

next_node = self._head.next

ifnotnext_node:

self._head.next = value_node

value_node.prev = self._head

self._last = value_node

else:

value_node.next = next_node

next_node.prev = value_node

value_node.prev = self._head

self._head.next = value_node

self._data[key] = value_node

def__delitem__(self, key: typing.Any):

ifkeynotinself._data:

return

value_node = self._data[key]

ifvalue_node == self._last:

self._last = value_node.prev

self._last.next =None

else:

prev_node = value_node.prev

next_node = value_node.next

prev_node.next = next_node

next_node.prev = prev_node

delself._data[key]

delvalue_node

(此段代码,如有错乱,烦请将浏览字体调小几号)

这只是一个OrderedDict 的简化版,如果想完成一个完整的 OrderedDict 还有很多很多的 corner case 要去处理。不过现在,我们可以使用内置的数据结构去完成我们需求。怎么样,是不是有了一种幸福的感觉?

python培训:http://www.baizhiedu.com/python2019

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值