在Python中,基类、元类和抽象类是面向对象编程的重要概念。以下是对这三种类的介绍及其使用示例
1. 基类 (Base Class)
基类是其他类的父类,提供了共享的属性和方法。其他类可以继承基类,从而获得其功能。
继承的执行顺序
单一继承:当一个类继承自一个基类时,基类的构造函数(init)会在子类的构造函数之前被调用。
class Animal: # 基类
def __init__(self):
print("Animal initialized")
def speak(self):
return "Animal speaks"
class Dog(Animal): # Dog类继承自Animal类
def __init__(self):
super().__init__() # 调用基类构造函数
print("Dog initialized")
def speak(self):
return "Woof!"
class Cat(Animal): # Cat类继承自Animal类
def __init__(self):
super().__init__() # 调用基类构造函数
print("Cat initialized")
def speak(self):
return "Meow!"
# 使用示例
dog = Dog()
cat = Cat()
print(dog.speak()) # 输出: Woof!
print(cat.speak()) # 输出: Meow!
多重继承:在多重继承中,Python使用一种称为C3线性化(C3 Linearization)的算法来确定方法解析顺序(MRO,Method Resolution Order)。
MRO决定了在调用方法时,Python将按照什么顺序查找类及其基类。
class A:
def method(self):
print("Method from A")
class B(A):
def method(self):
print("Method from B")
super().method() # 调用A的方法
class C(A):
def method(self):
print("Method from C")
super().method() # 调用A的方法
class D(B, C): # D类同时继承B和C
def method(self):
print("Method from D")
super().method() # 调用B的方法
d = D()
d.method()
# 输出:
# Method from D
# Method from B
# Method from C
# Method from A
查看MRO:可以使用ClassName.__mro__或ClassName.mro()来查看类的MRO。
print(D.__mro__)
# 输出:
# (<class '__main__.D'>, <class '__main__.B'>, <class '__main__.C'>, <class '__main__.A'>, <class 'object'>)
2. 元类 (Metaclass)
元类是用于创建类的类。通过元类,可以控制类的创建过程,修改类的属性和方法
class Meta(type): # 定义一个元类
def __new__(cls, name, bases, attrs):
# 添加一个新的方法
def new_method(self):
return "This is a new method added by the metaclass."
attrs["new_method"] = new_method
# 添加一个只读属性
attrs["readonly_attr"] = property(lambda self: "This is read-only")
# 添加一个欢迎信息
attrs["greeting"] = "Hello"
# 添加一个类方法
@classmethod
def class_method(cls):
return "This is a class method added by the metaclass."
attrs["class_method"] = class_method
# 添加一个静态方法
@staticmethod
def static_method():
return "This is a static method added by the metaclass."
attrs["static_method"] = static_method
return super().__new__(cls, name, bases, attrs)
class MyClass(metaclass=Meta): # 使用Meta作为MyClass的元类
pass
# 使用示例
obj = MyClass()
print(obj.greeting) # 输出: Hello
print(obj.new_method()) # 输出: This is a new method added by the metaclass.
print(obj.readonly_attr) # 输出: This is read-only
# obj.readonly_attr = "Hi" # AttributeError: can't set attribute,无法修改
# 调用类方法
print(MyClass.class_method()) # 输出: This is a class method added by the metaclass.
# 调用静态方法
print(MyClass.static_method()) # 输出: This is a static method added by the metaclass.
# 在日常中,元类可以用于<例如django orm>:
# 1. 自动注册类:在定义类时,使用元类可以自动将类注册到某个注册表中。
# 2. 类的自动配置:元类可以在类定义时自动配置类的某些属性或方法。
# 3. 类的验证:元类可以在类定义时验证类的定义是否符合某些规则。
# 4. 类的修改:元类可以在类定义时修改类的定义,例如添加或修改方法。
# 5. 类的继承控制:元类可以控制类的继承关系,例如限制某些类不能被继承。
3. 抽象类 (Abstract Class)
抽象类是不能被实例化的类,通常包含一个或多个抽象方法(没有实现的方法)。它们用于定义接口,子类必须实现这些抽象方法。
from abc import ABC, abstractmethod
class PaymentGateway(ABC): # 抽象类
@abstractmethod
def pay(self, amount):
pass
@abstractmethod
def refund(self, amount):
pass
class Alipay(PaymentGateway): # Alipay类继承自PaymentGateway类
def __init__(self, account):
self.account = account
def pay(self, amount):
print(f"使用支付宝账号{self.account}支付{amount}元")
def refund(self, amount):
print(f"使用支付宝账号{self.account}退款{amount}元")
class WechatPay(PaymentGateway): # WechatPay类继承自PaymentGateway类
def __init__(self, account):
self.account = account
def pay(self, amount):
print(f"使用微信账号{self.account}支付{amount}元")
def refund(self, amount):
print(f"使用微信账号{self.account}退款{amount}元")
class BankTransfer(PaymentGateway): # BankTransfer类继承自PaymentGateway类
def __init__(self, account):
self.account = account
def pay(self, amount):
print(f"使用银行账号{self.account}支付{amount}元")
def refund(self, amount):
print(f"使用银行账号{self.account}退款{amount}元")
# 使用示例
alipay = Alipay("1234567890")
wechatpay = WechatPay("wechat123")
banktransfer = BankTransfer("1234567890")
orders = [
{"amount": 100, "payment_gateway": alipay},
{"amount": 200, "payment_gateway": wechatpay},
{"amount": 300, "payment_gateway": banktransfer},
]
for order in orders:
order["payment_gateway"].pay(order["amount"])
order["payment_gateway"].refund(order["amount"])
# 使用支付宝账号1234567890支付100元
# 使用支付宝账号1234567890退款100元
# 使用微信账号wechat123支付200元
# 使用微信账号wechat123退款200元
# 使用银行账号1234567890支付300元
# 使用银行账号1234567890退款300元