如何在 Python 类中正确实现折扣方法

本文详解因缩进错误导致类方法不可访问的问题,通过修复 `apply_discount()` 方法的缩进层级,使其成为类的实例方法而非 `__init__` 内部函数,从而成功对商品价格应用折扣。

在 Python 面向对象编程中,方法的缩进层级直接决定其作用域和可访问性。原始代码中,apply_discount() 和 calculate_total_price() 被错误地缩进至 __init__ 方法内部,导致它们实际成为 __init__ 函数内的局部函数(即闭包),而非 Item 类的实例方法。因此,创建实例后调用 item1.apply_discount() 会触发 AttributeError: 'Item' object has no attribute 'apply_discount'。

✅ 正确做法是:将所有实例方法与 __init__ 并列对齐(同属类体层级),使用 4 空格缩进(PEP 8 推荐),确保它们被定义为类的成员方法。

以下是修复后的完整、可运行代码:

class Item:
    pay_rate = 0.8  # 全局类属性:统一 20% 折扣率

    def __init__(self, name: str, price: float, quantity=0):
        # 参数校验
        assert price >= 0, f"Price {price} is not greater than or equal to zero!"
        assert quantity >= 0, f"Quantity {quantity} is not greater than or equal to zero!"
        # 实例属性赋值
        self.name = nam

e self.price = price self.quantity = quantity def calculate_total_price(self) -> float: return self.price * self.quantity def apply_discount(self) -> None: self.price = self.price * Item.pay_rate # 使用类属性计算折扣后价格 # 使用示例 item1 = Item("Phone", 100.0, 1) print(f"原价: {item1.price}") # 输出: 100.0 item1.apply_discount() print(f"折后价: {item1.price}") # 输出: 80.0

? 关键注意事项:

  • ❌ 错误缩进(def apply_discount(...) 缩进在 __init__ 内)→ 方法仅在初始化时存在,且无法被实例调用;
  • ✅ 正确缩进(与 def __init__ 左对齐)→ 方法绑定到类,自动接收 self,可被所有实例调用;
  • Item.pay_rate 是类属性,推荐通过类名访问(而非 self.pay_rate),避免意外被实例覆盖;
  • 若需支持不同商品差异化折扣,可改为实例属性或参数化设计(如 apply_discount(discount_rate=0.8))。

总结:Python 中“缩进即语法”,类中方法必须与 __init__ 同级定义。一次缩进修正,即可让折扣逻辑真正生效——这是面向对象基础却极易忽略的关键细节。