Python 3.7新功能之dataclass裝飾器詳解
前言
Python 3.7 將于今年夏天發(fā)布,Python 3.7 中將會(huì)有許多新東西:
- 各種字符集的改進(jìn)
- 對(duì)注釋的推遲評(píng)估
- 以及對(duì)dataclass的支持
最激動(dòng)人心的新功能之一是 dataclass 裝飾器。
什么是 Data Class
大多數(shù) Python 開發(fā)人員編寫過(guò)很多像下面這樣的類:
class MyClass: def __init__(self, var_a, var_b): self.var_a = var_a self.var_b = var_b
dataclass 可以為簡(jiǎn)單的情況自動(dòng)生成方法,例如,一個(gè)__init__接受這些參數(shù)并將其分配給自己,之前的小例子可以重寫為:
@dataclass class MyClass: var_a: str var_b: str
那么通過(guò)一個(gè)例子來(lái)看看如何使用吧
星球大戰(zhàn) API
可以使用 requests 從星球大戰(zhàn) API 獲取資源:
response = requests.get('https://swapi.co/api/films/1/') dictionary = response.json()
讓我們來(lái)看看 dictionary (簡(jiǎn)化過(guò))的結(jié)果:
{ 'characters': ['https://swapi.co/api/people/1/',… ], 'created': '2014-12-10T14:23:31.880000Z', 'director': 'George Lucas', 'edited': '2015-04-11T09:46:52.774897Z', 'episode_id': 4, 'opening_crawl': 'It is a period of civil war.\r\n … ', 'planets': ['https://swapi.co/api/planets/2/', … ], 'producer': 'Gary Kurtz, Rick McCallum', 'release_date': '1977-05-25', 'species': ['https://swapi.co/api/species/5/',…], 'starships': ['https://swapi.co/api/starships/2/',…], 'title': 'A New Hope', 'url': 'https://swapi.co/api/films/1/', 'vehicles': ['https://swapi.co/api/vehicles/4/',…]
封裝 API
為了正確地封裝一個(gè) API,我們應(yīng)該創(chuàng)建一個(gè)用戶可以在其應(yīng)用程序中使用的對(duì)象,因此,在Python 3.6 中定義一個(gè)對(duì)象來(lái)包含requests對(duì) /films/endpoint的響應(yīng):
class StarWarsMovie: def __init__(self, title: str, episode_id: int, opening_crawl: str, director: str, producer: str, release_date: datetime, characters: List[str], planets: List[str], starships: List[str], vehicles: List[str], species: List[str], created: datetime, edited: datetime, url: str ): self.title = title self.episode_id = episode_id self.opening_crawl= opening_crawl self.director = director self.producer = producer self.release_date = release_date self.characters = characters self.planets = planets self.starships = starships self.vehicles = vehicles self.species = species self.created = created self.edited = edited self.url = url if type(self.release_date) is str: self.release_date = dateutil.parser.parse(self.release_date) if type(self.created) is str: self.created = dateutil.parser.parse(self.created) if type(self.edited) is str: self.edited = dateutil.parser.parse(self.edited)
仔細(xì)的讀者可能已經(jīng)注意到這里有一些重復(fù)的代碼。
這是使用 dataclass 裝飾器的經(jīng)典案例,我們需要?jiǎng)?chuàng)建一個(gè)主要用來(lái)保存數(shù)據(jù)的類,只需一點(diǎn)驗(yàn)證,所以讓我們來(lái)看看我們需要修改什么。
首先,data class 自動(dòng)生成一些 dunder 方法,如果我們沒(méi)有為 data class 裝飾器指定任何選項(xiàng),則生成的方法有:__init__,__eq__和__repr__,如果你已經(jīng)定義了__repr__但沒(méi)定義__str__,默認(rèn)情況下 Python(不僅僅是 data class)將實(shí)現(xiàn)返回__repr__的輸出__str__方法。因此,只需將代碼更改為以下代碼即可實(shí)現(xiàn)四種 dunder 方法:
@dataclass class StarWarsMovie: title: str episode_id: int opening_crawl: str director: str producer: str release_date: datetime characters: List[str] planets: List[str] starships: List[str] vehicles: List[str] species: List[str] created: datetime edited: datetime url: str
我們?nèi)サ袅薩_init__方法,以確保 data class 裝飾器可以添加它生成的對(duì)應(yīng)方法。不過(guò),我們?cè)谶@個(gè)過(guò)程中失去了一些功能,我們的 Python 3.6 構(gòu)造函數(shù)不僅定義了所有的值,還試圖解析日期,我們?cè)鯓硬拍苡?data class 來(lái)做到這一點(diǎn)呢?
如果要覆蓋 __init__,我們將失去 data class 的優(yōu)勢(shì),因此,如果要處理任何附加功能可以使用新的 dunder 方法:__post_init__,讓我們看看__post_init__方法對(duì)于我們的包裝類來(lái)說(shuō)是什么樣子的:
def __post_init__(self): if type(self.release_date) is str: self.release_date = dateutil.parser.parse(self.release_date) if type(self.created) is str: self.created = dateutil.parser.parse(self.created) if type(self.edited) is str: self.edited = dateutil.parser.parse(self.edited)
就是這樣! 我們可以使用 data class 裝飾器在用三分之二的代碼量實(shí)現(xiàn)我們的類。
更多好東西
通過(guò)使用裝飾器的選項(xiàng),可以為用例進(jìn)一步定制 data class,默認(rèn)選項(xiàng)是:
@dataclass(init=True, repr=True, eq=True, order=False, unsafe_hash=False, frozen=False)
- init決定是否生成__init__ dunder 方法
- repr決定是否生成__repr__ dunder方法
- eq對(duì)__eq__ dunder 方法也是如此,它決定相等性檢查的行為(your_class_instance == another_instance)
- order 實(shí)際上創(chuàng)建了四種 dunder 方法,它們確定所有檢查小于,and/or,大于的行為,如果將其設(shè)置為 true,則可以對(duì)對(duì)象列表進(jìn)行排序。
最后兩個(gè)選項(xiàng)確定對(duì)象是否可以被哈?;?,如果你想使用你的 class 的對(duì)象作為字典鍵的話,這是必要的。
更多信息請(qǐng)參考:PEP 557 -- Data Classes
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)腳本之家的支持。
相關(guān)文章
Django單元測(cè)試工具test client使用詳解
這篇文章主要介紹了Django單元測(cè)試工具test client使用詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-08-08基于python實(shí)現(xiàn)的抓取騰訊視頻所有電影的爬蟲
這篇文章主要介紹了用python實(shí)現(xiàn)的抓取騰訊視頻所有電影的爬蟲,這個(gè)程序使用芒果存, 所以大家需要下載使用mongodb才可以2016-04-04Python代碼實(shí)現(xiàn)動(dòng)圖倒放
這篇文章主要介紹了Python代碼實(shí)現(xiàn)動(dòng)圖倒放,文章通過(guò)利用gif動(dòng)圖實(shí)現(xiàn)倒放效果,具有一定的參考價(jià)值,需要的小伙伴可以參考一下,希望對(duì)你的學(xué)習(xí)有所幫助2022-03-03pandas實(shí)現(xiàn)按照Series分組示例
本文主要介紹了pandas按照Series分組示例,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-08-08Python queue隊(duì)列原理與應(yīng)用案例分析
這篇文章主要介紹了Python queue隊(duì)列原理與應(yīng)用,結(jié)合具體案例形式分析了Python queue隊(duì)列的原理、功能、實(shí)現(xiàn)方法與使用技巧,需要的朋友可以參考下2019-09-09Python在信息學(xué)競(jìng)賽中的運(yùn)用及Python的基本用法(詳解)
下面小編就為大家?guī)?lái)一篇Python在信息學(xué)競(jìng)賽中的運(yùn)用及Python的基本用法(詳解)。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-08-08淺析Python自帶性能強(qiáng)悍的標(biāo)準(zhǔn)庫(kù)itertools
itertools是python內(nèi)置的模塊,使用簡(jiǎn)單且功能強(qiáng)大。這篇文章就主要介紹了通過(guò)itertools實(shí)現(xiàn)可迭代對(duì)象的無(wú)限迭代、有限迭代和排列組合。感興趣的同學(xué)可以關(guān)注一下2021-12-12Python使用random模塊實(shí)現(xiàn)擲骰子游戲的示例代碼
這篇文章主要介紹了Python使用random模塊實(shí)現(xiàn)擲骰子游戲的示例代碼,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-04-04