news 2026/5/1 7:09:00

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/18 9:44:56

逻辑门组合电路设计:超详细版基础入门指南

从零开始学数字电路&#xff1a;用逻辑门搭建你的第一个组合电路你有没有想过&#xff0c;手机里每秒执行数十亿条指令的处理器&#xff0c;其实是由最简单的“开关”一步步搭起来的&#xff1f;这些“开关”不是物理按钮&#xff0c;而是我们今天要讲的主角——逻辑门。在嵌入…

作者头像 李华
网站建设 2026/4/28 5:03:48

电商市场的用户反馈分析与应用

电商市场的用户反馈分析与应用 关键词:电商市场、用户反馈分析、文本挖掘、情感分析、数据应用 摘要:本文聚焦于电商市场的用户反馈分析与应用。在电商行业竞争日益激烈的当下,用户反馈蕴含着巨大的价值。通过对用户反馈的深入分析,电商企业能够了解用户需求、改进产品与服…

作者头像 李华
网站建设 2026/5/1 3:49:31

电源管理芯片同步整流技术深度剖析其硬件实现

同步整流如何让电源效率“起飞”&#xff1f;——从MOSFET到PMIC的硬核拆解你有没有想过&#xff0c;为什么现在的手机充电越来越快、待机越来越久&#xff0c;而机身却还能越做越薄&#xff1f;背后的功臣之一&#xff0c;正是藏在主板深处、默默工作的电源管理芯片&#xff0…

作者头像 李华
网站建设 2026/4/30 3:07:17

vivado许可证与高级综合(HLS)工具集成要点

从代码到硅片&#xff1a;绕不开的 Vivado 许可证与 HLS 集成实战指南 你有没有遇到过这样的场景&#xff1f; 写好了一段 C 算法&#xff0c;信心满满地打开 Vivado HLS&#xff0c;准备一键综合成硬件 IP。结果刚运行 open_solution &#xff0c;控制台就跳出一行红字&am…

作者头像 李华
网站建设 2026/4/16 21:45:15

无源蜂鸣器驱动电路设计核心要点解析

无源蜂鸣器驱动电路设计&#xff1a;从原理到实战的完整指南在嵌入式系统开发中&#xff0c;声音提示早已不是“锦上添花”&#xff0c;而是人机交互的关键一环。无论是洗衣机完成洗涤时的一声“嘀”&#xff0c;还是智能门锁识别失败的连续警示音&#xff0c;背后都离不开一个…

作者头像 李华
网站建设 2026/4/30 11:11:39

基于Vivado的工业控制器烧写操作指南

Vivado烧写实战&#xff1a;让工业控制器真正“断电不丢魂” 你有没有遇到过这样的场景&#xff1f; 调试了整整一周的FPGA逻辑&#xff0c;终于在板子上跑通了EtherCAT主站协议&#xff0c;心跳信号稳定&#xff0c;IO响应及时。满心欢喜地拔掉JTAG线、断电重启——结果&…

作者头像 李华