Ваш дизайн выглядит так, как будто вы делаете mixins . Вот пример игрушечной фабрики классов, которая может создавать миксины динамически.
figure.py
data = ((0, 1),(0, 3),(0, 4),(2, 1),(2, 3),(2, 7),(6, 3))
class FT:
@staticmethod
def ge():
return data
class FT1:
@staticmethod
def ge():
return [(x*x,y*y) for x,y in data]
def compose(ftype):
'''Returns a class composed of F and ftype.'''
return type(f'F_{ftype.__name__}',(F,ftype),{})
some_module.py:
import importlib
class F:
def __init__(self):
self.x = 'foo'
def a(self):
s = '|'.join(f'{thing}' for thing in self.ge())
return s
def b(self):
return 'baz'
def compose(ftype):
cls = getattr(importlib.import_module('figures'),ftype)
return type(f'F_{ftype}',(F,cls),{})
z = compose('FT')()
y = compose('FT1')()
Оба объекта имеют одинаковые b
методы.
>>> z.b(), y.b()
('baz', 'baz')
Оба имеют одинаковые a
методы, использующие данные из определенных ge
методов
>>> print(z.a())
(0, 1)|(0, 3)|(0, 4)|(2, 1)|(2, 3)|(2, 7)|(6, 3)
>>> y.a()
'(0, 1)|(0, 9)|(0, 16)|(4, 1)|(4, 9)|(4, 49)|(36, 9)'
Каждый объектимеет определенные ge
методы
>>> z.ge()
((0, 1), (0, 3), (0, 4), (2, 1), (2, 3), (2, 7), (6, 3))
>>> y.ge()
[(0, 1), (0, 9), (0, 16), (4, 1), (4, 9), (4, 49), (36, 9)]
>>>
z
и y
являются экземплярами различных классов.
>>> z.__class__, y.__class__
(<class '__main__.F_FT'>, <class '__main__.F_FT1'>)
>>>
Несколько экземпляров составного класса.
>>> One = compose(FT)
>>> q,r,s = One(),One(),One()
>>> q,r,s
(<__main__.F_FT object at 0x0000000003163860>,
<__main__.F_FT object at 0x000000000D334198>,
<__main__.F_FT object at 0x000000000D334828>)
>>>
Если все находится в одном модуле, тогда compose становится
def compose(ftype):
return type(f'F_{ftype.__name__}',(F,ftype),{})
z = compose(FT)
y = compose(FT1)
В чем разница между миксином и наследованием?
Аналогичное решение с использованием абстрактного базового класса. - G не может быть создан, пока ge
не будет переопределено.
import importlib
import abc
class G(abc.ABC):
def __init__(self):
self.x = 'foo'
def a(self):
s = '|'.join(f'{thing}' for thing in self.ge())
return s
def b(self):
return 'baz'
@staticmethod
@abc.abstractmethod
def ge():
pass
# either of these work
def new(ftype):
cls = getattr(importlib.import_module('figures'),ftype)
return type(cls.__name__,(cls,G),{})
#def new(ftype):
# cls = getattr(importlib.import_module('figures'),ftype)
# return type(cls.__name__,(G,),{'ge':staticmethod(cls.ge)})
#usage
# A = new('FT')
Аналогично, за исключением того, что конкретные методы static являются просто обычными функциями и используют ABC сверху
def FT_ge():
return ((0, 1),(0, 3),(0, 4),(2, 1),(2, 3),(2, 7),(6, 3))
def other_new(f):
return type(f.__name__.split('_')[0],(G,),{'ge':staticmethod(f)})
# usage
# B = other_new(FT_ge)