python 多继承 __new___Python3中的__new__方法以及继承不可变类型类的问题
最近在學(xué)到Python中的__new__方法時(shí)被弄懵逼了,一開始實(shí)在是很難理解,有很多地方想不通(本人強(qiáng)迫癥)。最近自己慢慢思索得出了能說服自己的理解:
說__new__方法之前要先提到__init__方法,__init__方法是普遍認(rèn)為的Python類的構(gòu)造方法,在我們對(duì)類進(jìn)行實(shí)例化的時(shí)候,Python解釋器會(huì)調(diào)用__init__方法對(duì)我們?cè)趇nit方法中定義的屬性進(jìn)行初始化,比如:
classdemo():def __init__(self,arg,kwarg): #定義屬性并初始化
self.arg=arg
self.kwarg=kwargdefOutput(self):print(self.arg)print(self.kwarg)
a= demo("NMSL","WSND") #實(shí)例化
a.Output() #調(diào)用類中的Output方法
但是其實(shí)在Python中,__init__并不是真正的構(gòu)造函數(shù),準(zhǔn)確的說,__new__加__init__才是真正的構(gòu)造函數(shù),下面詳細(xì)說一下__new__方法。
當(dāng)我們?cè)趯?duì)類進(jìn)行實(shí)例化的時(shí)候,Python解釋器會(huì)從__new__方法的返回值中獲取到實(shí)例對(duì)象的信息,在上面代碼的例子中,我們把類demo進(jìn)行實(shí)例化,對(duì)象為a
在這個(gè)過程中,類demo是怎么確定它的實(shí)例對(duì)象是a呢,就是__new__方法返回了這個(gè)值,這個(gè)值就是a,然后Python解釋器就知道了demo這個(gè)類的對(duì)象為a,其實(shí)這里的返回值最后被Python解釋器告訴__init__方法,init方法中的關(guān)鍵字self其實(shí)就是這個(gè)a,因?yàn)閟elf代表的是類的對(duì)象。
__new__方法還有另一個(gè)用處就是用來實(shí)現(xiàn)單例設(shè)計(jì)模式以及繼承不可變類的時(shí)候方便我們定制類:
classnewfloat(float):def __new__(cls,value):return super().__new__(cls,round(value,4))
i= newfloat(3.14529)print(i)
這里的思路是用來接收__new__方法的返回值再打印出來。
老實(shí)說我在很多文章下面看到過關(guān)于__new__方法的應(yīng)用,都是拿這個(gè)舉例子,但是我覺得根本沒有必要這樣,我覺得可以這樣:
classdemo(int):def __init__(self,arg):
self.arg=arg
self.arg= round(self.arg,4)print(self.arg)
a= demo(3.1415926)
這樣的寫法同樣能實(shí)現(xiàn)功能。
單例設(shè)計(jì)模式是為了解決一個(gè)類有多個(gè)對(duì)象的時(shí)候,多個(gè)對(duì)象引用同一個(gè)內(nèi)存地址,以減少內(nèi)存占用的問題。
實(shí)現(xiàn)思路:
重寫父類的__new__方法,使每次返回的內(nèi)存地址引用都為同一個(gè)。
classdemo(object):
ins=Nonedef __new__(cls):if cls.ins ==None:
cls.ins= super().__new__(cls)return cls.ins
a = demo()
b = demo()
print(a)
print(b)
這里的思路是:定義一個(gè)類屬性ins為空值,再在重寫__new__方法時(shí)候用if語句進(jìn)行判斷,如果ins為空則賦給它__new__方法的返回值,當(dāng)用a去實(shí)例化demo時(shí),
實(shí)際上是Python解釋器為這次實(shí)例化開辟了一片內(nèi)存空間,a就是引用了這個(gè)內(nèi)存地址。
然后我們?cè)儆胋去實(shí)例化demo,由于ins的值已經(jīng)被改寫,所以返回的是同樣的之前a對(duì)象的內(nèi)存地址,這樣就成功實(shí)現(xiàn)了單例設(shè)計(jì)模式。
這里必須好好解釋一下__new__方法的參數(shù),在定義new方法時(shí),在參數(shù)欄里不能寫self,而是寫cls,self代表的是類的實(shí)例,而cls則代表的是類本身。
繼承不可變類型,我們知道Python中的不可變類型有int,str和tuple,但是Python中有這么一個(gè)思想:萬物皆對(duì)象。也就是說,int,str和tuple這中數(shù)據(jù)類型其實(shí)也是一種對(duì)象,拿字符串舉例,我們可以利用count之類的方法直接對(duì)字符串進(jìn)行操作,但是這其實(shí)是Python內(nèi)置類str中定義的方法,我們可以用dir()方法查看類中定義有哪些方法,但是在重寫new方法時(shí)最多只能多傳一個(gè)參數(shù),這是為什么?
下面時(shí)我自己的解釋,我們平時(shí)用內(nèi)置方法比如count()時(shí),一次只能對(duì)一個(gè)字符串進(jìn)行操作,理所應(yīng)當(dāng),每次當(dāng)我們重寫str類中定義的方法時(shí),只能傳入一個(gè)值。
其他注意事項(xiàng):
__init__方法不能有返回值,而__new__方法卻必須有返回值。
如有錯(cuò)誤,請(qǐng)指正,感激不盡。
總結(jié)
以上是生活随笔為你收集整理的python 多继承 __new___Python3中的__new__方法以及继承不可变类型类的问题的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 华尔街英语学习软件_华尔街英语核心课程功
- 下一篇: python坐标怎么打_python导入