How can I add properties to a class using a decorator that takes a list of names as argument?

2024/10/10 8:21:01

I would like to add many dummy-properties to a class via a decorator, like this:

def addAttrs(attr_names):def deco(cls):for attr_name in attr_names:def getAttr(self):return getattr(self, "_" + attr_name)def setAttr(self, value):setattr(self, "_" + attr_name, value)prop = property(getAttr, setAttr)setattr(cls, attr_name, prop)setattr(cls, "_" + attr_name, None) # Default value for that attributereturn clsreturn deco@addAttrs(['x', 'y'])
class MyClass(object):pass

Unfortunately, the decoarator seems to keep the reference of attr_name instead of its content. Therefore, MyClass.x and MyClass.y access both MyClass._y:

a = MyClass()
a.x = 5
print a._x, a._y
>>> None, 5
a.y = 8
print a._x, a._y
>>> None, 8

What do I have to change to get the expected behavior?

Answer

You almost had it working. There is just one nit. When creating the inner functions, bind the current value of attr_name into the getter and setter functions:

def addAttrs(attr_names):def deco(cls):for attr_name in attr_names:def getAttr(self, attr_name=attr_name):return getattr(self, "_" + attr_name)def setAttr(self, value, attr_name=attr_name):setattr(self, "_" + attr_name, value)prop = property(getAttr, setAttr)setattr(cls, attr_name, prop)setattr(cls, "_" + attr_name, None) # Default value for that attributereturn clsreturn deco@addAttrs(['x', 'y'])
class MyClass(object):pass

This produces the expected result:

>>> a = MyClass()
>>> a.x = 5
>>> print a._x, a._y
5 None
>>> a.y = 8
>>> print a._x, a._y
5 8

Hope this helps. Happy decorating :-)

https://en.xdnf.cn/q/69919.html

Related Q&A

How to reshape only last dimensions in numpy?

Suppose I have A of shape (...,96) and want to reshape it into (...,32,3) keeping both lengths and number of preceding dimensions, if ever, intact.How to do this?If I writenp.reshape(A, (-1, 32, 2))it…

Relative import of submodule

In Python, how do I perform the equivalent of the followingimport http.clientbut using a relative import:from . import http.client import .http.clientFor a package http in the current package? I want …

Python regular expression to replace everything but specific words

I am trying to do the following with a regular expression:import re x = re.compile([^(going)|^(you)]) # words to replace s = I am going home now, thank you. # string to modify print re.sub(x, _, s)T…

How do I raise a window that is minimized or covered with PyGObject?

Id been using the answer provided in the PyGTK FAQ, but that doesnt seem to work with PyGObject. For your convenience, here is a test case that works with PyGTK, and then a translated version that does…

How to bind multiple widgets with one bind in Tkinter?

I am wondering how to bind multiple widgets with one "bind".For expample: I have three buttons and I want to change their color after hovering.from Tkinter import *def SetColor(event):event.w…

Iterate a large .xz file line by line in python

I have a large .xz file (few gigabytes). Its full of plain text. I want to process the text to create custom dataset. I want to read it line by line because it is too big. Anyone have an idea how to do…

Detect multiple circles in an image

I am trying to detect the count of water pipes in this picture. For this, I am trying to use OpenCV and Python-based detection. The results, I am getting is a little confusing to me because the spread …

Need guidance with FilteredSelectMultiple widget

I am sorry if it question might turn to be little broad, but since I am just learning django (and I am just hobbyist developer) I need some guidance which, I hope, will help someone like me in the futu…

Django: determine which user is deleting when using post_delete signal

I want admins to be notified when certain objects are deleted but I also want to determine which user is performing the delete.Is it possible?This is the code:# models.py # signal to notify admins whe…

Double inheritance causes metaclass conflict

I use two django packages - django-mptt (utilities for implementing Modified Preorder Tree Traversal) and django-hvad (model translation).I have a model class MenuItem and I want to it extends Translat…