亚洲免费在线-亚洲免费在线播放-亚洲免费在线观看-亚洲免费在线观看视频-亚洲免费在线看-亚洲免费在线视频

Python中使用__new__實(shí)現(xiàn)單例模式并解析

系統(tǒng) 3063 0

單例模式是一個(gè)經(jīng)典設(shè)計(jì)模式,簡(jiǎn)要的說(shuō),一個(gè)類的單例模式就是它只能被實(shí)例化一次,實(shí)例變量在第一次實(shí)例化時(shí)就已經(jīng)固定。

?在Python中常見的單例模式有None,這就是一個(gè)很典型的設(shè)計(jì),通常使用 if xxx is None或者if xxx is not None來(lái)比較運(yùn)算。

Python實(shí)現(xiàn)單例模式

代碼如下:

            
class MyClass:
  _instance = None 
  _first_init = False 
  def __new__(cls, *args, **kwargs):
    if not cls._instance:
      cls._instance = super().__new__(cls)
    return cls._instance
  def __init__(self, var1, var2):
    cls = type(self)
    if not cls._first_init:
      self.var1 = var1
      self.var2 = var2
      cls._first_init = True
          

如上所示,我創(chuàng)建了一個(gè)MyClass的類,定義了兩個(gè)類變量,第一個(gè)是_instance,它負(fù)責(zé)保存該類創(chuàng)建的實(shí)例。第二個(gè)是_first_init,它是一個(gè)布爾值,保存是否為第一次實(shí)例化該類。

在__new__方法中(構(gòu)造函數(shù)),判斷是否存在_instance這個(gè)類變量,如果之前已經(jīng)實(shí)例化了,直接返回。如果是第一次實(shí)例化,就會(huì)為_instance類變量綁定實(shí)例,使用super().__new__(cls)創(chuàng)建實(shí)例,即調(diào)用父類object.__new__(MyClass)創(chuàng)建實(shí)例。

在__init__方法中(初始化函數(shù)),我們通過(guò)cls=type(self)獲取MyClass類,判斷是否第一次實(shí)例化。如果是第一次實(shí)例化,就綁定實(shí)例變量。否則什么都不做.

運(yùn)行效果

我們創(chuàng)建兩個(gè)實(shí)例,來(lái)對(duì)比

            
>>> instance1 = MyClass(1, 2)
>>> instance2 = MyClass(7, 5)
>>> id(instance1) == id(instance2)
True

>>> instance2.var1
1
          

可以看到,這兩個(gè)實(shí)例的內(nèi)存地址都相同,而且第一次實(shí)例化后變量已經(jīng)固定了,全局不會(huì)再改變。

這就是單例模式的實(shí)現(xiàn)。

ps:下面看下Python中類方法、__new__方法和__init__方法解析

在編程語(yǔ)言中創(chuàng)建一個(gè)類,有構(gòu)造方法這樣的一個(gè)術(shù)語(yǔ)。而在Python中,通常大家都認(rèn)為__init__是構(gòu)造方法,其實(shí)并不完全等同。在構(gòu)建類中,有一個(gè)特殊的方法__new__,這個(gè)才能等同為構(gòu)造方法。

__new__是一個(gè)類方法,我們?cè)诙x一個(gè)類方法時(shí)需要在函數(shù)前打上@classmethod裝飾器,而__new__不需要,因?yàn)樗?jīng)過(guò)特殊處理。為了理解__new__方法,我們先來(lái)看看類方法是什么。

類方法

            
class MyClass:

  @classmethod
  def test(cls):
    print(cls.__name__)
    
MyClass.test()
#輸出 MyClass
          

在MyClass類中,test方法就是類方法,它傳入第一個(gè)參數(shù)為cls,其實(shí)就是MyClass類,打印cls.__name__可以看到結(jié)果。類方法可以直接 類名.方法名()調(diào)用。通常類方法是備選構(gòu)造方法。

類方法的應(yīng)用

            
>>> from datetime import datetime
>>> datetime.fromtimestamp(324234)
datetime.datetime(1970, 1, 5, 2, 3, 54)
          

如上所示,內(nèi)置的datetime包中的fromtimestamp就是類方法,可以從多種方式構(gòu)造出datetime對(duì)象。

__new__方法

            
def __new__(cls, a):
  return super().__new__(cls)
          

__new__是類方法,所以第一個(gè)參數(shù)也是cls,剩下的參數(shù)就是構(gòu)造方法里需要的參數(shù)了。通常__new__都不需要定義,在元類編程中才需要,它可以控制類的生成過(guò)程。

__new__必須返回一個(gè)實(shí)例(instance),傳入到__init__方法中的self參數(shù),也就是實(shí)例變量。這里返回父類(object)的__new__方法用來(lái)創(chuàng)建一個(gè)新的實(shí)例。相當(dāng)于

            
obj = object.__new__(MyClass)
obj = MyClass()
          

#obj是實(shí)例,上面兩個(gè)方式等同

其中,MyClass是類,obj是實(shí)例(instance)

__init__方法

__new__是構(gòu)造方法,那么__init__就是初始化函數(shù),它負(fù)責(zé)將變量綁定到實(shí)例中,更新實(shí)例的__dict__字典。其中第一個(gè)參數(shù)self就是__new__的返回值,是類的實(shí)例。__new__方法先于__init__方法執(zhí)行

            
def __init__(self, a):
  self.a = a
          

結(jié)合使用

            
class MyClass:
  def __new__(cls, a):
    return super().__new__(cls)

  def __init__(self, a):
    self.a = a
obj = MyClass(3)
print(obj.a)
          

要點(diǎn)

1.__new__是構(gòu)造方法,__init__是初始化函數(shù)。
2.__new__通常不需要手動(dòng)定義,一般在元類編程中使用,控制類的生成過(guò)程。
3.__new__第一個(gè)被執(zhí)行,然后執(zhí)行__init__綁定實(shí)例變量。
4.__new__必須有返回值,返回值是該類的實(shí)例,它會(huì)被__init__函數(shù)接收,通常叫做self變量

總結(jié)

以上所述是小編給大家介紹的Python中使用__new__實(shí)現(xiàn)單例模式并解析,希望對(duì)大家有所幫助,如果大家有任何疑問(wèn)請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!
如果你覺(jué)得本文對(duì)你有幫助,歡迎轉(zhuǎn)載,煩請(qǐng)注明出處,謝謝!


更多文章、技術(shù)交流、商務(wù)合作、聯(lián)系博主

微信掃碼或搜索:z360901061

微信掃一掃加我為好友

QQ號(hào)聯(lián)系: 360901061

您的支持是博主寫作最大的動(dòng)力,如果您喜歡我的文章,感覺(jué)我的文章對(duì)您有幫助,請(qǐng)用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點(diǎn)擊下面給點(diǎn)支持吧,站長(zhǎng)非常感激您!手機(jī)微信長(zhǎng)按不能支付解決辦法:請(qǐng)將微信支付二維碼保存到相冊(cè),切換到微信,然后點(diǎn)擊微信右上角掃一掃功能,選擇支付二維碼完成支付。

【本文對(duì)您有幫助就好】

您的支持是博主寫作最大的動(dòng)力,如果您喜歡我的文章,感覺(jué)我的文章對(duì)您有幫助,請(qǐng)用微信掃描上面二維碼支持博主2元、5元、10元、自定義金額等您想捐的金額吧,站長(zhǎng)會(huì)非常 感謝您的哦!??!

發(fā)表我的評(píng)論
最新評(píng)論 總共0條評(píng)論
主站蜘蛛池模板: 五月天婷婷缴情五月免费观看 | 成人午夜影视全部免费看 | 久久精品国产视频在热 | 久久久久久亚洲精品影院 | 奇米影视狠狠 | 中国明星一级毛片免费 | 天天色天天射天天操 | 亚洲综合免费视频 | 欧美一级a俄罗斯毛片 | 久久免费视频播放 | 亚洲免费视频一区 | 免费在线亚洲 | 免费在线激情视频 | 日日夜夜嗷嗷叫 | 欧美一区二区三区综合色视频 | 一区二区免费播放 | 黄黄视频免费看 | 久99久精品免费视频热77 | 尤物久久99热国产综合 | 99热这里只有精品在线观看 | 亚洲第一伊人 | 草草视频免费观看 | 日本亚洲精品成人 | 精品久久在线观看 | 久久亚洲国产精品一区二区 | 香蕉在线播放 | 夜夜爱夜夜爽 | 在线日韩麻豆一区 | 欧美日韩亚洲区久久综合 | 99久久国产综合色 | 国产精品久久久免费视频 | 五月婷婷亚洲 | 婷婷日韩| 国产九九在线 | 爱久久www.35669 | 元龙第三季免费观看 | 一级毛片免费视频网站 | 香蕉视频影院 | 久久影院国产 | 国产日韩精品视频一区二区三区 | 老司机亚洲精品 |