О метаклассе в Python
Я пытаюсь сделать класс в Python (с XSI / Softimage), чтобы переопределить методы по умолчанию.
class transform(object):
def __init__ (self) :
self.object = self._build()
self.type = ''
def _build (self):
object = None
return object
@property
def name(self):
name = xsi.getValue(str(self.object) + '.Name')
return str(name)
@name.setter
def name(self, value):
name = xsi.setValue(str(self.object) + '.Name', value)
self.object = str(name)
################## TRANSLATE ######################
@property
def tx(self):
tx = xsi.getValue(str(self.object) + '.kine.local.posx')
return tx
@tx.setter
def tx(self, value):
tx = xsi.setValue(str(self.object) + '.kine.local.posx', value)
@property
def ty(self):
ty = xsi.getValue(str(self.object) + '.kine.local.posy')
return ty
@ty.setter
def ty(self, value):
ty = xsi.setValue(str(self.object) + '.kine.local.posy', value)
@property
def tz(self):
tz = xsi.getValue(str(self.object) + '.kine.local.posz')
return tz
@tz.setter
def tz(self, value):
tz = xsi.setValue(str(self.object) + '.kine.local.posz', value)
Но, как вы видите, я много дублирую. Как я могу упростить это? Может быть с метаклассом?
2 ответа
Решение
property
type - это всего лишь один (удобный) ярлык для дескрипторов. Для вашего случая самое простое решение - это пользовательский дескриптор, а именно:
class XsiDescriptor(object):
def __init__(self, xsi):
self.xsi = xsi
def __get__(self, instance, cls=None):
if instance is None:
return self
key = "%s.%s" % (instance.object, self.xsi)
return xsi.getValue(key)
def __set__(self, instance, value):
key = "%s.%s" % (instance.object, self.xsi)
xsi.setValue(key, value)
class Transform(object):
# your init etc here
tx = XsiDescriptor("kine.local.posx")
ty = XsiDescriptor("kine.local.posy")
# etc
Вам не нужен метакласс здесь. Вы можете сделать что-то вроде этого:
def make_xsi_property(name):
def get_prop(self):
return xsi.getValue('{}.{}'.format(self.object, name))
def set_prop(self, value):
tx = xsi.setValue('{}.{}'.format(self.object, name), value)
return property(get_prop, set_prop)
class MyClass(object):
tx = make_xsi_property('kine.local.posx')
ty = make_xsi_property('kine.local.posy')
#...