在學習Python的時候,很多人都不理解為什么在方法(method)前面會加好幾個下劃線,有時甚至兩邊都會加,比如像 __this__ 這種。在我看到上面的文章之前,我一直以為Python中這些下劃線的作用就像Golang中方法/函數的大小寫一樣,或是一些其他語言中的 private 、 public 的作用一樣,但仔細深究,這不全是Python這樣設計的初衷。下面我們具體分析。
主要存在四種命名
1. object # 公用方法
2. __object__ # 內建方法,用戶不要這樣定義
3. __object # 全私有,全保護
4. _object # 半保護
核心風格:避免用下劃線作為變量名的開始。
因為下劃線對解釋器有特殊的意義,而且是內建標識符所使用的符號,我們建議程序員避免用下劃線作為變量名的開始。一般來講,變量名_object被看作是“私有的”,在模塊或類外不可以使用,不能用'from module import *'導入。當變量是私有的時候,用_object來表示變量是很好的習慣。
python/294016.html">單下劃線+類名,eg:_Class__object 機制就可以訪問__object__了。因為變量名__object__對Python 來說有特殊含義,對于普通的變量應當避免這種命名風格。
“單下劃線” 開始的成員變量叫做保護變量,意思是只有類對象和子類對象自己能訪問到這些變量;”雙下劃線” 開始的是私有成員,意思是只有類對象自己能訪問,連子類對象也不能訪問到這個數據。(如下列所示)
以單下劃線開頭_foo的代表不能直接訪問的類屬性,需通過類提供的接口進行訪問,不能用“from xxx import *”而導入;以雙下劃線開頭的__foo代表類的私有成員;以雙下劃線開頭和結尾的__foo__代表python里特殊方法專用的標識,如 __init__()代表類的構造函數。
class Foo(): def __init__(): ... def public_method(): print 'This is public method' def __fullprivate_method(): print 'This is fullprivate_method' def _halfprivate_method(): print 'This is halfprivate_method' f = Foo() f.public_method() # OK f.__fullprivate_method() # Error occur f._halfprivate_method() # OK f._Foo__fullprivate_method() # OK
從上面的例子可以看出,f._halfprivate_method()可以直接訪問,確實是。不過根據python的約定,應該將其視作private,而不要在外部使用它們,(如果你非要使用也沒轍),良好的編程習慣是不要在外部使用它。同時,根據Python docs的說明,_object和__object的作用域限制在本模塊內。
大家看下面這段程序的輸出:
class A(object): def __init__(self): self.__private() self.public() def __private(self): print 'A.__private()' def public(self): print 'A.public()'class B(A): def __private(self): print 'B.__private()' def public(self): print 'B.public()'b = B()
初探
正確的答案是:
A.__private()
B.public()
我們分別看下類A和類B的屬性都有哪些:
>>> print '/n'.join(dir(A))_A__private__init__public>>> print '/n'.join(dir(B))_A__private_B__private__init__public
為什么類A有個名為_A__private的 屬性呢?而且__private消失了!這就要談談Python的私有變量“矯直”了。
Python把以兩個或以上下劃線字符開頭且沒有以兩個或以上下劃線結尾的變量當作私有變量。私有變量會在代碼生成之前被轉換為長格式(變為公有)。轉換機制是這樣的:在變量前端插入類名,再在前端加入一個下劃線字符。這就是所謂的私有變量矯直(Private name mangling)。如類 A里的__private標識符將被轉換為_A__private,這就是上一節出現_A__private和__private消失的原因了。
再講兩點題外話:
一是因為矯直會使標識符變長,當超過255的時候,Python會切斷,要注意因此引起的命名沖突。
二是當類名全部以下劃線命名的時候,Python就不再執行矯直。如:
class ____(object): def __init__(self): self.__method() def __method(self): print '____.__method()'print '/n'.join(dir(____))__class____delattr____dict____doc____getattribute____hash____init____method # 沒被矯直__module____new____reduce____reduce_ex____repr____setattr____str____weakref__obj = ____()____.__method()obj.__method() # 可以外部調用____.__method()
現在我們回過頭來看看為什么會輸出“A.__private()”吧!
矯直之后,類A的代碼就變成這樣了:
class A(object): def __init__(self): self._A__private() # 這行變了 self.public() def _A__private(self): # 這行也變了 print 'A.__private()' def public(self): print 'A.public()'
因為在類B定義的時候沒有覆蓋__init__方法,所以調用的仍然是A.__init__,即執行了self._A__private(),自然輸出“A.__private()”了。
下面的兩段代碼可以增加說服力,增進理解:
class C(A): def __init__(self): # 重寫 __init__ ,不再調用self._A__private self.__private()# 這里綁定的是 _C_private self.public() def __private(self): print 'C.__private()' def public(self): print 'C.public()'c = C()答案:C.__private()C.public()
class A(object): def __init__(self): self._A__private() # 調用一個沒有定義的函數,但可執行 self.public() def __private(self): print 'A.__private()' def public(self): print 'A.public()'a = A()答案:A.__private()A.public()
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持VEVB武林網。
新聞熱點
疑難解答