Python教程(23)——Python类中常用的特殊成员

在Python中,类特殊成员是指以双下划线开头和结尾的属性和方法,也被称为魔术方法(Magic methods)或特殊方法(Special methods)。这些特殊成员在类的定义中具有特殊的语法和功能,用于实现对象的特定行为和操作。

特殊方法一般由Python解释器调用,无需手动调用。通过在类中定义这些特殊方法,可以改变对象的默认行为,使其具备更多的功能和操作。特殊方法提供了一种更加Pythonic的面向对象编程的方式,可以让代码更加简洁和易读。
在这里插入图片描述

__init__

__init__ 是Python中的一个特殊方法,也被称为构造方法。它在创建对象时自动调用,用于初始化对象的属性。通过在类中定义 __init__ 方法,我们可以在对象创建时为其赋予初始状态,设置属性的默认值,或执行其他必要的初始化操作。

__init__ 方法的格式通常如下:

def __init__(self, arg1, arg2, ...):
    self.attribute1 = arg1
    self.attribute2 = arg2
    # ...

self 代表对象本身,arg1arg2 等是用于接收传入的参数的形参,我们可以根据实际需要为 __init__ 方法传入不同的参数。

class Person:
    def __init__(self, name, age):
        self.name = name
        self.age = age

person1 = Person("Alittle", 30)

__init__ 方法中,我们可以执行其他的初始化操作,例如连接数据库、打开文件、设置默认值等。这使得我们能够在创建对象时,做一些必要的准备工作,确保对象在初始化后即可用。

需要注意的是,Python 中的 __init__ 方法是可选的,不是必须定义的。如果类中没有定义 __init__ 方法,Python 会使用默认的空的 __init__ 方法。但通常情况下,我们会定义 __init__ 方法来初始化对象的属性。

__del__

__del__ 是Python中的一个特殊方法,也被称为析构方法。它在对象被销毁前自动调用,用于进行清理工作。当对象的引用计数为零时(即没有任何变量引用该对象),Python解释器会自动触发 __del__ 方法的调用。

__del__ 方法的格式通常如下:

def __del__(self):
    # 清理代码

__del__ 方法中,self 代表对象本身。我们可以在 __del__ 方法中编写需要在对象销毁之前执行的清理代码,例如关闭文件、释放资源、记录日志等。

class FileHandler:
    def __init__(self, filename):
        self.file = open(filename, 'w')

    def write_data(self, data):
        self.file.write(data)

    def __del__(self):
        self.file.close()

handler = FileHandler("data.txt")
handler.write_data("Hello, World!")
# 在退出程序或手动执行 del handler 时,会自动调用 __del__ 方法关闭文件

我们定义了一个 FileHandler 类,其中的 __init__ 方法用于打开文件并创建一个 file 对象。write_data 方法用于向文件写入数据。在 __del__ 方法中,我们通过 self.file.close() 关闭了文件。当程序退出或手动执行 del handler 时,对象 handler 被销毁,会自动调用 __del__ 方法关闭文件,确保资源的正常释放。

另外需要注意的是,__del__ 方法的调用是由Python解释器自动控制的,并不是我们手动调用的。因此,我们不能依赖 __del__ 方法来进行关键性的清理工作,如释放内存。一般来说,Python有自己的垃圾回收机制,会自动管理对象的内存释放。我们只需确保在 __del__ 方法中执行一些简单的清理操作即可。此外,应尽量避免在 __del__ 方法中抛出异常,以免影响其他代码的执行。

__str__

__str__ 也是Python中的一个特殊方法,也被称为字符串表示方法。它定义了当我们对一个对象使用内置的 str() 函数或 print() 函数进行输出时,应该返回的字符串形式表示。简而言之,__str__ 方法用于定制对象的字符串输出。

__str__ 方法的格式通常如下:

def __str__(self):
    # 返回表示对象的字符串

__str__ 方法中,self 代表对象本身,没有其他的参数了,我们可以在该方法中编写需要返回的表示对象的字符串形式的代码,例如组织对象的属性信息、状态等。

以下是一个示例,展示了如何使用 __str__ 方法来定制对象的字符串输出:

class Point:
    def __init__(self, x, y):
        self.x = x
        self.y = y

    def __str__(self):
        print("call __str__")
        return f"Point({self.x}, {self.y})"

p = Point(3, 4)
print(p)
print("-----------------")
str(p)

上面的示例中的输出

call __str__
Point(3, 4)
-----------------
call __str__

我们定义了一个 Point 类,其中的 __init__ 方法用于初始化点的坐标。在 __str__ 方法中,我们使用格式化字符串 f-string 将点的坐标表示为 ‘(x, y)’ 的形式。当我们对 p 对象使用 print(p)str(p) 时,会自动调用 __str__ 方法,并返回该方法中定义的字符串 (3, 4)

__repr__

__repr____str__ 方法类似,下面直接用一个例子来说明这个和 __str__ 的区别。

class Point:
    def __init__(self, x, y):
        self.x = x
        self.y = y

    def __str__(self):
        print("call __str__")
        return f"Point1({self.x}, {self.y})"

    def __repr__(self):
        print("call __repr__")
        return f"Point2({self.x}, {self.y})"

p = Point(3, 4)
print(p)
print("-----------------")
str(p)
repr(p)

上面的示例中的输出

call __str__
Point1(3, 4)
-----------------
call __str__
call __repr__

从上面的输出可以看出,repr(p) 函数可以自动调用 __repr__,默认情况下,__str____repr__ 同时存在的话,str()print()函数就会优先调用 __str__,而如果只有 __repr__ 的话,就会调用 __repr__,不信你们可以把上面例子中的 __str__ 注释掉,然后看看输出情况。

__len__

__len__ 用于定义对象的长度。它主要被用于内置函数 len() 的操作,用于返回一个对象的长度或元素的个数。

__len__ 方法的格式通常如下:

def __len__(self):
    # 返回对象的长度或元素的个数

__len__ 方法中,self 代表对象本身。我们可以在该方法中编写代码,返回一个整数,表示对象的长度或元素的个数。

以下是一个示例,展示了如何使用 __len__ 方法来定义一个自定义的容器类并使用 len() 函数获取其长度:

class MyContainer:
    def __init__(self):
        self.data = []

    def __len__(self):
        return len(self.data)

    def add(self, element):
        self.data.append(element)

container = MyContainer()
container.add(1)
container.add(2)
container.add(3)

print(len(container))  # 输出 3,调用了 __len__ 方法

在上面的示例中,我们定义了一个名为 MyContainer 的容器类,该类包含一个名为 data 的列表用于存储元素。在 __len__ 方法中,我们使用内置函数 len() 计算了 data 列表的长度,并返回该长度。当我们通过调用 len(container) 来获取 container 对象的长度时,实际上会自动调用 __len__ 方法,并返回该方法中定义的长度值。

__len__ 方法应该返回一个整数,表示对象的长度或元素的个数。如果一个类没有定义 __len__ 方法,或者 __len__ 方法返回的值不是整数类型,那么调用 len() 函数时会抛出 TypeError 异常。

__getitem__和__setitem__

__getitem____setitem__ 几乎都是成对出现的,__getitem__ 用于定义对象的索引操作,即允许通过索引值访问对象的元素,__setitem__ 用于定义对象的赋值操作,即允许通过索引值设置对象的元素值,它主要用于支持下标操作和切片操作。

__getitem____setitem__ 方法的格式通常如下:

def __getitem__(self, index):
    # 返回指定索引位置的元素

def __setitem__(self, index, value):
    # 设置指定索引位置的元素为指定的值

这两个方法一个就是取值,一个就是赋值,主要就是应用在对象上面,相对而言比较好理解。

class MyList:
    def __init__(self):
        self.data = []

    def __getitem__(self, index):
        return self.data[index]

    def __setitem__(self, index, value):
        self.data[index] = value

mylist = MyList()
mylist.data = [1, 2, 3, 4, 5]

print(mylist[2])  # 输出 3,调用了 __getitem__ 方法
mylist[2] = 10  # 调用了 __setitem__ 方法赋值
print(mylist[2])  # 输出 10,调用了 __getitem__ 方法

在上面的示例中,我们定义了一个名为 MyList 的列表类,该类包含一个名为 data 的列表用于存储元素。通过实现 __getitem____setitem__方法,我们可以使用类似于列表的方式通过索引来访问和设置 data 列表中的元素。
__getitem__ 方法通常与 __setitem__ 方法一起使用,以支持对象的索引和切片操作。通过定义这些方法,我们可以使自定义的类对象能够像内置的容器类型一样进行元素的访问和修改。

如果一个类没有定义 __getitem____setitem__方法,或者__getitem____setitem__方法不能处理给定的索引值或切片对象,那么当我们尝试通过索引或切片来访问(设置)对象时,会抛出 TypeError 异常。

__call__

__call__ 用于使对象能够像函数一样被调用。通过定义 __call__ 方法,我们可以将一个对象变成一个可调用的实例,类似于函数的行为。

__call__ 方法的格式通常如下:

def __call__(self, *args, **kwargs):
    # 定义对象的调用逻辑

__call__ 方法中,self 代表对象本身,argskwargs 是传递给对象调用时的参数。

当我们像调函数一样使用对象时,Python 解释会自动调用对象的 __call__ 方法,并将传入的参数作为参数递给该方法。我们可以在 __call__ 方法中定义对象的调用逻,然后执行相应的操作。

以下一个示例,展示了如何使用 __call__ 方法来定义一个可调用的对象:

class Adder:
    def __call__(self, x, y):
        return x + y

add = Adder()

# 将对象 add 当作函数进行调用
result = add(3, 5)
print(result)  # 输出:8

在上述示例中,我们定义了一个 Adder 类,并实现了 __call__ 方法。在该方法中,我们将传入的两个参数进行相加,Python 解释器会自动调用对象 add__call__ 方法,并将传递给该方法。__call__ 方法中的逻辑会被执行,参数进行相加操作,然后返回结果。

通过使用 __call__ 方法,我们可以将一个对象实例化后,即可像函数一样进行调用,并执行预定义的逻辑。这样可以增加对象的灵活性,使其更加接近函数的行为。__call__ 方法只有在对象被调用时才会被调用,也就是对象被函数那样调用。

通常情况下,__call__ 方法常用于实现可调用的对象,如自定义的函数对象、装饰器、上下文管理器等。通过定义 __call__ 方法,我们可以使对象具有函数的特性,并能够直接调用执行相应的逻辑。

更多精彩内容,请关注同名公众:一点sir(alittle-sir)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/317994.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

行为型设计模式——迭代器模式

迭代器模式 迭代器模式也是非常的简单&#xff0c;定义如下&#xff1a; 提供一个对象来顺序访问聚合对象中的一系列数据&#xff0c;而不暴露聚合对象的内部表示。 相信大家都使用过类似下面的迭代器&#xff1a; List<String> list new ArrayList<>(); Iterat…

3d建模软件有哪些?3d云渲染推荐

3D建模软件有很多&#xff0c;有的非常复杂难以上手&#xff0c;那么适合新手的有哪些呢&#xff1f;一起来看看吧。 1、SketchUp SketchUp是一个用户友好且直观的建模软件&#xff0c;能与V-Ray渲染器一起使用&#xff0c;适合初学者。2、Blender Blender是一个功能强大且免费…

图片无损放大神器PhotoZoom8 Pro中文破解版

摘要 BenVista PhotoZoom中文版是一款采用专利技术的图片无损放大软件&#xff0c;被誉为放大神器。该软件以其卓越的效果而著称&#xff0c;特别是在无锯齿、不失真的完美放大图像质量方面表现出色。 软件介绍 BenVista PhotoZoom中文版是一款采用专利技术的图片无损放大软…

视频封面提取工具,批量提取视频封面

你是否曾为从大量的视频中提取封面而感到困扰&#xff1f;传统的做法是逐个打开视频&#xff0c;然后手动截图。这不仅费时费力&#xff0c;还容易出错。现在&#xff0c;有了我们【媒体梦工厂】&#xff0c;这些烦恼全部消失。不论视频数量多少&#xff0c;一键即可批量提取&a…

sentinel熔断与限流

文章目录 一、sentinel简介Sentinel 是什么&#xff1f;Sentinel安装 二、sentinel整合工程新建cloudalibaba-sentinel-service8401微服务引入依赖yml配置主启动类添加EnableDiscoveryClient业务类测试 三、sentinel流控规则基本介绍流控模式直接&#xff08;默认&#xff09;关…

掌握WPF控件:熟练常用属性(一)

WPF布局常用控件&#xff08;一&#xff09; Border Border控件是一个装饰控件&#xff0c;用于围绕其他元素绘制边框和背景。它提供了一种简单的方式来为其他控件添加边框和背景样式&#xff0c;而无需自定义控件的绘制逻辑。 常用属性描述Background用于设置背景颜色或图像…

当浏览器输入url的时候会发生什么?

说在前面 当我们在浏览器中输入URL并按下回车时&#xff0c;背后发生了一系列神秘的操作。本文将带您深入了解&#xff0c;从URL解析到页面渲染&#xff0c;揭秘浏览器输入URL的完整流程。 具体步骤 当浏览器输入URL时&#xff0c;一般经过以下细节步骤&#xff1a; 1、引言 …

通用外设-2.8‘TFT屏的使用

前言 一、验证连接是否正确 二、更改自己想用的图像 1.取模软件 Image2Lcd 2.9 的使用 2.使用 总结 前言 本文在中景园的代码上改写而来&#xff0c;主要记录下使用记录 一、验证连接是否正确 1.按内容说明进行线路连接 2.运行程序&#xff0c;因为内部有图片样本&…

Elasticsearch聚合优化 | 聚合速度提升5倍!

1、聚合为什么慢&#xff1f; 大多数时候对单个字段的聚合查询还是非常快的&#xff0c; 但是当需要同时聚合多个字段时&#xff0c;就可能会产生大量的分组&#xff0c;最终结果就是占用 Elasticsearch大量内存&#xff0c;从而导致 OOM 的情况发生。 实践应用发现&#xff0…

飞鱼CRM接入第三方系统 飞鱼API对接详细教程

场景描述 在白码低代码开发平台中&#xff0c;是支持外部crm系统的线索通过接口流入到白码系统里面&#xff0c;换而言之&#xff0c;只要外部的系统有线索api接口&#xff0c;白码系统可以接收线索并在白码系统上进行后续操作。本文以飞鱼crm系统为例&#xff0c;讲解如何接收…

LeetCode讲解篇之78. 子集

文章目录 题目描述题解思路题解代码 题目描述 题解思路 初始化一个start变量记录当前从哪里开始遍历搜索nums 搜索过程的数字组合加入结果集 然后从start下标开始遍历nums&#xff0c;更新start&#xff0c;递归搜索 直到搜索完毕&#xff0c;返回结果集 题解代码 class …

Open3D 获取点云坐标最值(17)

Open3D 获取点云坐标最值(17) 一、算法介绍二、算法实现1.代码2.结果人生天地间,忽如远行客 一、算法介绍 快速获取点云块,沿着 x y z 各方向的坐标最值,这些在点云处理中的应用范围是如此广泛,这也是点云最常被用到的关键信息,后续的很多算法都会设置到这一处理方法。…

学习selenium+python使用 XPath 表达式来实现找到目标元素时智能封装等待,执行测试代码启动Chrome浏览器后,地址栏只显示data;

背景 学习使用 XPath 表达式来实现找到目标元素时智能封装等待执行测试代码启动Chrome浏览器后&#xff0c;地址栏只显示data&#xff1b; 代码如下 import unittest from selenium import webdriver from selenium.common.exceptions import NoSuchElementException from …

关于Quartz远程调用服务方法失败如何解决,@Inner详细介绍

1.单独在要调用服务的controller写上相关方法&#xff08;Inner(value true)要走aop&#xff0c;会检测是否有内部调用标识&#xff09;具体见下述 2. 编写Feign远程调用的接口&#xff0c;注意加上RequestHeader(SecurityConstants.FROM) String from。因为inner(value true…

OpenCV 基于C++图像读取及存储API函数

OpenCV可以从存储介质中读入图像&#xff0c;也可以将摄像头(Camera)抓取的图像载入内存&#xff0c;然后进行处理。而存储图像就是将内存中的图像数据写入存储介质中&#xff0c;如写入硬盘、优盘等。 OpenCV要读入图像、操作图像。首先要用到Mat类&#xff0c;并且需创建Mat对…

飞桨分子动力学模拟-论文复现第六期:复现TorchMD

飞桨分子动力学模拟-论文复现第六期&#xff1a;复现TorchMD Paddle for MD 飞桨分子动力学模拟科学计算 复现论文-TorchMD: A deep learning framework for molecular simulations 本项目可在AIStudio一键运行&#xff1a;飞桨分子动力学模拟PaddleMD-复现TorchMD 【论文复…

【STM32】HAL库的RCC复位状态判断及NVIC系统软件复位

【STM32】HAL库的RCC复位状态判断及NVIC系统软件复位 在实际开发中 有时候会遇到复位状态不同 导致结果不同的情况 比如在上电复位时 电压不稳定 可能导致一些外部芯片无法正常工作 从而导致进行了错误的操作流程 所以 可以在程序运行后 加一个复位状态判断 用来检测是否正常复…

linux 如何创建文件

我们在写一些教程的时候&#xff0c;经常会需要创建一些用于演示的文档&#xff0c;这些文档往往需要填充一些不特定的内容。那么如何快速的创建演示用的文档呢&#xff1f; docfaker.py docfaker.py是一个py脚本&#xff0c;用于创建一个简单的txt文档&#xff0c;docfaker.…

力扣每日一练(24-1-13)

如果用列表生成式&#xff0c;可以满足输出的型式&#xff0c;但是不满足题意&#xff1a; nums[:] [i for i in nums if i ! val]return len(nums) 题意要求是&#xff1a; 你需要原地修改数组&#xff0c;并且只使用O(1)的额外空间。这意味着我们不能创建新的列表&#xff…

vivado 指定顶部模块和重新排序源

指定顶部模块和重新排序源 文件夹默认情况下&#xff0c;Vivado Design Suite会自动确定设计的顶层添加到的源文件的层次结构和细化、合成和模拟的顺序项目这可以通过右键单击中的“层次更新”设置进行控制“源”窗口的菜单。请参阅中的“源”窗口中的“层次更新”命令Vivado …
最新文章