Python中的單例模式與反射機制詳解

單例模式

一般情況下,類可以生成任意個實例,而單例模式隻生成一個實例

我們先用單例模式設計一個Rectangle類

然後用__new__方法設計單例模式,代碼如下

class Rectangle:
    def __init__(self,length,width):
        self.length=length
        self.width=width
    def __new__(cls, *args, **kwargs):#__new__至少要有一個參數cls,代表要實例化的類
        if not hasattr(cls,'obj'):  #判斷類當中有沒有實例,如果沒有則新建
            cls.obj=object.__new__(cls)  #生成實例對象
        return cls.obj

然後我們來驗證下,單例模式下是否隻能生成一個實例

rec1 = Rectangle(10,8)#判斷類中沒有實例,則新建實例rec1
print(rec1.length,rec1.width)#這時候我們打印下rec1這個長方形的長和寬
>>>10 8#是我們輸入的值
rec2 = Rectangle(6,4)
#Rectangle已經創建過rec1實例瞭,所以不會再生成新的實例,此時的rec1和rec2指向是同一個對象:Rectangle(6,4)
print(rec1.length,rec1.width)#打印下rec1的長和寬
print(rec2.length,rec2.width)#打印下rec2的長和寬
>>>
6 4 
6 4 
print(id(rec1)==id(rec2))
>>>Ture #對象/實例隻有一個,隻是賦給瞭不同的變量名

單例模式在程序設計中比較典型的應用場景:多個用戶同時調用某個模塊時,會生成一些日志,我們希望這些日志存在同一個文件內,而不是多個文件。

在生成日志模塊我們就可以采用單例模式進行設計。

反射

概念:簡單來說就是可以利用字符串來映射模塊中的相應方法然後可以操作模塊中相應的方法

我們以一個飯店點菜的實際場景來理解Python的反射機制

class Food_list:#創建一個飯店菜單的類
    def yxrs(self):
        return '魚香肉絲'
    def gbjd(self):
        return '宮保雞丁'
    def hspg(self):
        return '紅燒排骨'

hasatter

hasatter(對象,屬性或方法名)

判斷對象中是否有某個屬性或某個方法,返回值是佈爾型

guke1 = Food_list()#實例化一個顧客對象
while True:
    diancai = input('請點菜:  ')
    if hasattr(guke1,diancai):#判斷顧客點的菜有沒有
        print('好的,馬上去做')
        break
    else:
        print('這個沒有,換個別的吧')
>>>請點菜:  佛跳墻
>>>這個沒有,換個別的吧
>>>請點菜:  yxrs
>>>好的,馬上去做

getattr

getattr(對象,屬性或方法名,缺省值) 判斷對象中是否有某個屬性或某個方法,如果有返回方法本身,沒有則返回缺省值

guke1 = Food_list()#實例化一個顧客對象
def fun1():#定義一個方法,用作getattr的缺省值
    return '沒有這道菜'
foodname = input('請輸入要翻譯的菜名:')
a = getattr(guke1,foodname,fun1)
#判斷guke1中有沒有foodname方法,有則返回這個方法,沒有則返回fun1方法
print(a())#執行返回的方法
>>>請輸入要翻譯的菜名:yxrs#guke1中有yxrs方法,則返回這個方法
>>>魚香肉絲
>>>請輸入要翻譯的菜名:dsfsfs#guke1中有沒有dsfsfs方法,則返回fun1方法
>>>沒有這道菜

setattr

setattr(對象,屬性,新值)

將實例的屬性改為新的值,如果屬性不存在則新建

我們給實例guke1加個價格屬性

guke1 = Food_list()#實例化一個顧客對象
setattr(guke1,'proce','23元')
print(guke1.proce)
>>>
23元

總結

本篇文章就到這裡瞭,希望能夠給你帶來幫助,也希望您能夠多多關註WalkonNet的更多內容!

推薦閱讀: