Python 面向对象(类,对象,方法,属性,魔术方法)

devtools/2025/2/12 10:09:39/

前言:在讲对象>面向对象之前,我们先将面向过程和对象>面向对象进行一个简单的分析比较,这样我们可以更好的理解与区分,然后我们在详细的讲解对象>面向对象的优势。

        面向过程(Procedure-Oriented Programming,POP)和对象>面向对象(Object-Oriented Programming,OOP)是两种不同的编程范式,它们在编程理念、代码组织方式、可维护性等方面存在明显差异,下面为你详细介绍。

🥇面向过程

概念

面向过程是一种以过程为中心的编程范式,它将一个大的任务分解为一系列的步骤,每个步骤用一个函数来实现,程序的执行流程就是依次调用这些函数。重点在于分析解决问题所需的步骤,然后用函数把这些步骤一步一步实现。

特点
  • 强调功能分解:把一个复杂的问题拆分成多个简单的子问题,每个子问题对应一个函数,函数之间通过参数和返回值进行数据传递和交互。
  • 数据和操作分离:数据和处理数据的函数是分开的,函数可以独立于数据存在,不同的函数可以对相同的数据进行操作。
  • 执行流程明确:程序按照函数调用的顺序依次执行,执行流程清晰,易于理解和调试。
示例

下面是一个使用面向过程编程实现的简单加法程序:

python"># 定义一个函数用于执行加法操作
def add_numbers(a, b):return a + b# 定义一个函数用于显示结果
def display_result(result):print(f"两数相加的结果是: {result}")# 主程序
num1 = 5
num2 = 3
# 调用加法函数
sum_result = add_numbers(num1, num2)
# 调用显示结果函数
display_result(sum_result)

在这个例子中,我们将加法操作和结果显示操作分别封装在两个函数中,程序的执行流程就是依次调用这些函数,先计算加法,再显示结果。

适用场景

面向过程编程适用于处理简单、规模较小的问题,以及对执行效率要求较高的场景,因为它的执行流程简单直接,没有过多的抽象和封装。例如,一些脚本程序、算法实现等。

🥇对象>面向对象

概念

对象>面向对象是一种以对象为中心的编程范式,它将数据和操作数据的方法封装在一起,形成对象。通过定义来创建对象对象的模板,描述了对象的属性和行为。对象之间通过消息传递进行交互。

特点
  • 封装:将数据(属性)和操作数据的方法(方法)捆绑在一起,形成一个独立的单元,对外提供统一的接口,隐藏内部实现细节。这样可以提高代码的安全性和可维护性。
  • 继承:一个可以继承另一个属性和方法,被继承的称为父(基),继承的称为子(派生)。继承可以实现代码的复用和扩展,减少代码的重复编写。
  • 多态:不同的对象可以对同一消息做出不同的响应。通过方法重写和接口实现等方式,实现不同对象对相同方法的不同实现,提高代码的灵活性和可扩展性。
示例

下面是一个使用对象>面向对象编程实现的简单加法程序:

python">class Calculator:def __init__(self):passdef add(self, a, b):return a + bdef display_result(self, result):print(f"两数相加的结果是: {result}")# 创建 Calculator 对象
calc = Calculator()
num1 = 5
num2 = 3
# 调用对象的加法方法
sum_result = calc.add(num1, num2)
# 调用对象的显示结果方法
calc.display_result(sum_result)

在这个例子中,我们定义了一个 Calculator ,将加法操作和结果显示操作封装在的方法中。通过创建 Calculator 对象 calc,调用对象的方法来完成加法和结果显示的操作。

适用场景

对象>面向对象编程适用于处理复杂、规模较大的问题,以及需要高度复用和扩展的场景。例如,大型软件系统、游戏开发、图形用户界面(GUI)编程等。

对比总结(面向过程与对象>面向对象的区别)

  • 编程思想:面向过程关注的是解决问题的步骤和过程,而对象>面向对象关注的是对象及其之间的交互。
  • 代码组织:面向过程的代码以函数为中心,函数之间的调用关系构成程序的执行流程;对象>面向对象的代码以对象为中心,通过对象之间的消息传递来实现程序的功能。
  • 可维护性和可扩展性:面向过程的代码在处理复杂问题时,随着功能的增加,函数之间的调用关系会变得复杂,导致代码的可维护性和可扩展性降低;对象>面向对象通过封装、继承和多态等特性,可以更好地组织代码,提高代码的可维护性和可扩展性。

🥇对象>面向对象(简化):

再将对象>面向对象讲的简单一点,打个比方若在某游戏中设计一个乌龟的角色,应该如何来实现呢?使用对象>面向对象的思想会更简单,可以分为如下两个方面进行描述:

  1. 从表面特征来描述,例如,绿色的、有 4 条腿、重 10 kg、有外壳等等。
  2. 从所具有的的行为来描述,例如,它会爬、会吃东西、会睡觉、会将头和四肢缩到壳里。

如果将乌龟用代码来表示,则其表面特征可以用变量来表示,其行为特征可以通过建立各种函数来表示。参考代码如下所示:

python">class tortoise:bodyColor = "绿色"footNum = 4weight = 10hasShell = True#会爬def crawl(self):print("乌龟会爬")#会吃东西def eat(self):print("乌龟吃东西")#会睡觉def sleep(self):print("乌龟在睡觉")#会缩到壳里def protect(self):print("乌龟缩进了壳里")

因此,从某种程序上,相比较只用变量或只用函数,使用对象>面向对象的思想可以更好地模拟现实生活中的事物。
不仅如此,在 Python 中,所有的变量其实也都是对象,包括整形(int)、浮点型(float)、字符串(str)、列表(list)、元组(tuple)、字典(dict)和集合(set)。以字典(dict)为例,它包含多个函数供我们使用,例如使用 keys() 获取字典中所有的键,使用 values() 获取字典中所有的值,使用 item() 获取字典中所有的键值对,等等。

🥇对象>面向对象中的常用术语:

  • :可以理解是一个模板,通过它可以创建出无数个具体实例。比如,前面编写的 tortoise 表示的只是乌龟这个物种,通过它可以创建出无数个实例来代表各种不同特征的乌龟(这一过程又称为的实例化)。
  • 对象并不能直接使用,通过创建出的实例(又称对象)才能使用。这有点像汽车图纸和汽车的关系,图纸本身()并不能为人们使用,通过图纸创建出的一辆辆车(对象)才能使用。
  • 属性:中的所有变量称为属性。例如,tortoise 这个中,bodyColor、footNum、weight、hasShell 都是这个拥有的属性。
  • 方法:中的所有函数通常称为方法。不过,和函数所有不同的是,方法至少要包含一个 self 参数(后续会做详细介绍)。例如,tortoise 中,crawl()、eat()、sleep()、protect() 都是这个所拥有的方法,方法无法单独使用,只能和对象一起使用。

🥇对象的关系和定义:

是抽象的概念,是对象的模板,它定义了对象属性和方法对象是具体的实体,是的实例,具有所定义的属性和方法,并且每个对象都有自己独立的状态。对象相互依存,对象提供了统一的规范,对象则是的具体体现。

 对象的关系对象是抽象与具体、模板与实例的关系。是对一事物属性和行为的抽象描述,是创建对象的模板。对象的具体实例,依据创建,有独立状态。定义通用规则,对象遵循规则并展现具体特征与行为。一个可以创建多个对象,它们之间是独立的,互相不影响。

对象的定义:是一种用户自定义的数据型,它由数据和方法组成。数据表示属性,方法表示行为。一个可以包含多个属性和方法。属性是的成员变量,可以存储数据。方法是一组操作数据的代码,它们可以实现某些功能或者改变属性的值。对象>面向对象编程中的抽象概念,是对具有相同属性和方法的一组对象的描述,如同模板或蓝图,定义了对象应有的特征和行为规范。对象的具体实例,基于创建,有自己独特的属性值,能执行定义的方法,是抽象概念的具象体现。

的定义:

在Python中,我们可以有两种的定义方式:Python2(经典)和 Python3(新式
经典:不由任意内置型派生出的,称之为经典
python">class 名:# 属性# ⽅法

新式

python">class 名(object):# 属性# ⽅法

实例:基本语法

python">class Person(object):# 属性# ⽅法(函数)def eat(self):print('吃零⻝')def drink(self):print('喝可乐')

的实例化(创建对象):

的实例化就是把抽象的事务具体为现实世界中的实体。

的实例化就是通过得到对象

只是对象的⼀种规范,本身基本上什么都做不了,必须利⽤得到对象,这个过程就叫作的实例化!

python"># 基本语法
# 对象名 = 名()
# 1、定义⼀个
class Person(object):# 定义相关⽅法def eat(self):print('吃零⻝')def drink(self):print('喝可乐')# 2、实例化对象
p1 = Person()
# 3、调⽤中的⽅法
p1.eat()
p1.drink()
p2 = Person()

输出结果:

注意:在其他的编程语⾔中,的实例化⼀般是通过new关键字实例化⽣成的,但是在Python中,我们不需要new关键字,只需要名+()括号就代表的实例。

是⼀个抽象概念,在定义时,其并不会真正的占⽤计算机的内存空间。但是对象是⼀个具体的事务,所以其要占⽤计算机的内存空间。

 🥇中的self关键字:

定义

self 是一个约定俗成的参数名,在 Python 的实例方法中,它作为第一个参数出现。它代表的实例对象本身,当调用实例方法时,Python 会自动将调用该方法的对象作为第一个参数传递给 self

作用

访问实例属性

的方法中,可以使用 self 来访问和修改实例对象的属性。实例属性是每个对象独有的数据,通过 self 可以确保操作的是当前对象的属性。

python">class Person:def __init__(self, name, age):# 初始化实例属性self.name = nameself.age = agedef introduce(self):# 使用 self 访问实例属性print(f"我叫 {self.name},今年 {self.age} 岁。")# 创建 Person 的实例
p = Person("张三", 25)
p.introduce()  # 输出: 我叫 张三,今年 25 岁。

 在上述代码中,__init__ 方法通过 self.nameself.age 初始化了实例对象的属性,introduce 方法通过 self.nameself.age 访问了这些属性。

调用实例方法

的一个实例方法中,可以使用 self 来调用该对象的其他实例方法。

python">class Calculator:def __init__(self, num):self.num = numdef square(self):return self.num ** 2def double_square(self):# 使用 self 调用 square 方法return 2 * self.square()# 创建 Calculator 的实例
calc = Calculator(5)
result = calc.double_square()
print(result)  # 输出: 50
double_square 方法中,通过 self.square() 调用了 square 方法。
注意事项
命名约定

虽然 self 只是一个约定俗成的名称,并不是 Python 的关键字,但建议始终使用 self 作为实例方法的第一个参数名,这样可以提高代码的可读性和可维护性,让其他开发者更容易理解代码的含义。

自动传递

当调用实例方法时,不需要手动传递 self 参数,Python 会自动将调用该方法的对象作为 self 参数传递给方法。

🥇对象的属性添加与获取:

属性的基本概念:

在Python中,任何⼀个对象都应该由两部分组成:属性 + ⽅法

        属性即是特征,⽐如:

                                ⼈的姓名、年龄、身⾼、体重…都是对象的属性。

                                ⻋的品牌、型号、颜⾊、载重量...都是对象的属性。

对象属性既可以在外⾯添加和获取,也能在⾥⾯添加和获取。

的外面添加属性和获取属性:

在 Python 里,的实例对象可以动态地添加属性,也就是在定义之外为实例对象赋予新的属性。只需使用实例对象名,后跟点号(.)和新属性名,再通过赋值语句给该属性赋一个值即可。

示例代码:
python"># 定义一个简单的
class Person:def __init__(self, name):self.name = name# 创建一个 Person 的实例
person = Person("Alice")# 在的外面添加属性
person.age = 25
person.gender = "Female"# 获取属性
print(f"Name: {person.name}")
print(f"Age: {person.age}")
print(f"Gender: {person.gender}")
代码解释
  1. 定义 Person:在这个中,有一个 __init__ 方法,它在创建实例时被调用,会将传入的 name 参数赋值给实例的 name 属性。
  2. 创建实例:创建了一个 Person 的实例 person,并传入名字 "Alice"
  3. 添加属性:在的外面,通过 person.age = 25person.gender = "Female" 分别为 person 实例添加了 agegender 属性。
  4. 获取属性:使用 print 函数,通过 person.nameperson.ageperson.gender 获取并打印这些属性的值。
注意事项
  • 这种动态添加属性的方式只对当前实例对象有效,不会影响的其他实例。
  • 如果尝试访问一个不存在的属性,会引发 AttributeError 异常。因此,在访问属性之前,最好先检查属性是否存在,可以使用 hasattr() 函数进行判断。
的内部获取外部定义的属性:

代码示例:

python"># 1、定义⼀个Person
class Person():def speak(self):print(f'我的名字:{self.name},我的年龄:{self.age},我的住址:{self.address}')
# 2、实例化Person,⽣成p1对象
p1 = Person()
# 3、添加属性
p1.name = '孙悟空'
p1.age = 500
p1.address = '花果⼭⽔帘洞'
# 4、调⽤speak⽅法
p1.speak()

🥇魔术方法

魔术方法的基本概念:

在 Python 里,魔术方法(Magic Methods)也被叫做特殊方法(Special Methods),它们是一具有特殊命名规则的方法,名字以双下划线 __ 开头和结尾,例如 __init____str__ 等。这些方法在特定的场景下会被 Python 解释器自动调用,无需手动调用,能让开发者实现一些特殊的功能,极大增强了的功能和灵活性。

__init__() 当实例化对象时,其会动被触发(被调

__del__() 当手工删除对象对象被销毁时,其会动被触发(被调

基本概念要点

  1. 命名规则:双下划线开头和结尾是魔术方法的显著标识,这一命名方式是 Python 的约定,用于区分普通方法。
  2. 自动调用:Python 解释器会在特定操作发生时自动调用相应的魔术方法,而非像普通方法那样需要手动调用。
  3. 增强的功能:借助实现不同的魔术方法,能够让自定义模拟内置型的行为,或者实现特定的操作。

 🥇__init__() 方法(初始化方法或构造方法)

基本概念

__init__() 方法在创建的实例时会被 Python 解释器自动调用,主要用于对实例对象进行初始化操作,即给实例对象的属性赋初始值。该方法定义在的内部,其名称固定为 __init__,并且第一个参数必须是 self,它代表的实例对象本身。

语法结构
python">class ClassName:def __init__(self, parameter1, parameter2, ...):# 初始化操作self.attribute1 = parameter1self.attribute2 = parameter2...
代码示例
python">class Person:def __init__(self, name, age):# 将传入的 name 参数赋值给实例的 name 属性self.name = name# 将传入的 age 参数赋值给实例的 age 属性self.age = agedef introduce(self):# 打印实例的 name 和 age 属性print(f"我叫 {self.name},今年 {self.age} 岁。")# 创建 Person 的实例,同时传入初始化参数
p1 = Person("Alice", 25)
# 调用实例的 introduce 方法
p1.introduce()

在上述代码中:

  1. 定义了一个 Person ,其中包含 __init__ 方法和 introduce 方法。
  2. __init__ 方法接收两个参数 nameage,并将它们分别赋值给实例的 nameage 属性。
  3. 创建 Person 的实例 p1 时,传入了 "Alice"25 作为初始化参数,__init__ 方法会自动被调用,将这两个值赋给 p1nameage 属性。
  4. 调用 p1introduce 方法,打印出实例的 nameage 属性信息。
注意事项
  • self 参数__init__ 方法的第一个参数必须是 self,这是 Python 的规定。当创建实例时,Python 会自动将实例对象作为第一个参数传递给 __init__ 方法。
  • 可选参数__init__ 方法的参数可以是可选的,即可以为参数设置默认值。

🥇__str__() 方法

基本概念

__str__() 方法是对象的字符串表示方法,当使用内置的 str() 函数将对象转换为字符串,或者使用 print() 函数打印对象时,Python 解释器会自动调用该对象__str__() 方法。如果中没有定义 __str__() 方法,Python 会使用默认的字符串表示,通常是名和对象的内存地址。

语法结构
python">class ClassName:def __str__(self):# 返回对象的字符串表示return "对象的字符串描述"
代码示例
python">class Point:def __init__(self, x, y):self.x = xself.y = ydef __str__(self):# 返回对象的字符串表示return f"Point(x={self.x}, y={self.y})"# 创建 Point 的实例
p = Point(3, 4)# 直接打印对象
print(p)  # 输出: Point(x=3, y=4)

在这个示例中,Point 定义了 __str__() 方法,该方法返回一个格式化字符串,包含对象xy 属性值。当使用 print() 函数打印 p 对象时,Python 解释器会自动调用 __str__() 方法,并将返回的字符串输出,这样的输出对用户来说更加直观和易读。

使用场景
  • 调试和日志记录:在调试代码或记录日志时,__str__() 方法可以提供对象的关键信息,方便开发者快速了解对象的状态。
  • 用户交互:当需要向用户展示对象的信息时,__str__() 方法可以返回一个友好的字符串,提升用户体验。

🥇__del__() 方法(删除方法或析构方法)

基本概念

__del__() 方法会在对象的引用计数降为 0 且即将被垃圾回收机制销毁时自动调用。它的主要用途是在对象被销毁前执行一些清理工作,像关闭文件、释放网络连接、释放数据库连接等资源。

语法结构
python">class ClassName:def __del__(self):# 执行清理操作的代码print("对象即将被销毁,执行清理操作")
示例代码
python">class Person():# 构造函数__init__def __init__(self, name, age):self.name = nameself.age = age# 析构⽅法__del__def __del__(self):print(f'{self}对象已经被删除')
# 实例化对象
p1 = Person('⽩⻣精', 100)
# 删除对象p1
del p1
注意事项

引用计数和垃圾回收机制

  • Python 使用引用计数来跟踪对象的引用情况,当对象的引用计数降为 0 时,对象就会被标记为可回收。不过,垃圾回收机制并不是实时运行的,所以 __del__ 方法的调用时机可能会有延迟。
总结:

__init__() :初始化方法或者称之为“构造函数”,在对象初始化时执行,其主要作用就是在对象初 始化时,对对象进行初始化操作(如赋予属性)

__str__() :对象字符串方法,当我们在的外部,使用print方法输出对象时被触发,其主要功就

是对对象进行打印输出操作,要求方法必须使用return返回字符串格式的数据。

__del__() :删除方法或者称之为“析构方法”,在对象被删除时触发(调用del删除对象或文件执行 结束后),其主要作用就是适用于关闭文件、关闭数据库连接等等。

                 好啦对象>面向对象的基础讲到这里,下一篇就开始讲对象>面向对象的三大特征


http://www.ppmy.cn/devtools/158176.html

相关文章

React(三)

动态控制显示和css import { useState } from "react"; import "./index.css"; const list [{ id: 1, username: "aaName", content: "一条评论", ctime: "10-18 08:15" },{ id: 2, username: "bbName", conten…

第36天:安全开发-JavaEE应用第三方组件Log4j日志FastJson序列化JNDI注入

时间轴: 演示案例: Java-三方组件-Log4J&JNDI Java-三方组件-FastJson&反射 Maven的下载及配置: IDEA配置Maven的超详细步骤_java_脚本之家 Java-三方组件-Log4J&JNDI JNDI 注入: ( 见图 ) Java Naming and Dire…

算法15(力扣347)——前k个高频元素

1、问题 给你一个整数数组 nums 和一个整数 k ,请你返回其中出现频率前 k 高的元素。你可以按 任意顺序 返回答案。 2、示例 (1) 输入: nums [1,1,1,2,2,3], k 2 输出: [1,2] (2) 输入: nums [1], k 1 输出: [1…

https的论述

HTTPS(HyperText Transfer Protocol Secure)是HTTP协议的安全版本,用于保护数据传输的安全性和完整性。在HTTPS中,通过使用SSL(Secure Socket Layer)或TLS(Transport Layer Security&#xff09…

python2048游戏

实现了一个完整的2048游戏,并将其展示在一个图形化界面上。具体功能包括: 初始化游戏板:创建一个4x4的二维列表,表示游戏板,并在初始状态下随机放置两个数字(通常是2或4)。绘制游戏板&#xff…

8.flask+websocket

http是短连接,无状态的。 websocket是长连接,有状态的。 flask中使用websocket from flask import Flask, request import asyncio import json import time import websockets from threading import Thread from urllib.parse import urlparse, pars…

玩转工厂模式

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 什么是工厂模式?工厂方法模式适合应用场景实现方式工厂方法模式优缺点什么是工厂模式? 工厂方法模式是一种创建型设计模式,其在父类中提供一个创建对象的方法,允许子类决定实例化对象的类型。…

Visual Studio 使用 “Ctrl + /”键设置注释和取消注释

问题:在默认的Visual Studio中,选择单行代码后,按下Ctrl /键会将代码注释掉,但再次按下Ctrl /键时,会进行双重注释,这不是我们想要的。 实现效果:当按下Ctrl /键会将代码注释掉,…