Python庫(kù)functools示例詳解
functools模塊是Python的標(biāo)準(zhǔn)庫(kù)的一部分,它是為高階函數(shù)而實(shí)現(xiàn)的。高階函數(shù)是作用于或返回另一個(gè)函數(shù)或多個(gè)函數(shù)的函數(shù)。一般來(lái)說(shuō),對(duì)這個(gè)模塊而言,任何可調(diào)用的對(duì)象都可以作為一個(gè)函數(shù)來(lái)處理。
functools 提供了 11個(gè)函數(shù):
1. cached_property
將類的方法轉(zhuǎn)換為一個(gè)屬性,該屬性的值計(jì)算一次,然后在實(shí)例的生命周期中將其緩存作為普通屬性。與 property() 類似,但添加了緩存,對(duì)于在其他情況下實(shí)際不可變的高計(jì)算資源消耗的實(shí)例特征屬性來(lái)說(shuō)該函數(shù)非常有用。
# cached_property 緩存屬性
class cached_property(object):
"""
Decorator that converts a method with a single self argument into a
property cached on the instance.
Optional ``name`` argument allows you to make cached properties of other
methods. (e.g. url = cached_property(get_absolute_url, name='url') )
"""
def __init__(self, func, name=None):
# print(f'f: {id(func)}')
self.func = func
self.__doc__ = getattr(func, '__doc__')
self.name = name or func.__name__
def __get__(self, instance, type=None):
# print(f'self func: {id(self.func)}')
# print(f'instance: {id(instance)}')
if instance is None:
return self
res = instance.__dict__[self.name] = self.func(instance)
return res
class F00():
@cached_property
def test(self):
# cached_property 將會(huì)把每個(gè)實(shí)例的屬性存儲(chǔ)到實(shí)例的__dict__中, 實(shí)例獲取屬性時(shí), 將會(huì)優(yōu)先從__dict__中獲取,則不會(huì)再次調(diào)用方法內(nèi)部的過(guò)程
print(f'運(yùn)行test方法內(nèi)部過(guò)程')
return 3
@property
def t(self):
print('運(yùn)行t方法內(nèi)部過(guò)程')
return 44
f = F00()
print(f.test) # 第一次將會(huì)調(diào)用test方法內(nèi)部過(guò)程
print(f.test) # 再次調(diào)用將直接從實(shí)例中的__dict__中直接獲取,不會(huì)再次調(diào)用方法內(nèi)部過(guò)程
print(f.t) # 調(diào)用方法內(nèi)部過(guò)程取值
print(f.t) # 調(diào)用方法內(nèi)部過(guò)程取值
# 結(jié)果輸出
# 運(yùn)行test方法內(nèi)部過(guò)程
# 3
# 3
# 運(yùn)行t方法內(nèi)部過(guò)程
# 44
# 運(yùn)行t方法內(nèi)部過(guò)程
# 44
2. cmp_to_key
在 list.sort 和 內(nèi)建函數(shù) sorted 中都有一個(gè) key 參數(shù)
x = ['hello','worl','ni'] x.sort(key=len) print(x) # ['ni', 'worl', 'hello']
3. lru_cache
允許我們將一個(gè)函數(shù)的返回值快速地緩存或取消緩存。
該裝飾器用于緩存函數(shù)的調(diào)用結(jié)果,對(duì)于需要多次調(diào)用的函數(shù),而且每次調(diào)用參數(shù)都相同,則可以用該裝飾器緩存調(diào)用結(jié)果,從而加快程序運(yùn)行。
該裝飾器會(huì)將不同的調(diào)用結(jié)果緩存在內(nèi)存中,因此需要注意內(nèi)存占用問(wèn)題。
from functools import lru_cache
@lru_cache(maxsize=30) # maxsize參數(shù)告訴lru_cache緩存最近多少個(gè)返回值
def fib(n):
if n < 2:
return n
return fib(n-1) + fib(n-2)
print([fib(n) for n in range(10)])
fib.cache_clear() # 清空緩存4. partial
用于創(chuàng)建一個(gè)偏函數(shù),將默認(rèn)參數(shù)包裝一個(gè)可調(diào)用對(duì)象,返回結(jié)果也是可調(diào)用對(duì)象。
偏函數(shù)可以固定住原函數(shù)的部分參數(shù),從而在調(diào)用時(shí)更簡(jiǎn)單。
from functools import partial
int2 = partial(int, base=8)
print(int2('123'))
# 835. partialmethod
對(duì)于python 偏函數(shù)partial理解運(yùn)用起來(lái)比較簡(jiǎn)單,就是對(duì)原函數(shù)某些參數(shù)設(shè)置默認(rèn)值,生成一個(gè)新函數(shù)。而如果對(duì)于類方法,因?yàn)榈谝粋€(gè)參數(shù)是 self,使用 partial 就會(huì)報(bào)錯(cuò)了。
class functools.partialmethod(func, /, *args, **keywords) 返回一個(gè)新的 partialmethod 描述器,其行為類似 partial 但它被設(shè)計(jì)用作方法定義而非直接用作可調(diào)用對(duì)象。
func 必須是一個(gè) descriptor 或可調(diào)用對(duì)象(同屬兩者的對(duì)象例如普通函數(shù)會(huì)被當(dāng)作描述器來(lái)處理)。
當(dāng) func 是一個(gè)描述器(例如普通 Python 函數(shù), classmethod(), staticmethod(), abstractmethod() 或其他 partialmethod 的實(shí)例)時(shí), 對(duì) __get__ 的調(diào)用會(huì)被委托給底層的描述器,并會(huì)返回一個(gè)適當(dāng)?shù)?部分對(duì)象 作為結(jié)果。
當(dāng) func 是一個(gè)非描述器類可調(diào)用對(duì)象時(shí),則會(huì)動(dòng)態(tài)創(chuàng)建一個(gè)適當(dāng)?shù)慕壎ǚ椒ā?當(dāng)用作方法時(shí)其行為類似普通 Python 函數(shù):將會(huì)插入 self 參數(shù)作為第一個(gè)位置參數(shù),其位置甚至?xí)幱谔峁┙o partialmethod 構(gòu)造器的 args 和 keywords 之前。
from functools import partialmethod
class Cell:
def __init__(self):
self._alive = False
@property
def alive(self):
return self._alive
def set_state(self, state):
self._alive = bool(state)
set_alive = partialmethod(set_state, True)
set_dead = partialmethod(set_state, False)
print(type(partialmethod(set_state, False)))
# <class 'functools.partialmethod'>
c = Cell()
c.alive
# False
c.set_alive()
c.alive
# True6. reduce
函數(shù)的作用是將一個(gè)序列歸納為一個(gè)輸出reduce(function, sequence, startValue)
from functools import reduce l = range(1,50) print(reduce(lambda x,y:x+y, l)) # 1225
7. singledispatch
單分發(fā)器,用于實(shí)現(xiàn)泛型函數(shù)。根據(jù)單一參數(shù)的類型來(lái)判斷調(diào)用哪個(gè)函數(shù)。
from functools import singledispatch
@singledispatch
def fun(text):
print('String:' + text)
@fun.register(int)
def _(text):
print(text)
@fun.register(list)
def _(text):
for k, v in enumerate(text):
print(k, v)
@fun.register(float)
@fun.register(tuple)
def _(text):
print('float, tuple')
fun('i am is hubo')
fun(123)
fun(['a','b','c'])
fun(1.23)
print(fun.registry) # 所有的泛型函數(shù)
print(fun.registry[int]) # 獲取int的泛型函數(shù)
# String:i am is hubo
# 123
# 0 a
# 1 b
# 2 c
# float, tuple
# {<class 'object'>: <function fun at 0x106d10f28>, <class 'int'>: <function _ at 0x106f0b9d8>, <class 'list'>: <function _ at 0x106f0ba60>, <class 'tuple'>: <function _ at 0x106f0bb70>, <class 'float'>: <function _ at 0x106f0bb70>}
# <function _ at 0x106f0b9d8>
8. singledispatchmethod
與泛型函數(shù)類似,可以編寫一個(gè)使用不同類型的參數(shù)調(diào)用的泛型方法聲明,根據(jù)傳遞給通用方法的參數(shù)的類型,編譯器會(huì)適當(dāng)?shù)靥幚砻總€(gè)方法調(diào)用。
class Negator:
@singledispatchmethod
def neg(self, arg):
raise NotImplementedError("Cannot negate a")
@neg.register
def _(self, arg: int):
return -arg
@neg.register
def _(self, arg: bool):
return not arg9. total_ordering
它是針對(duì)某個(gè)類如果定義了lt、le、gt、ge這些方法中的至少一個(gè),使用該裝飾器,則會(huì)自動(dòng)的把其他幾個(gè)比較函數(shù)也實(shí)現(xiàn)在該類中
from functools import total_ordering
class Person:
# 定義相等的比較函數(shù)
def __eq__(self,other):
return ((self.lastname.lower(),self.firstname.lower()) ==
(other.lastname.lower(),other.firstname.lower()))
# 定義小于的比較函數(shù)
def __lt__(self,other):
return ((self.lastname.lower(),self.firstname.lower()) <
(other.lastname.lower(),other.firstname.lower()))
p1 = Person()
p2 = Person()
p1.lastname = "123"
p1.firstname = "000"
p2.lastname = "1231"
p2.firstname = "000"
print p1 < p2 # True
print p1 <= p2 # True
print p1 == p2 # False
print p1 > p2 # False
print p1 >= p2 # False10. update_wrapper
使用 partial 包裝的函數(shù)是沒(méi)有__name__和__doc__屬性的。update_wrapper 作用:將被包裝函數(shù)的__name__等屬性,拷貝到新的函數(shù)中去。
from functools import update_wrapper
def wrap2(func):
def inner(*args):
return func(*args)
return update_wrapper(inner, func)
@wrap2
def demo():
print('hello world')
print(demo.__name__)
# demo11. wraps
warps 函數(shù)是為了在裝飾器拷貝被裝飾函數(shù)的__name__。
就是在update_wrapper上進(jìn)行一個(gè)包裝
from functools import wraps
def wrap1(func):
@wraps(func) # 去掉就會(huì)返回inner
def inner(*args):
print(func.__name__)
return func(*args)
return inner
@wrap1
def demo():
print('hello world')
print(demo.__name__)
# demo參考文獻(xiàn)
Python的Functools模塊簡(jiǎn)介_函數(shù)
到此這篇關(guān)于Python庫(kù)functools詳解的文章就介紹到這了,更多相關(guān)Python庫(kù)functools內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- Python functools模塊學(xué)習(xí)總結(jié)
- Python中functools模塊的常用函數(shù)解析
- Python中functools模塊函數(shù)解析
- Python使用functools實(shí)現(xiàn)注解同步方法
- Python3標(biāo)準(zhǔn)庫(kù)之functools管理函數(shù)的工具詳解
- Python編程functools模塊創(chuàng)建修改的高階函數(shù)解析
- Python的functools模塊使用及說(shuō)明
- Python中的functools partial詳解
- python高階函數(shù)functools模塊的具體使用
- Python中Functools模塊的高級(jí)操作詳解
- Python函數(shù)式編程模塊functools的使用與實(shí)踐
相關(guān)文章
python檢測(cè)空間儲(chǔ)存剩余大小和指定文件夾內(nèi)存占用的實(shí)例
今天小編就為大家分享一篇python檢測(cè)空間儲(chǔ)存剩余大小和指定文件夾內(nèi)存占用的實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-06-06
Python generator生成器和yield表達(dá)式詳解
這篇文章主要介紹了Python generator生成器和yield表達(dá)式詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-08-08
Python下利用BeautifulSoup解析HTML的實(shí)現(xiàn)
這篇文章主要介紹了Python下利用BeautifulSoup解析HTML的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-01-01
python實(shí)現(xiàn)簡(jiǎn)單圖書管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了python實(shí)現(xiàn)簡(jiǎn)單圖書管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-11-11
python中判斷類型函數(shù)isinstance()示例詳解
isinstance()函數(shù)是Python的內(nèi)置函數(shù),用于判斷一個(gè)變量是否是某個(gè)類型或者是該類型的子類的實(shí)例,在Python中,所有類都繼承自object,所以任何實(shí)例都會(huì)是object的實(shí)例,本文給大家介紹python中判斷類型函數(shù)isinstance(),感興趣的朋友一起看看吧2024-10-10
Python更新數(shù)據(jù)庫(kù)腳本兩種方法及對(duì)比介紹
這篇文章給大家介紹了Python更新數(shù)據(jù)庫(kù)腳本兩種方法及數(shù)據(jù)庫(kù)查詢?nèi)N方式,然后在文章下面給大家介紹了兩種方式對(duì)比介紹,非常不錯(cuò),感興趣的朋友參考下吧2017-07-07
Python類和對(duì)象的定義與實(shí)際應(yīng)用案例分析
這篇文章主要介紹了Python類和對(duì)象的定義與實(shí)際應(yīng)用,結(jié)合三個(gè)具體案例形式分析了Python面向?qū)ο蟪绦蛟O(shè)計(jì)中類與對(duì)象的定義、應(yīng)用、設(shè)計(jì)模式等相關(guān)操作技巧,需要的朋友可以參考下2018-12-12
Django框架之登錄后自定義跳轉(zhuǎn)頁(yè)面的實(shí)現(xiàn)方法
這篇文章主要介紹了Django框架之登錄后自定義跳轉(zhuǎn)頁(yè)面的實(shí)現(xiàn)方法,非常不錯(cuò),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2019-07-07

