Python帶_的變量或函數命名
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帶_的變量或函數命名