In python, is there a way to prevent adding new class variables after defining the object?
For example:
class foo:
def __init__(self):
self.a = 1
self.b = 2
self.c = 3
bar = foo()
try:
bar.d = 4
except Exception, e:
print "I want this to always print"
Alternatively, is there a way to count the number of variables in an object?
class foo:
def __init__(self):
self.a = 1
self.b = 2
self.c = 3
def count(self):
...
bar = foo()
if bar.count() == 3:
print "I want this to always print"
The only way I thought of doing this was using a dictionary or list:
class foo:
def __int__(self):
self.dict = {'foo':1, 'bar':2}
self.len = 2
def chk():
return self.len == len(self.list)
However, doing this feels rather cumbersome for python. (obj.dict[‘foo’]). I’d prefer just obj.foo if possible.
I want to have this so that I never accidentally declare a variable when I mean to change an existing one.
f = foo()
f.somename = 3
...
f.simename = 4 #this is a typo
if f.somename == 3:
solve_everything()
I suggest using
__setattr__to avoid the oddities of__slots__.You always have to be careful when messing with
__setattr__, since it takes care of setting all instance attributes, including those you set in__init__. Therefore it has to have some way of knowing when to allow the setting of an attribute, and when to deny it. In this solution I’ve designated a special attribute that controls whether new attributes are allowed or not:Testing:
Result:
Also see gnibblers answer that wraps this concept neatly up in a class decorator, so it doesn’t clutter up the class definition and can be reused in several classes without duplicating code.
EDIT:
Coming back to this answer a year later, I realize a context manager might solve this problem even better. Here’s a modified version of gnibbler’s class decorator:
And here’s how to use it:
So whenever you want to set new attributes, just use the
withstatement as above. It can also be done from outside the instance: