我正在努力尝试在创建类时选择要从其继承的父类而不影响任何子类。
有一个类的金字塔结构,在顶部,我尝试选择从中继承的位置:threading.Thread
或multiprocessing.Process
(在子类上指定一个参数)。
我仅在金字塔的第一层就创建了一个返回基类的函数,代码如下:
class ProcessClass(object):
'''A thread with different inheritance.
'''
def __new__(cls, *args, **kwargs):
thr_type = kwargs.pop("thr_type","threading")
print("HELLO", thr_type)
if(thr_type == "threading"):
new_cls = threading.Thread
if(thr_type == "multiprocessing"):
new_cls = multiprocessing.Process
instance = super(ProcessClass, cls).__new__(obtain_thread_class(new_cls))
instance.__init__(*args, **kwargs)
return instance
def obtain_thread_class(new_cls):
class BaseClass(new_cls):
"""Class with its methods"""
def __init__(self, daemon=False, *args, **kwargs):
# THREADING
super(BaseClass, self).__init__(*args, **kwargs)
self.daemon=daemon
def hello(self):
print("Hello World")
return BaseClass
但是现在,当我尝试在子类中继承ProcessClass
时,该类将失去其所有属性,并且无法正确传递参数(我认为这是由于__new__
而引起的)。 / p>
子示例类为:
class ExampleClass(ProcessClass):
def __init__(self, arg1, arg2, arg3="1", *args, **kwargs):
self.list_args = [arg1, arg2]
print("arg3 is", arg3)
super(ExampleClass, self)
def hello2(self):
print("Hello from child!")
有了这个,如果我执行:
a = ExampleClass(thr_type="threading")
print(type(a))
我获得:
HELLO threading
<class '__main__.obtain_thread_class.<locals>.BaseClass'>
但是,我只能打hello()
,不能打hello2()
:
a.hello()
Hello World
a.hello2()
Traceback (most recent call last):
File "<ipython-input-11-2505bb83de52>", line 1, in <module>
a.hello2()
AttributeError: 'BaseClass' object has no attribute 'hello2'
我认为使用__new__
会破坏孩子的创造力。
我也尝试过使用装饰器,并看到了__metaclasses__
的东西,但是在深入了解这些领域之前,我想知道是否有更简单的东西,因为我不是python的专家。
此外,我不希望此更改影响我已经在运行的任何程序(许多类都从ProcessClass
继承,但是目前仅threading.Thread
)。我正在尝试为整个程序保留此更改unnoticiable
。
答案 0 :(得分:1)
最后我选择了构图。没找到如何在运行时选择继承。但这仍然有效。
class ProcessClass(object):
'''A thread with different inheritance.
'''
def __init__(self, thr_type="multiprocessing", *args, **kwargs):
if(thr_type == "threading"):
new_cls = threading.Thread
if(thr_type == "multiprocessing"):
new_cls = multiprocessing.Process
self.thread = new_cls(*args, **kwargs)
def start(self):
return self.thread.start()
def run(self):
return self.thread.run()