淺談python對(duì)象數(shù)據(jù)的讀寫(xiě)權(quán)限
面向?qū)ο蟮木幊陶Z(yǔ)言在寫(xiě)大型程序的的時(shí)候,往往比面向過(guò)程的語(yǔ)言用起來(lái)更方便,安全。其中原因之一在于:類(lèi)機(jī)制。
類(lèi),對(duì)眾多的數(shù)據(jù)進(jìn)行分類(lèi),封裝,讓一個(gè)數(shù)據(jù)對(duì)象成為一個(gè)完整的個(gè)體,貼近現(xiàn)實(shí)生活,高度抽象化。但是,python對(duì)類(lèi)的封裝并不好,因?yàn)樗械膶傩院头椒ǘ际枪_(kāi)的,你可以隨意訪(fǎng)問(wèn)或者寫(xiě)入,你可以在類(lèi)的外部對(duì)類(lèi)的屬性進(jìn)行修改,甚至添加屬性。這的確讓人感到不安。
下面就來(lái)總結(jié)一下學(xué)習(xí)后的解決方案。
1,使用2個(gè)下劃線(xiàn)前綴隱藏屬性或者方法。
__xxx
#!/usr/bin/python3
#-*- coding:utf-8 -*-
class Student:
def __init__(self,name,score):
self.name = name
self.__score = score #將score隱藏起來(lái),使之僅在類(lèi)內(nèi)部可用。
def __show(self): #一個(gè)隱藏的方法,同樣只在內(nèi)部可用
print(self.name,self.__score)#使用被隱藏的屬性__score
def Show(self):
self.__show() #注意被隱藏方法的調(diào)用方式。
def main():
he = Student('Bob',95)
he.Show() #顯示:Bob 95
#print(he.__score) #AttributeError: 'Student' object has no attribute '__score'
#he.__show() #AttributeError: 'Student' object has no attribute '__show'
#隱藏屬性真的被隱藏了嗎?其實(shí)仍然可使用,使用格式 obj._className__attributeName
#但是僅僅作為了解,不建議使用隱藏屬性。
print(he._Student__show()) #顯示:Bob 95
print(he._Student__score) # 顯示: 95
if __name__=="__main__":
main()
雙下劃線(xiàn)對(duì)類(lèi)屬性的影響:
1. 使屬性只用于本類(lèi)的內(nèi)部,外部以及子類(lèi)都不可直接讀取修改。
2. 使用 _ _ 的類(lèi)的屬性,在實(shí)現(xiàn)時(shí)會(huì)被更改名稱(chēng),如類(lèi)中的__age 最后會(huì)變?yōu)開(kāi)A__age (名稱(chēng)重整),這個(gè)好處是:通常用于涉及到繼承的父類(lèi)中使用。這樣避免被子類(lèi)屬性覆蓋。
2.創(chuàng)建可管理的屬性。
有時(shí)候我們需要對(duì)屬性的寫(xiě)入做額外的檢查,對(duì)不合法的值拒絕寫(xiě)入,引發(fā)異常。
#!/usr/bin/python3
#-*- coding:utf-8 -*-
class Student:
def __init__(self,name,score):
self.name = name
self.score = score
@property #實(shí)現(xiàn)屬性的讀取方法,讀取實(shí)例的score值時(shí),就會(huì)調(diào)用這個(gè)函數(shù)
def score(self):
return self.__score
@score.setter #實(shí)現(xiàn)屬性寫(xiě)入方法,寫(xiě)入實(shí)例的score屬性時(shí),調(diào)用這個(gè)函數(shù)
def score(self,newVal):
if not isinstance(newVal,(int,float)):
raise TypeError('score value must be a number')
if newVal>100 or newVal<0:
raise ValueError('score value must between 0 and 100')
self.__score = newVal
def main():
he = Student('Bob',95)
he.score = 100 #重新寫(xiě)入
print(he.score) #讀取
if __name__=="__main__":
main()
我們可以發(fā)現(xiàn): self.__score是 屬性值 的真正存儲(chǔ)的地方,而self.score是函數(shù)(只不過(guò)它用起來(lái)像一個(gè)屬性),它是獲取和寫(xiě)入屬性值的方法。
初始化的時(shí)候也會(huì)調(diào)用socre.setter 裝飾的函數(shù),因?yàn)開(kāi)_init__()函數(shù)下出現(xiàn)了self.score的調(diào)用
既然self.__score僅僅用來(lái)引用屬性的值,可不可以用別的命名呢?如saveScore....當(dāng)然是可以的,但是,它“暴露”了,我們不想讓它在外部可用,還是應(yīng)該
加 __ 將它隱藏,防止意外修改。
有時(shí)候,你確定某個(gè)類(lèi)不會(huì)涉及到繼承,那么,就可以將上述的雙下劃線(xiàn)改寫(xiě)為單下滑線(xiàn),雖然不會(huì)達(dá)到隱藏的作用,但是:一方面,這樣不會(huì)引發(fā)名稱(chēng)重整機(jī)制,
避免小題大做,另一面,用一個(gè)下劃線(xiàn)開(kāi)頭,可以提醒使用者,這個(gè)屬性不應(yīng)該直接使用。那么,這就靠自覺(jué)了。
一個(gè)實(shí)例對(duì)象可以在外部隨意添加屬性。
#!/usr/bin/python3
#-*- coding:utf-8 -*-
class Student:
def __init__(self,name,score):
self.name = name
self.score = score
def main():
he = Student('Bob',95)
he.age = 19
print(he.age)
if __name__=="__main__":
main()
使用__slots__
#!/usr/bin/python3
#-*- coding:utf-8 -*-
class Student:
__slots__ = ('name','score') #將屬性名以字符串形式加入元組
def __init__(self,name,score):
self.name = name
self.score = score
def main():
he = Student('Bob',95)
he.age = 19 #AttributeError: 'Student' object has no attribute 'age'
print(he.age)
if __name__=="__main__":
main()
這樣,對(duì)象的屬性就限定在類(lèi)的內(nèi)部了。
但是__slots__不能被繼承。而且,__slots__的設(shè)計(jì)本意并不是上面的用法,而是創(chuàng)建大量(萬(wàn)計(jì))對(duì)象時(shí)對(duì)內(nèi)存占用進(jìn)行優(yōu)化。
總結(jié):
寫(xiě)著寫(xiě)著,我發(fā)覺(jué)上面的技巧意義不大。類(lèi)的設(shè)計(jì)是程序員本人,使用者也是本人,那么,對(duì)象屬性的
讀和寫(xiě)就應(yīng)該自己把握,類(lèi)設(shè)計(jì)時(shí)本身不需要太多的保護(hù)代碼,否則會(huì)很臃腫,而且效率降低。保護(hù)措施應(yīng)該發(fā)生在類(lèi)的外部,讓類(lèi)對(duì)象接受到的數(shù)據(jù)永遠(yuǎn)是合法的,這樣會(huì)更加輕巧靈活。這是我的感受。
以上這篇淺談python對(duì)象數(shù)據(jù)的讀寫(xiě)權(quán)限就是小編分享給大家的全部?jī)?nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
用Python實(shí)現(xiàn)通過(guò)哈希算法檢測(cè)圖片重復(fù)的教程
這篇文章主要介紹了用Python實(shí)現(xiàn)通過(guò)哈希算法檢測(cè)圖片重復(fù)的教程,這個(gè)方法被Iconfinder用作防盜版技術(shù),需要的朋友可以參考下2015-04-04
用python實(shí)現(xiàn)一幅春聯(lián)實(shí)例代碼
大家好,本篇文章主要講的是用python實(shí)現(xiàn)一幅春聯(lián)實(shí)例代碼,感興趣的同學(xué)趕快來(lái)看一看吧,對(duì)你有幫助的話(huà)記得收藏一下2022-01-01
Python命令行參數(shù)解析之a(chǎn)rgparse模塊詳解
這篇文章主要介紹了Python命令行參數(shù)解析之a(chǎn)rgparse模塊詳解,argparse?是?Python?的一個(gè)標(biāo)準(zhǔn)庫(kù),用于命令行參數(shù)的解析,這意味著我們無(wú)需在代碼中手動(dòng)為變量賦值,而是可以直接在命令行中向程序傳遞相應(yīng)的參數(shù),再由變量去讀取這些參數(shù),需要的朋友可以參考下2023-08-08
Zabbix實(shí)現(xiàn)微信報(bào)警功能
這篇文章主要介紹了Zabbix實(shí)現(xiàn)微信報(bào)警的相關(guān)資料,本文圖文并茂介紹的非常詳細(xì),需要的朋友可以參考下2016-10-10
Python利用xmltodict模塊實(shí)現(xiàn)處理XML數(shù)據(jù)
理解和處理XML數(shù)據(jù)在Python中是一項(xiàng)常見(jiàn)任務(wù),xmltodict便是一個(gè)Python庫(kù),用于將XML數(shù)據(jù)解析為易于處理的Python字典,下面我們就來(lái)學(xué)習(xí)一下xmltodict庫(kù)的具體使用吧2023-11-11
python爬蟲(chóng)爬取網(wǎng)頁(yè)數(shù)據(jù)并解析數(shù)據(jù)
這篇文章主要介紹了python爬蟲(chóng)如何爬取網(wǎng)頁(yè)數(shù)據(jù)并解析數(shù)據(jù),幫助大家更好的利用爬蟲(chóng)分析網(wǎng)頁(yè),感興趣的朋友可以了解下2020-09-09
Python學(xué)習(xí)筆記之字典,元組,布爾類(lèi)型和讀寫(xiě)文件
這篇文章主要為大家詳細(xì)介紹了Python的字典,元組,布爾類(lèi)型和讀寫(xiě)文件,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來(lái)幫助2022-02-02
在windows下快速搭建web.py開(kāi)發(fā)框架方法
這篇文章主要介紹了在windows下快速搭建web.py開(kāi)發(fā)框架方法,需要的朋友可以參考下2016-04-04
Python隨機(jī)數(shù)random模塊使用指南
本文給大家分享的是Python隨機(jī)數(shù)random模塊的幾個(gè)常用的方法,非常的簡(jiǎn)單,小伙伴們喜歡的話(huà),后續(xù)繼續(xù)深入探討2016-09-09

