1. 程式人生 > >Python帶_的變量或函數命名

Python帶_的變量或函數命名

attribute eth 機制 初始化 索引操作 self 內置函數 bject str

python中的標識符可以包含數字、字母和_,但必須以字母或者_開頭,其中以_開頭的命名一般具有特殊的意義。

前後均帶有雙下劃線__的命名

一般用於特殊方法的命名,用來實現對象的一些行為或者功能,比如__new__()方法用來創建實例,__init__()方法用來初始化對象,

x + y操作被映射為方法x.__add__(y),序列或者字典的索引操作x[k]映射為x.__getitem__(k),__len__()、__str__()分別被內置函數len()、str()調用等等。

僅開頭帶雙下劃線__的命名

用於對象的數據封裝,以此命名的屬性或者方法為類的私有屬性或者私有方法。

class Foo(object):
    def __init__(self):
        self.__name = ‘private name‘

    def getname(self):
        return self.__name

    def __spam(self):
        print ‘private method‘

    def bar(self):
        self.__spam()

如果在外部直接訪問私有屬性或者方法:

>>> f = Foo()
>>> f.__name

Traceback (most recent call last):
  File "<pyshell#1>", line 1, in <module>
    f.__name
AttributeError: ‘Foo‘ object has no attribute ‘__name‘
>>> f.__spam()

Traceback (most recent call last):
  File "<pyshell#2>", line 1, in <module>
    f.__spam()
AttributeError: ‘Foo‘ object has no attribute ‘__spam‘

是不可行的,這就起到了隱藏數據的作用,但是這種實現機制並不是很嚴格,機制是通過自動"變形"實現的,類中所有以雙下劃線開頭的名稱__name都會自動變為"_類名__name"的新名稱:

>>> f._Foo__name
‘private name‘
>>> f._Foo__spam()
private method

這樣就可以訪問了。

這種機制可以阻止繼承類重新定義或者更改方法的實現,比如,定義一個Foo的派生類:

class Goo(Foo):
    def __spam(self):
        print ‘private method of Goo‘

重寫了__spam方法,運行:

>>> g = Goo()
>>> g.bar()
private method

調用bar()方法的時候依然執行的是Foo類的__spam()方法,因為在bar()方法的實現中,self.__spam()已自動變形為self._Foo__spam(),Goo繼承的bar()方法也是如此。

以單下劃線_開頭的命名

一般用於模塊中的"私有"定義的命名。

from module import * 語句用於加載模塊中的所有名稱,要控制導入的名稱,一種方法是定義列表__all__,只有在__all__中的名稱才能通過*導入,

另一種方法就是以單下劃線開頭命名定義了,這種定義不會被*導入。

當然,在類中也可以用單下劃線開頭來命名屬性或者方法,這只是表示類的定義者希望這些屬性或者方法是"私有的",但實際上並不會起任何作用。

Python帶_的變量或函數命名