Python类为何继承object(python类继承可以有多个父类)
技术背景
在Python编程中,我们常常会看到类似class MyClass(object):的类声明方式。这涉及到Python中“旧风格类”(old-style classes)和“新风格类”(new-style classes)的概念。Python在2.2版本引入了新风格类,它对旧风格类进行了改进。旧风格类在Python 2.x时代存在一些缺陷,而新风格类通过继承object类来解决这些问题。在Python 3.x中,情况变得更加简单,所有类默认都是新风格类。
实现步骤
Python 2.x
在Python 2.x中,根据是否继承object类,存在两种风格的类:
- 旧风格类:不继承object类。
class ClassicSpam:
pass
print(ClassicSpam.__bases__) # 输出: ()
- 新风格类:直接或间接继承object类。
class NewSpam(object):
pass
print(NewSpam.__bases__) # 输出: (<type 'object'>,)
class IntSpam(int):
pass
print(IntSpam.__bases__) # 输出: (<type 'int'>,)
print(IntSpam.__bases__[0].__bases__) # 输出: (<type 'object'>,)
Python 3.x
在Python 3.x中,所有类默认都是新风格类,以下三种类声明方式是等价的:
class ClassicSpam:
pass
class NewSpam(object):
pass
class Spam():
pass
print([object in cls.__bases__ for cls in {Spam, NewSpam, ClassicSpam}]) # 输出: [True, True, True]
核心代码
以下是一些展示新风格类特性的代码示例:
classmethod和 staticmethod
class MyClass:
@classmethod
def class_method(cls):
print(f"Class method called on {cls.__name__}")
@staticmethod
def static_method():
print("Static method called")
MyClass.class_method() # 输出: Class method called on MyClass
MyClass.static_method() # 输出: Static method called
property
class MyClass:
def __init__(self):
self._value = 0
@property
def value(self):
return self._value
@value.setter
def value(self, new_value):
self._value = new_value
obj = MyClass()
print(obj.value) # 输出: 0
obj.value = 10
print(obj.value) # 输出: 10
__slots__
class MyClass:
__slots__ = ['value']
def __init__(self):
self.value = 0
obj = MyClass()
# 尝试添加不在 __slots__ 中的属性会报错
# obj.new_attr = 1 # 会引发 AttributeError
最佳实践
- Python 2.x:总是显式地继承object类,以获得新风格类的优点。
class MyClass(object):
pass
- Python 3.x:如果代码需要同时兼容Python 2和Python 3,可以继承object类;否则,不需要显式继承,因为Python会在背后自动处理。
# 兼容Python 2和3
class MyClass(object):
pass
# 仅用于Python 3
class MyClass:
pass
常见问题
旧风格类和新风格类有哪些具体区别?
旧风格类不支持描述符(如classmethod、staticmethod、property)、__new__静态方法、新的方法解析顺序(MRO)和super调用等特性。而新风格类则具备这些功能。
在Python 3中不继承object类会有问题吗?
在Python 3中,不继承object类不会有问题,因为所有类默认都是新风格类。但如果代码需要兼容Python 2,建议显式继承object类。
新风格类有什么缺点?
新风格类的类对象本身会占用更多的内存。不过,除非创建大量的类对象,否则这通常不是一个问题。