python的__class__()

lm_y發表於2017-09-03

python class 的屬性

Class 有一些特殊的屬性,便於我們獲得一些額外的資訊。
>>> class Class1(object):
    """Class1 Doc."""
    def __init__(self):
        self.i = 1234

>>> Class1.__doc__ # 型別幫助資訊
'Class1 Doc.'
>>> Class1.__name__ # 型別名稱
'Class1'
>>> Class1.__module__ # 型別所在模組
'__main__'
>>> Class1.__bases__ # 型別所繼承的基類
(<type 'object'>,)
>>> Class1.__dict__ # 型別字典,儲存所有型別成員資訊。
<dictproxy object at 0x00D3AD70>
>>> Class1().__class__ # 型別
<class '__main__.Class1'>
>>> Class1().__module__ # 例項型別所在模組
'__main__'
>>> Class1().__dict__ # 物件字典,儲存所有例項成員資訊。
{'i': 1234}

成員

Python Class 同樣包含型別和例項兩種成員。

>>> class Class1:
    i = 123 # Class Field
    def __init__(self):
        self.i = 12345 # Instance Field
        
>>> print Class1.i
123
>>> print Class1().i
12345
兩個比較重要的取例項屬性的函式:

getattr(例項,屬性名),hasattr(例項,屬性名)判斷屬性名是否屬於例項。

如上例:s='i',getattr(Class1(),s)=12345

hasattr(Class1(),s)=True
-----------------------

有幾個很 "特殊" 的 "規則" 需要注意。

(1) 我們可以通過例項引用訪問型別成員。因此下面的例子中 self.i 實際指向 Class1.i,直到我們為例項新增了一個成員 i。

>>> class Class1:
    i = 123
    def __init__(self):
        print self.i 
        print hex(id(self.i))

>>> hex(id(Class1.i)) # 顯示 Class1.i
'0xab57a0'
>>> a = Class1() # 建立 Class1 例項,我們會發現 self.i 實際指向 Class1.i。
123
0xab57a0
>>> Class1.__dict__ # 顯示 Class1 成員
{'i': 123, '__module__': '__main__', '__doc__': None, '__init__': <function __init__ at 0x00D39470>}
>>> a.__dict__ # 顯示例項成員
{}
>>> a.i = 123456789 # 為例項新增一個成員 i
>>> hex(id(a.i)) # 顯示新增例項成員地址
'0xbbb674'
>>> a.__dict__ # 顯示例項成員
{'i': 123456789}

我們可以在成員名稱前新增 "__" 使其成為私有成員。

>>> class Class1:
    __i = 123
    def __init__(self):
        self.__x = 0
    def __test(self):
        print id(self)


事實上這只是一種規則,並不是編譯器上的限制。我們依然可以用特殊的語法來訪問私有成員。
>>> Class1._Class1__i
123
>>> a = Class1()
>>> a._Class1__x
0
>>> a._Class1__test()
13860376
-----------------------

除了靜態(型別)欄位,我們還可以定義靜態方法。
>>> class Class1:
    @staticmethod
    def test():
        print "static method"
>>> Class1.test()
static method

過載
Python 支援一些特殊方法和運算子過載。

>>> class Class1:
    def __init__(self):
        self.i = 0
    def __str__(self):
        return "id=%i" % id(self)
    def __add__(self, other):
        return self.i + other.i

>>> a = Class1()
>>> a.i = 10
>>> str(a)
'id=13876120'
>>> b = Class1()
>>> b.i = 20
>>> a + b
30
通過過載 "__eq__",我們可以改變 "==" 運算子的行為。
>>> class Class1:
    pass

>>> a = Class1()
>>> b = Class1()
>>> a == b
False

>>> class Class1:
    def __eq__(self, x):
        return 55
>>> a = Class1()
>>> b = Class1()
>>> a == b
55

相關文章