1. 程式人生 > >Python中常用內建屬性:__getattribute__屬性攔截器使用詳解

Python中常用內建屬性:__getattribute__屬性攔截器使用詳解

      因為python中所有類預設繼承object類。而object類提供了了很多原始的內建屬性和方法,所以使用者自定義的類在Python中也會繼承這些內建屬性。可以使用dir()函式可以檢視,雖然python提供了很多內建屬性但實際開發中常用的不多。而很多系統提供的內建屬性實際開發中使用者都需要重寫後才會使用。對於python來說,屬性或者函式都可以被理解成一個屬性

1.內建屬性__getattribute__的用法

1.重寫__getattribute__方法
class Student(object):
    country = "china"  #類屬性不會放到__dict__中
    def __init__(self,name,age):
        self.name = name
        self.age = age

    def __getattribute__(self, attr): #注意:attr是傳入的屬性名,不是屬性值
        print("開始屬性校驗攔截功能")
        print(attr)
        return object.__getattribute__(self, attr) #返回屬性名

s1 = Student("tom",19)
print(Student.country,s1.country,s1.name,s1.age) #呼叫屬性,會呼叫__getattribute__方法

'''注意結果,呼叫了四次屬性,但是卻只調用了3次 __getattribute__方法。

開始屬性校驗攔截功能
country
開始屬性校驗攔截功能
name
開始屬性校驗攔截功能
age
china china tom 19
'''

  分析總結: 

   1.__getattribute__是屬性訪問攔截器,就是當這個類的屬性被例項訪問時,會自動呼叫類的__getattribute__方法。當例項呼叫屬性時,比如s1.name,會把name作為實參傳進__getattribute__方法中,經過一系列操作後,再把name處理後的結果返回。Python中只要定義了繼承object的類,就預設存在屬性攔截器,只不過是攔截後沒有進行任何操作,而是直接返回。所以我們可以自己改寫__getattribute__方法來實現相關功能,比如檢視許可權、列印log日誌等。 

    2.如果重寫了__getattribute__,則類會呼叫重寫的方法,所以這個方法必須要有renturn返回值,返回傳進去的屬性,否則呼叫屬性會出現失敗的情況。

    3.注意如果是直接用類名.類屬性的形式呼叫類屬性,是不會呼叫 __getattribute__方法。

2.重寫__getattribute__實現屬性攔截功能

class Student(object):
    country = "china"  #類屬性不會放到__dict__中
    def __init__(self,name,age):
        self.name = name
        self.age = age

    def __getattribute__(self, attr): #注意:attr是傳入的屬性名,不是屬性值
        print("開始屬性校驗攔截功能")
        print(attr)
        if attr == "name":  #注意這裡引用原屬性名不用self,直接引號引起來即可。
            print("現在開始呼叫的是name屬性")
        elif attr =="age":
            print("現在開始呼叫的是age屬性")
        else:
            print("現在呼叫的是其他屬性")
        return object.__getattribute__(self, attr) #返回屬性名

s1 = Student("tom",19)
print(s1.name,s1.age,s1.country)

'''結果如下:
開始屬性校驗攔截功能
name
現在開始呼叫的是name屬性
開始屬性校驗攔截功能
age
現在開始呼叫的是age屬性
開始屬性校驗攔截功能
country
現在呼叫的是其他屬性
tom 19 china
'''

總結要點:1.__getattribute__(self,*args,**kwgs)中傳入的引數是屬性名,不是屬值,很多初學者有誤區。

                  2.使用類名呼叫類屬性時,不會經過__getattribute__方法,只爭取例項物件對屬性的呼叫,包括呼叫類屬性

                  3.__getattribute__是屬性攔截器,屬性呼叫會傳入處理,最後要有返回值,將傳入屬性處理後返回給呼叫者。