news 2026/4/23 16:41:45

Day 16:【99天精通Python】面向对象编程(OOP)下篇 - 魔术方法与类属性

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Day 16:【99天精通Python】面向对象编程(OOP)下篇 - 魔术方法与类属性

Day 16:【99天精通Python】面向对象编程(OOP)下篇 - 魔术方法与类属性

前言

欢迎来到第16天!

在之前的两天里,我们构建了 OOP 的大厦框架。今天,我们要进行内部装修,学习一些 Python 特有的"黑魔法"。

你是否好奇过:

  • 为什么print(obj)打印出来的是一串看不懂的地址,而别人的对象打印出来是清晰的信息?
  • 为什么 Python 的+号既能算数字加法,又能拼接字符串,甚至还能合并列表?
  • 所有的属性都必须绑定在self上吗?有没有属于类本身的属性?

本节内容:

  • 类属性 vs 实例属性
  • 实例方法、类方法、静态方法
  • 魔术方法 (Magic Methods) 详解
  • 运算符重载 (__add__,__eq__等)
  • 实战练习

一、类属性 vs 实例属性

1.1 概念区别

  • 实例属性:定义在__init__中,使用self.xxx每个对象独有一份,互不干扰。
  • 类属性:定义在类内部,方法之外。所有对象共享同一份

1.2 使用场景

classTool:# 类属性:所有工具共享的计数器count=0def__init__(self,name):# 实例属性:每个工具的名字不同self.name=name# 每创建一个对象,类属性 count 加 1Tool.count+=1t1=Tool("锤子")t2=Tool("扳手")print(f"工具1:{t1.name}")# 锤子print(f"工具2:{t2.name}")# 扳手print(f"当前工具总数:{Tool.count}")# 2

注意:访问类属性建议直接用类名.属性名(如Tool.count)。虽然t1.count也能访问,但容易引起混淆。


二、三种方法:各司其职

Python 的类中可以定义三种方法,它们通过装饰器来区分。

2.1 实例方法 (Instance Method)

  • 默认的方法。
  • 第一个参数是self(指向对象)。
  • 最常用,用于操作实例属性。

2.2 类方法 (Class Method)

  • 使用@classmethod装饰。
  • 第一个参数是cls(指向类本身,而不是对象)。
  • 常用于:工厂模式(不通过__init__创建对象) 或操作类属性。

2.3 静态方法 (Static Method)

  • 使用@staticmethod装饰。
  • 没有selfcls参数。
  • 它就像是一个普通函数,只是单纯地寄生在类里面,通常作为辅助工具

2.4 代码对比

classDateUtil:def__init__(self,year,month,day):self.year=year self.month=month self.day=day# 1. 实例方法:打印日期defshow(self):print(f"{self.year}-{self.month}-{self.day}")# 2. 类方法:通过字符串创建对象 (工厂模式)@classmethoddeffrom_string(cls,date_str):# date_str 格式 "2026-01-01"y,m,d=map(int,date_str.split("-"))returncls(y,m,d)# 相当于 DateUtil(y, m, d)# 3. 静态方法:验证日期是否有效 (不需要访问实例或类属性)@staticmethoddefis_valid(date_str):return"-"indate_str# 使用# 静态方法检查ifDateUtil.is_valid("2026-05-20"):# 类方法创建对象d=DateUtil.from_string("2026-05-20")# 实例方法显示d.show()# 2026-5-20

三、魔术方法 (Magic Methods)

在 Python 中,以双下划线开头和结尾的方法被称为魔术方法(也叫 Dunder Methods)。它们会在特定时机被自动调用

__init__就是最常见的魔术方法。

3.1 __str__ 与 __repr__:让对象"说人话"

当你print(obj)时,默认打印的是内存地址。实现__str__可以自定义打印内容。

classPerson:def__init__(self,name,age):self.name=name self.age=age# 面向用户:print() 时调用def__str__(self):returnf"Person(name='{self.name}', age={self.age})"# 面向开发者:命令行交互时调用 (通常用来排查问题)def__repr__(self):returnf"<Person{self.name}>"p=Person("Alice",25)print(p)# Person(name='Alice', age=25)

3.2 运算符重载:让对象支持 + - * /

为什么 Python 的字符串和列表可以相加?因为它们实现了__add__方法。我们也可以让自定义对象支持运算符。

示例:二维向量相加

classVector:def__init__(self,x,y):self.x=x self.y=ydef__str__(self):returnf"Vector({self.x},{self.y})"# 实现 + 号运算def__add__(self,other):# 返回一个新的 Vector 对象returnVector(self.x+other.x,self.y+other.y)# 实现 == 号比较def__eq__(self,other):returnself.x==other.xandself.y==other.y v1=Vector(2,4)v2=Vector(3,1)v3=v1+v2# 自动调用 v1.__add__(v2)print(v3)# Vector(5, 5)print(v1==v2)# Falseprint(v1==Vector(2,4))# True

3.3 其他常用魔术方法

  • __len__(self): 响应len(obj)
  • __call__(self): 让对象能像函数一样被调用obj()
  • __getitem__(self, key): 让对象支持索引访问obj[key]

四、实战练习

练习1:购物车 (支持 len 和打印)

定义Cart类,内部用列表存储商品。

  1. add(item): 添加商品。
  2. 实现__len__: 返回商品数量。
  3. 实现__str__: 打印所有商品名称。
classCart:def__init__(self):self.items=[]defadd(self,item):self.items.append(item)def__len__(self):returnlen(self.items)def__str__(self):returnf"购物车包含:{', '.join(self.items)}"my_cart=Cart()my_cart.add("苹果")my_cart.add("牛奶")print(len(my_cart))# 2print(my_cart)# 购物车包含: 苹果, 牛奶

练习2:分数类 (Fraction)

实现一个简单的分数类,支持分数的相加。
例如:1/2 + 1/4 = 3/4

classFraction:def__init__(self,top,bottom):self.top=top# 分子self.bottom=bottom# 分母def__str__(self):returnf"{self.top}/{self.bottom}"def__add__(self,other):# 通分公式: a/b + c/d = (ad + bc) / bdnew_top=self.top*other.bottom+other.top*self.bottom new_bottom=self.bottom*other.bottomreturnFraction(new_top,new_bottom)f1=Fraction(1,2)f2=Fraction(1,4)print(f1+f2)# 6/8 (暂时不涉及约分逻辑)

五、OOP 总结与回顾

到今天为止,OOP 的核心内容就讲完了。让我们回顾一下:

OOP 体系

类与对象

class 定义

obj 实例

self 关键字

三大特性

保护数据

复用代码

灵活调用

高级特性

类属性 vs 实例属性

@classmethod...

str,add...


六、常见问题

Q1:__init____new__有什么区别?

  • __new__是真正创建对象实例的方法(构造器)。
  • __init__是对象创建好后初始化属性的方法(初始化器)。
  • 99% 的情况我们只需要写__init__,除非你在做单例模式或继承不可变类型。

Q2:什么时候用@staticmethod

当你写了一个方法,发现它既不需要访问self(实例属性),也不需要访问cls(类属性),但逻辑上又跟这个类有关联时,就把它定义为静态方法。


七、小结

OOP 进阶

属性类型

方法类型

魔术方法

实例属性 (self.x) - 独享

类属性 (Class.x) - 共享

实例方法 (self)

类方法 @classmethod (cls)

静态方法 @staticmethod

str(打印)

add(运算)

len(长度)

关键要点

  1. 想要所有对象共享数据?用类属性
  2. 想要自定义print()的显示效果?写__str__
  3. 想要让对象支持+运算?写__add__
  4. 区分self(对象) 和cls(类) 的使用场景。

八、课后作业

  1. 计数器类:创建一个Person类,每实例化一个对象,类属性population加 1。实现一个__del__方法(析构函数,对象销毁时调用),让population减 1。
  2. 时间类运算:完善MyTime类,包含hour,minute。实现__add__方法,支持两个时间相加(注意进位,如 1:30 + 1:40 = 3:10)。
  3. 单例模式(挑战题):查阅资料,尝试用__new__方法实现一个单例类(无论创建多少次,都只返回同一个对象)。

下节预告

Day 17:异常处理 (Try-Except)- 程序报错了怎么办?直接崩溃吗?不!我们要学会优雅地处理错误,让程序坚不可摧。


系列导航

  • 上一篇:Day 15 - 面向对象编程(中)
  • 下一篇:Day 17 - 异常处理(待更新)
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/23 13:47:36

SiFive平台移植RISC-V裸机程序从零实现指南

从零开始在 SiFive 平台运行 RISC-V 裸机程序&#xff1a;不只是“点灯”&#xff0c;而是真正理解底层启动机制你有没有试过&#xff0c;在一块全新的开发板上连一个 LED 都点不亮&#xff1f;不是代码写错了&#xff0c;也不是接线问题——而是程序根本没跑起来。这种情况在裸…

作者头像 李华
网站建设 2026/4/22 20:02:24

驱动程序安装方式对比:图形化vs命令行通俗解释

驱动安装的两种“语言”&#xff1a;图形界面 vs 命令行&#xff0c;你该用哪一种&#xff1f;你有没有遇到过这种情况——新买了一台打印机&#xff0c;插上电脑却提示“未识别设备”&#xff0c;于是你打开厂商官网&#xff0c;下载了一个.exe文件&#xff0c;双击运行&#…

作者头像 李华
网站建设 2026/4/22 3:23:18

Multisim安装教程:离线安装包部署方法详解

Multisim离线安装实战指南&#xff1a;无网络环境下的高效部署全解析 你有没有遇到过这样的场景&#xff1f;实验室电脑全部内网隔离&#xff0c;项目涉密不能联网&#xff0c;可偏偏急需安装 NI Multisim 做电路仿真——在线安装走不通&#xff0c;官方下载器卡死&#xff…

作者头像 李华
网站建设 2026/4/17 20:44:33

抖音娱乐直播行业中,为什么公认“最好的工会”是史莱克学院?

一、行业背景&#xff1a;娱乐直播进入“重运营、重安全感”时代随着抖音娱乐直播行业的成熟&#xff0c;主播与工会之间的关系&#xff0c;正在从“流量红利期”进入“长期合作期”。 行业开始更加关注以下核心问题&#xff1a; 工会是否具备真实的运营能力 是否存在合同风险与…

作者头像 李华
网站建设 2026/4/18 5:42:52

TTL电平转换芯片在驱动安装中的作用全面讲解

搞懂TTL电平转换芯片&#xff1a;为什么你的USB转串口总是连不上&#xff1f;你有没有遇到过这样的情况&#xff1a;手里的开发板明明接好了线&#xff0c;电脑也装了驱动&#xff0c;可设备管理器就是不认“COM口”&#xff0c;或者刚识别出来一会儿又掉线&#xff1f;串口调试…

作者头像 李华
网站建设 2026/4/23 8:59:08

基于USB转串口驱动的PLC通信方案:系统学习教程

如何用USB转串口稳定连接PLC&#xff1f;从芯片到代码的工业通信实战指南 在工厂自动化现场&#xff0c;你是否遇到过这样的场景&#xff1a;手里的新工控机连个RS-232接口都没有&#xff0c;而产线上的西门子S7-200或三菱FX系列PLC却只支持串口通信&#xff1f;面对这种“新电…

作者头像 李华