
本文详解因缩进错误导致类方法不可访问的问题,通过修复 `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 = name
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? 关键注意事项:
立即学习“Python免费学习笔记(深入)”;
- ❌ 错误缩进(def apply_discount(...) 缩进在 __init__ 内)→ 方法仅在初始化时存在,且无法被实例调用;
- ✅ 正确缩进(与 def __init__ 左对齐)→ 方法绑定到类,自动接收 self,可被所有实例调用;
- Item.pay_rate 是类属性,推荐通过类名访问(而非 self.pay_rate),避免意外被实例覆盖;
- 若需支持不同商品差异化折扣,可改为实例属性或参数化设计(如 apply_discount(discount_rate=0.8))。
总结:Python 中“缩进即语法”,类中方法必须与 __init__ 同级定义。一次缩进修正,即可让折扣逻辑真正生效——这是面向对象基础却极易忽略的关键细节。










