Python __slots__限制动态添加变量
Python是一種非常靈活的動(dòng)態(tài)語言,有時(shí)感覺太靈活以至于不知道遵循什么樣的規(guī)則去駕馭。不過Python已經(jīng)是非常完備的語言,想實(shí)現(xiàn)什么樣的功能都是有方法的,而且也很容易,比如限制一個(gè)類動(dòng)態(tài)添加成員變量。
一般情況下,我們定義完一個(gè)類,如果不加任何限制,還可以動(dòng)態(tài)地為該類的對(duì)象或該類添加成員變量。
class Employee:def __init__(self,name=''):self.name = nameif __name__ == "__main__":try:employee1 = Employee('Bob')#動(dòng)態(tài)為一個(gè)對(duì)象添加成員變量employee1.tel = '11111111'print(employee1.name,employee1.tel)employee2 = Employee('Tom')#employee2對(duì)象沒有tel成員變量print(employee2.name,employee2.tel)except AttributeError as e:print(e)#動(dòng)態(tài)為一個(gè)類添加成員變量Employee.tel = []print(employee2.name,employee2.tel)#Bob 11111111 #'Employee' object has no attribute 'tel' #Tom []看起來很方便,不過如果我們?nèi)绻幌胧褂谜咂茐念惖慕Y(jié)構(gòu)、隨便添加成員變量,要怎么做呢?
答案是,可以使用__slots__對(duì)象。
在類中,__slots__是一個(gè)以元組形式被定義的,只有在元組里的屬性,才可以被賦值,不在元組里的屬性賦值時(shí)會(huì)報(bào)錯(cuò)。
class Employee:__slots__ = ('name')def __init__(self,name=''):self.name = nameif __name__ == "__main__":employee1 = Employee('Bob')#動(dòng)態(tài)為一個(gè)對(duì)象添加成員變量,但不在__slots__定義的元組內(nèi),會(huì)報(bào)異常employee1.tel = '11111111'print(employee1.name,employee1.tel)#AttributeError: 'Employee' object has no attribute 'tel'__solts__不能被子類繼續(xù),如果想限制子類的成員變量,子類也要定義__slots__變量,同時(shí)會(huì)繼承父類的__slots__
''' 遇到問題沒人解答?小編創(chuàng)建了一個(gè)Python學(xué)習(xí)交流QQ群:778463939 尋找有志同道合的小伙伴,互幫互助,群里還有不錯(cuò)的視頻學(xué)習(xí)教程和PDF電子書! ''' class Employee:__slots__ = ('name')def __init__(self,name=''):self.name = nameclass Manager1(Employee):pass class Manager2(Employee):__slots__ = ('addr')if __name__ == "__main__":manager1 = Manager1('Bill')#動(dòng)態(tài)為一個(gè)對(duì)象添加成員變量manager1.tel = '22222222'print(manager1.name,manager1.tel)manager2 = Manager2()manager2.name = 'David'manager2.addr = 'BJ'print(manager2.name,manager2.addr)#動(dòng)態(tài)為一個(gè)對(duì)象添加成員變量,不在__slots__里,會(huì)報(bào)異常manager2.tel = '33333333'print(manager2.tel)#Bill 22222222 #David BJ #AttributeError: 'Manager2' object has no attribute 'tel'如果想進(jìn)一步限制對(duì)成員變量的使用,可以重載__setattr__, __getattr__,__getattribute__函數(shù),__setattr__函數(shù)可以在賦值前攔截;__getattr__在找不到屬性時(shí)被調(diào)用;__getattribute__則在獲取屬性值時(shí)無條件被調(diào)用,同時(shí)__getattr__不再被調(diào)用。注意不要在__getattribute__里使用self.attr來訪問變量,這會(huì)導(dǎo)致無限遞歸循環(huán)。
class Employee:__slots__ = ('name')def __init__(self,name=''):self.name = nameclass Manager2(Employee):__slots__ = ('addr')def __setattr__(self,name,value):if name == 'tel':raise AttributeError('manager has no tel')object.__setattr__(self, name, value)def __getattr__(self,name):if name == 'tel':return 0object.__getattribute__(self, name) if __name__ == "__main__":manager2 = Manager2('Jone')manager2.name = 'David'manager2.addr = 'BJ'try:manager2.tel = '11111111'except Exception as e:print(e)print(manager2.name,manager2.addr,manager2.tel)#manager has no tel #David BJ 0總結(jié)
以上是生活随笔為你收集整理的Python __slots__限制动态添加变量的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: python3 递归函数的理解
- 下一篇: websocket python爬虫_p