面向对象三大特性是继承、封装、多态:继承通过class Dog(Animal)复用代码并扩展行为;封装用_和__约定保护成员,辅以getter/setter控制访问;多态通过同名方法不同实现实现解耦,无需类型判断。

类继承:让代码复用更自然
继承不是为了“炫技”,而是解决重复写相似逻辑的问题。比如你写了Animal类,有name和eat()方法;之后要写Dog和Cat,它们都有名字、都会吃,但狗会叫、猫会抓老鼠——这时就该让Dog和Cat继承Animal。
写法很简单:
- 在类定义括号里写上父类名:
class Dog(Animal): - 子类自动拥有父类的属性和方法,不用重写
name或eat() - 想扩展行为?直接在子类里加新方法,比如
bark();想修改已有行为?重写同名方法(如重写eat()),再用super().eat()调用父类版本
封装:把不该暴露的藏好,把该用的留好接口
封装不是“锁起来不让看”,而是明确告诉别人:“这个属性/方法是内部用的,别直接改;要用功能,请走我提供的方法”。Python没有private关键字,但约定俗成:
- 单下划线
_age:表示“建议别直接访问”,属于保护成员 - 双下划线
__salary:触发名称改写(变成_Employee__salary),实际是防误改,不是绝对私有 - 真正关键的是提供合理的接口:比如用
get_age()和set_age(new_age)控制年龄赋值逻辑(比如不能设负数)
别为了“封装”而加一堆getter/setter。只有当属性需要校验、计算、或未来可能变化时,才值得封装。
立即学习“Python免费学习笔记(深入)”;
多态:同一调用,不同表现,靠的是对象自己的类型
多态不是“写一堆if判断类型”,而是让不同类的对象响应同一个方法名,各自按自己的方式执行。前提是:它们有相同的方法名,且最好继承自同一个父类(或实现相同协议)。
- 例如,
Animal定义了make_sound(),Dog里写print("汪汪"),Cat里写print("喵喵") - 写个函数
play_with(animal),里面只调animal.make_sound()——传Dog()就汪汪,传Cat()就喵喵,函数本身完全不用知道具体类型 - 这就是多态的价值:解耦。你新增
Bird类,只要也实现make_sound(),原有play_with()函数立刻就能用,不用改一行
实战小提醒:别掉进常见坑
学完三大特性,动手时容易卡住。几个真实场景帮你避雷:
-
继承链太深:别搞
A → B → C → D → E。超过三层就该想想:是不是职责没分清?能不能用组合(has-a)代替继承(is-a)?比如Car有Engine,而不是Car继承Engine -
重写方法忘了调父类:比如子类
__init__里新增了self.breed,但没写super().__init__(name),结果父类的self.name根本没初始化 -
误以为多态需要强制类型检查:写
if isinstance(obj, Dog): ... elif isinstance(obj, Cat): ...就等于放弃多态。真正的多态是删掉这些判断,靠方法调用动态分发
面向对象不是套模板,是帮你看清问题里的角色、关系和行为边界。写多了,你会自然觉得:这个该抽成父类,那个该藏成私有,这个接口留出来,以后换实现才不伤筋动骨。











