一文搞懂Python中的类和__init__函数

省流版

Python中有一种特殊的函数__init__,多数初学者不会去关注这个问题,但是__init__函数是很重要的,尤其是希望在Python中实现类和实例的操作。本文将带你搞清楚__init__函数在类和实例的应用中到底齐了什么样的作用!

文中部分内容来自GPT,他已经是我日常学习和工作中必不可少的工具。我了解到目前部分朋友还不清楚怎么开通和使用GPT,更多相关教程可以参见这篇博客:GPT4.0使用教程

或者直接通过右侧平台:GPT4.0升级平台


作为典型的面向对象的语言,Python中 定义使用是不可或缺的一部分知识。对于有面向对象的经验、对实例的概念已经足够清晰的人,学习Python的这套定义规则不过是语法的迁移。但对新手小白而言,要想相对快速地跨过__init__这道坎,还是结合一个简单例子来说比较好。

以创建一个“学生”为例,最简单的语句是

class Student():
    pass

当然,这样定义的类没有包含任何预定义的数据和功能。除了名字叫Student以外,它没有体现出任何“学生”应该具有的特点。但它是可用的,上述代码运行过后,通过类似

stu_1 = Student()

这样的语句,我们可以创建一个“学生”实例,即一个具体的“学生”对象。

通过class语句定义的类Student,就好像一个“模具”,它可以定义作为一个学生应该具有的各种特点(这里暂未具体定义);

而在类名Student后加圆括号(),组成一个类似函数调用的形式Student(),则执行了一个叫做实例化的过程,即根据定义好的规则,创建一个包含具体数据的学生对象(实例)。

为了使用创建的学生实例stu_1,我们可以继续为它添加或修改属性,比如添加一组成绩scores ,由三个整数组成:

stu_1.scores = [80, 90, 85]

但这样明显存在很多问题,一旦我们需要处理很多学生实例,比如stu_2, stu_3, ...,这样不但带来书写上的麻烦,还容易带来错误,万一某些地方scores打错了,或者干脆忘记了,相应的学生实例就会缺少正确的scores属性。更重要的是,这样的scores属性是暴露出来的,它的使用完全被外面控制着,没有起到“封装”的效果,既不方便也不靠谱

一个自然的解决方案是允许我们在执行实例化过程Student()传入一些参数,以方便且正确地初始化/设置一些属性值,那么如何定义这种初始化行为呢?答案就是在类内部定义一个__init__函数。这时,Student的定义将变成(我们先用一段注释占着__init__函数内的位置)。

class Student():
    def __init__(self, score1, score2, score3):
        # 相关初始化语句

定义__init__后,执行实例化的过程须变成Student(arg1, arg2, arg3)新建的实例本身,连带其中的参数,会一并传给__init__函数自动并执行它。所以__init__函数的参数列表会在开头多出一项,它永远指代新建的那个实例对象,Python语法要求这个参数必须要有,而名称随意,习惯上就命为self

新建的实例传给self后,就可以在__init__函数内创建并初始化它的属性了,比如之前的scores,就可以写为

class Student():
    def __init__(self, score1, score2, score3):
        self.scores = [score1, score2, score3]

此时,若再要创建拥有具体成绩的学生实例,就只需

stu_1 = Student(80, 90, 85)

此时,stu_1将已经具有设置好的scores属性。并且由于__init__规定了实例化时的参数,若传入的参数数目不正确,解释器可以报错提醒。你也可以在其内部添加必要的参数检查,以避免错误或不合理的参数传递

在其他方面, __init__就与普通函数无异了。考虑到新手可能对“函数”也掌握得很模糊,这里特别指出几个“无异”之处:
独立的命名空间,也就是说 函数内新引入的变量均为局部变量,新建的实例对象对这个函数来说也只是通过第一参数self从外部传入的,故无论设置还是使用它的属性都得利用 self.<属性名>。如果将上面的初始化语句写成
scores = [score1, score2, score3](少了 self.),
则只是在函数内部创建了一个scores变量,它在函数执行完就会消失,对新建的实例没有任何影响;
与此对应, self的属性名和函数内其他名称(包括参数)也是不冲突的,所以你可能经常见到类似这种写法,它正确而且规范。
class Student():
    def __init__(self, name, scores):
        # 这里增加了属性name,并将所有成绩作为一个参数scores传入
        # self.name是self的属性,单独的name是函数内的局部变量,参数也是局部变量
        self.name = name
        if len(scores) == 3:
            self.scores = scores
        else:
            self.scores = [0] * 3
从第二参数开始均可设置 变长参数默认值等,相应地将允许实例化过程 Student()中灵活地传入需要数量的参数;
其他……

说到最后,__init__还是有个特殊之处,那就是它不允许有返回值。如果你的__init__过于复杂有可能要提前结束的话,使用单独的return就好,不要带返回值。

上面代码的执行结果如下

  • 31
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
Python,无法直接接收的`__del__`方法的返回值。`__del__`方法是在对象被销毁时自动调用的,它主要用于释放资源和执行清理操作,而不是返回值。当对象被销毁时,Python解释器会自动调用`__del__`方法,但它不会返回任何值。如果你需要获取某个方法的返回值,你可以在该方法使用`return`语句将值返回给调用者。但是对于`__del__`方法来说,它的返回值会被忽略,因为它是由解释器自动调用的,而不是由代码显式调用的。所以,无法直接接收`__del__`方法的返回值。 #### 引用[.reference_title] - *1* *3* [python 深刻理解python __init__,__new__,__del__基本概念和用法](https://blog.csdn.net/weixin_47906106/article/details/118381168)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insert_down1,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* [一文搞懂python中类的__new__方法和__init__方法](https://blog.csdn.net/weixin_43912125/article/details/121781334)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insert_down1,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

小井正在努力中

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值