面向对象三大特性是继承、封装、多态:继承通过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): ...就等于放弃多态。真正的多态是删掉这些判断,靠方法调用动态分发面向对象不是套模板,是帮你看清问题里的角色、关系和行为边界。写多了,你会自然觉得:这个该抽成父类,那个该藏成私有,这个接口留出来,以后换实现才不伤筋动骨。