Python 訪問(wèn)限制 private public的詳細(xì)介紹
一、知識(shí)點(diǎn)
在一個(gè)模塊中,我們可能會(huì)定義很多函數(shù)和變量。但有的函數(shù)和變量我們希望能給別人使用,有的函數(shù)和變量我們希望僅僅在模塊內(nèi)部使用,so?
我們可以通過(guò)定義該函數(shù)、變量是公開(kāi)的還是私有的來(lái)達(dá)到該目的。
在Python中,是通過(guò)下劃線“_”前綴來(lái)實(shí)現(xiàn)的。
- public:公開(kāi)的。正常的函數(shù)和變量名為此類(lèi)型,可以被直接引用。比如變量abc、PI等;
- 特殊變量:格式為_(kāi)_xxx__ ,以__開(kāi)頭、以__結(jié)尾??梢灾苯颖灰?,但是有特殊用途。比如 __author__ 、__name__就是特殊變量。一般自己定義的變量不要用這種變量名。
- private:私有的、非公開(kāi)的,格式類(lèi)似于_xxx_ 和__xxx,例如__num。
不應(yīng)該被直接引用,只有內(nèi)部可以訪問(wèn),外部不能訪問(wèn)。
不能隨意修改對(duì)象內(nèi)部的狀態(tài),這樣通過(guò)訪問(wèn)限制的保護(hù),代碼更加健壯。
二、舉例說(shuō)明
在Class類(lèi)內(nèi)部,可以有屬性和方法。而外部代碼可以通過(guò)直接調(diào)用實(shí)例變量的方法來(lái)操作數(shù)據(jù),隱藏了內(nèi)部復(fù)雜邏輯。但是,外部代碼還是可以自由地修改一個(gè)實(shí)例的屬性。例如:
>>>b.score 99 >>>b.score = 59 >>>b.score 59
如果要讓內(nèi)部屬性不被外部訪問(wèn),可以把屬性的名稱(chēng)前加上兩個(gè)下劃線“__”,變成私有變量,如下:
class Student(object): def __init__(self, name, score): self.__name = name self.__score = score def print_score(self): print('%s: %s' % (self.__name, self.__score))
嘗試在外部對(duì)屬性進(jìn)行訪問(wèn),發(fā)現(xiàn)會(huì)報(bào)錯(cuò),因?yàn)樗接凶兞?,不能被外部訪問(wèn)。
>>> bart = Student('Bart Simpson', 98) >>> bart.__name # 私有變量:不能被外部訪問(wèn) Traceback (most recent call last): File "<stdin>", line 1, in <module> AttributeError: 'Student' object has no attribute '__name'
但是,如果外部代碼要獲取name和score怎么辦?
給Student類(lèi)增加獲取屬性的方法:get_name()和get_score(),如下:
class Student(object): ... def get_name(self): return self.__name def get_score(self): return self.__score
如果外部代碼修改score怎么辦?可以再給Student類(lèi)增加設(shè)置方法:set_score():
... def set_score(self, score): # 避免傳入無(wú)效參數(shù) if 0 <= score <= 100: self.__score = score else: raise ValueError('bad score')
那作為雙下劃線開(kāi)頭的私有實(shí)例變量是不是一定不能從外部訪問(wèn)呢?其實(shí)也不是。
不能直接訪問(wèn)__name是因?yàn)镻ython解釋器對(duì)外把__name變量改成了_Student__name,所以仍然可以通過(guò)_Student__name來(lái)訪問(wèn)__name變量。
>>> bart = Student('Bart Simpson', 98) >>> bart.get_name() 'Bart Simpson' >>> bart.__name = 'New Name' # 給bart新增的__name變量 >>> bart.__name # !與class內(nèi)部的__name變量不是一個(gè)變量! 'New Name' >>> bart.get_name() # get_name()內(nèi)部返回self.__name (_Student__name) 'Bart Simpson'
表面上看,外部代碼“成功”地設(shè)置了__name變量,但實(shí)際上這個(gè)__name變量和class內(nèi)部的__name變量不是一個(gè)變量!內(nèi)部的__name變量已經(jīng)被Python解釋器自動(dòng)改成了_Student__name,而外部代碼給bart新增了一個(gè)__name變量。
所以python并沒(méi)有一種方法可以完全限制訪問(wèn)private的函數(shù)或變量,所以不是“不能被直接引用”,從編程的習(xí)慣上不應(yīng)該引用private函數(shù)或變量。那他們的用處呢?
例如:
def _private_1 (name): return 'hello,%s ' % name def _private_2 (name): return 'hi , %s ' % name def greeting(name): if len(name) > 3: return _private_1 (name) else: return _private_2 (name)
在模塊里公開(kāi)greeting()函數(shù),而把內(nèi)部邏輯用private函數(shù)隱藏起來(lái)了。這樣,調(diào)用greeting()函數(shù)不用關(guān)心內(nèi)部的私有函數(shù)的細(xì)節(jié)。
這是一種非常有用的代碼封裝和抽象的方法,即:外部不需要引用的函數(shù)全部定義成private,只有外部需要引用的函數(shù)才定義為public。
三、完整代碼
class Student(object): def __init__(self, name, score): self.__name = name self.__score = score def print_score(self): print('%s: %s' % (self.__name, self.__score)) def get_name(self): return self.__name def get_score(self): return self.__score def set_score(self, score): # 避免傳入無(wú)效參數(shù) if 0 <= score <= 100: self.__score = score else: raise ValueError('bad score') def _private_1 (name): return 'hello,%s ' % name def _private_2 (name): return 'hi , %s ' % name def greeting(name): if len(name) > 3: return _private_1 (name) else: return _private_2 (name)
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
python?針對(duì)在子文件夾中的md文檔實(shí)現(xiàn)批量md轉(zhuǎn)word
這篇文章主要介紹了python?針對(duì)在子文件夾中的md文檔實(shí)現(xiàn)批量md轉(zhuǎn)word,但是自己保存的md文檔在不同的文件夾,而大部分只能實(shí)現(xiàn)同一文件夾內(nèi)的轉(zhuǎn)換,得出下列總結(jié),需要的朋友可以參考一下2022-04-04python的變量和簡(jiǎn)單數(shù)字類(lèi)型詳解
這篇文章給大家詳細(xì)介紹了python的變量和簡(jiǎn)單數(shù)字類(lèi)型,文中介紹的很詳細(xì),相信對(duì)大家的理解和學(xué)習(xí)很有幫助,有需要的朋友們可以參考借鑒2021-09-09淺談tensorflow中dataset.shuffle和dataset.batch dataset.repeat注意點(diǎn)
這篇文章主要介紹了淺談tensorflow中dataset.shuffle和dataset.batch dataset.repeat注意點(diǎn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-06-06