当前位置: 首页 > news >正文

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 - 异常处理(待更新)
http://www.jsqmd.com/news/231393/

相关文章:

  • 不同PWM频率下无源蜂鸣器声音效果对比分析
  • TI TPS系列在工业控制中的电源管理解决方案详解
  • 手把手教程:使用Verilog实现简单组合逻辑电路
  • 手把手教程:RISC-V指令集异常入口设置
  • 推荐Python、JavaScript或Scratch(儿童)。Python语法简洁,应用广泛;JavaScript适合
  • 温度传感器热响应时间研究:封装材料对动态性能的影响
  • ES索引分片策略设计:超详细版架构实践指南
  • 掌握 requests、BeautifulSoup 等库的网络爬虫基础,或使用 pandas 进行简单数据分析
  • 图解说明VHDL结构层次:顶层设计入门
  • 2026-01-12 全国各地响应最快的 BT Tracker 服务器(联通版)
  • 一文说清树莓派换源原理与常见问题解决方案
  • vivado2023.2下载安装超详细版:支持Win/Linux双平台
  • 安全继电器模块PCB原理图设计新手教程
  • 科技是把双刃剑ai到底是不是双刃剑
  • RabbitMQ 消息消费模式深度解析
  • 基于Web的模拟混频电路在线仿真操作指南
  • S8050三极管驱动LED灯时饱和状态判定:核心要点解析
  • 超详细版:Multisim搭建单级放大电路全过程
  • 模拟与数字混合电路板PCB设计的分区策略解析
  • 强电弱电混合布局:电路板PCB设计避坑指南
  • 驱动程序安装方式对比:图形化vs命令行通俗解释
  • 8位加法器Verilog实现通俗解释
  • 字符设备驱动内存管理最佳实践解析
  • Multisim14自定义虚拟仪器创建:从零开始教程
  • 多路选择器电路分析:数字电路实验一文说清
  • 蜂鸣器报警模块快速理解:核心要点与基础测试演示
  • HBuilderX安装与uni-app环境部署:新手手把手指导
  • HBuilderX中HTML5开发环境搭建:实战案例演示
  • 基于USB转串口驱动的PLC通信方案:系统学习教程
  • 为什么在抖音娱乐直播行业,公认“最好的工会”是史莱克学院