麻豆黑色丝袜jk制服福利网站-麻豆精品传媒视频观看-麻豆精品传媒一二三区在线视频-麻豆精选传媒4区2021-在线视频99-在线视频a

千鋒教育-做有情懷、有良心、有品質的職業教育機構

手機站
千鋒教育

千鋒學習站 | 隨時隨地免費學

千鋒教育

掃一掃進入千鋒手機站

領取全套視頻
千鋒教育

關注千鋒學習站小程序
隨時隨地免費學習課程

當前位置:首頁  >  技術干貨  > Python之__init__和__new__的區別是什么

Python之__init__和__new__的區別是什么

來源:千鋒教育
發布人:xqq
時間: 2023-11-07 13:33:58 1699335238

真假構造函數

如果你去面試Python工程師的崗位,面試官問你,請問Python當中的類的構造函數是什么?

你不假思索,當然是__init__啦!如果你這么回答,很有可能你就和offer無緣了。因為在Python當中__init__并不是構造函數,__new__才是。是不是有點蒙,多西得(日語:為什么)?我們不是一直將__init__方法當做構造函數來用的嗎?怎么又冒出來一個__new__,如果__new__才是構造函數,那么為什么我們創建類的時候從來不用它呢?

別著急,我們慢慢來看。首先我們回顧一下__init__的用法,我們隨便寫一段代碼:

classStudent:

def__init__(self,name,gender):

self.name=name

self.gender=gender

我們一直都是這么用的,對不對,毫無問題。但是我們換一個問題,我們在Python當中怎么實現單例(Singleton)的設計模式呢?怎么樣實現工廠呢?

從這個問題出發,你會發現只使用__init__函數是不可能完成的,因為__init__并不是構造函數,它只是初始化方法。也就是說在調用__init__之前,我們的實例就已經被創建好了,__init__只是為這個實例賦上了一些值。如果我們把創建實例的過程比喻成做一個蛋糕,__init__方法并不是烘焙蛋糕的,只是點綴蛋糕的。那么顯然,在點綴之前必須先烘焙出一個蛋糕來才行,那么這個烘焙蛋糕的函數就是__new__。

__new__函數

我們來看下__new__這個函數的定義,我們在使用Python面向對象的時候,一般都不會重構這個函數,而是使用Python提供的默認構造函數,Python默認構造函數的邏輯大概是這樣的:

def__new__(cls,*args,**kwargs):

returnsuper().__new__(cls,*args,**kwargs)

從代碼可以看得出來,函數當中基本上什么也沒做,就原封不動地調用了父類的構造函數。這里隱藏著Python當中類的創建邏輯,是根據繼承關系一級一級創建的。根據邏輯關系,我們可以知道,當我們創建一個實例的時候,實際上是先調用的__new__函數創建實例,然后再調用__init__對實例進行的初始化。我們可以簡單做個實驗:

classTest:

def__new__(cls):

print('__new__')

returnobject().__new__(cls)

def__init__(self):

print('__init__')

當我們創建Test這個類的時候,通過輸出的順序就可以知道Python內部的調用順序。

從結果上來看,和我們的推測完全一樣。

單例模式

那么我們重載__new__函數可以做什么呢?一般都是用來完成__init__無法完成的事情,比如前面說的單例模式,通過__new__函數就可以實現。我們來簡單實現一下:

classSingletonObject:

def__new__(cls,*args,**kwargs):

ifnothasattr(SingletonObject,"_instance"):

SingletonObject._instance=object.__new__(cls)

returnSingletonObject._instance

def__init__(self):

pass

當然,如果是在并發場景當中使用,還需要加上線程鎖防止并發問題,但邏輯是一樣的。

除了可以實現一些功能之外,還可以控制實例的創建。因為Python當中是先調用的__new__再調用的__init__,所以如果當調用__new__的時候返回了None,那么最后得到的結果也是None。通過這個特性,我們可以控制類的創建。比如設置條件,只有在滿足條件的時候才能正確創建實例,否則會返回一個None。

比如我們想要創建一個類,它是一個int,但是不能為0值,我們就可以利用__new__的這個特性來實現:

classNonZero(int):

def__new__(cls,value):

returnsuper().__new__(cls,value)ifvalue!=0elseNone

那么當我們用0值來創建它的時候就會得到一個None,而不是一個實例。

工廠模式

理解了__new__函數的特性之后,我們就可以靈活運用了。我們可以用它來實現許多其他的設計模式,比如大名鼎鼎經常使用的工廠模式。

所謂的工廠模式是指通過一個接口,根據參數的取值來創建不同的實例。創建過程的邏輯對外封閉,用戶不必關系實現的邏輯。就好比一個工廠可以生產多種零件,用戶并不關心生產的過程,只需要告知需要零件的種類。也因此稱為工廠模式。

比如說我們來創建一系列游戲的類:

classLast_of_us:

defplay(self):

print('theLastOfUsisreallyfunny')

classUncharted:

defplay(self):

print('theUnchartedisreallyfunny')

classPSGame:

defplay(self):

print('PShasmanygames')

然后這個時候我們希望可以通過一個接口根據參數的不同返回不同的游戲,如果不通過__new__,這段邏輯就只能寫成函數而不能通過面向對象來實現。通過重載__new__我們就可以很方便地用參數來獲取不同類的實例:

classGameFactory:

games={'last_of_us':Last_Of_us,'uncharted':Uncharted}

def__new__(cls,name):

ifnameincls.games:

returncls.games[name]()

else:

returnPSGame()

uncharted=GameFactory('uncharted')

last_of_us=GameFactory('last_of_us')

總結

相信看到這里,關于__new__這個函數的用法應該都能理解了。一般情況下我們是用不到這個函數的,只會在一些特殊的場景下使用。雖然如此,我們學會它并不只是用來實現設計模式,更重要的是可以加深我們對于Python面向對象的理解。

以上內容為大家介紹了Python之__init__和__new__的區別是什么,希望對大家有所幫助,如果想要了解更多Python相關知識,請關注IT培訓機構:千鋒教育。

tags: python培訓
聲明:本站稿件版權均屬千鋒教育所有,未經許可不得擅自轉載。
10年以上業內強師集結,手把手帶你蛻變精英
請您保持通訊暢通,專屬學習老師24小時內將與您1V1溝通
免費領取
今日已有369人領取成功
劉同學 138****2860 剛剛成功領取
王同學 131****2015 剛剛成功領取
張同學 133****4652 剛剛成功領取
李同學 135****8607 剛剛成功領取
楊同學 132****5667 剛剛成功領取
岳同學 134****6652 剛剛成功領取
梁同學 157****2950 剛剛成功領取
劉同學 189****1015 剛剛成功領取
張同學 155****4678 剛剛成功領取
鄒同學 139****2907 剛剛成功領取
董同學 138****2867 剛剛成功領取
周同學 136****3602 剛剛成功領取
相關推薦HOT
主站蜘蛛池模板: 国产精品素人福利| 伊人动漫| 欧美日韩国产精品自在自线| 玉蒲团之偷情宝鉴电影| 欧美一级大片在线观看| 杨幂精品国产福利在线| 老八吃屎奥利给原视频带声音的| 精品国产精品久久一区免费式| 色综合视频一区二区三区| 2021日韩麻豆| 国产成人综合精品一区| 久久久久久久综合色一本| 国产成在线观看免费视频成本人 | 成人亚洲欧美日韩在线观看| 久久香蕉综合色一综合色88| 国产一区二区精品| 久久www免费人成看片入口| 一本色道久久88加勒比—综合| 国产精品久久久久无码av| 吃奶摸下的激烈免费视频播放| 中文字幕2020| 中文字幕黑人借宿神宫寺| 性欧美hd| 伦理片中文字幕2019在线| 欧美中文字幕在线| 国色天香精品一卡2卡3卡| 波多野结衣四虎| 欧美ol丝袜高跟秘书在线播放| 色诱久久av| 西西人体44rt大胆高清日韩| 美国bbbbbbbbb免费毛片| 红色一片免费高清影视| 亚洲欧洲日产国产最新| 国产精品9999久久久久仙踪林| 抽搐一进一出在深一点| 国产高清免费在线观看| 在车子颠簸中进了老师的身体| 在车上狠狠的吸她的奶| 国产国语对白露脸在线观看| 日韩精品一区二区三区在线观看l| 小镇姑娘hd电影在线观看|