百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

读懂python中的self

itomcoil 2025-01-18 20:43 27 浏览

一、self的作用

self指的是调用该函数的对象(是一个实例)。Python中self等价于Java中的this。

首先明确的是self只有在类中的方法中才会有,独立的函数或方法是不必带有self的。

例如:

# 定义方法
def showTime(name):
    print(f'大家好我是{name},多多关照!')

# 调用方法
showTime('齐天大圣')

"""
输出结果:
大家好我是齐天大圣,多多关照!
"""

二、self的使用注意事项

1、self代表类的实例,而非类。

# self代表类的实例,而非类
class TestDemo():
    # 可将self理解为实例td
    def testFn(self):  
        print(f"谁调用我我就是谁,此时调用我的是{self}")
        # 实例调用__class__属性时会指向该实例对应的类
        print(f"我是按照{self.__class__}创建出来的")
# td为TestDemo的实例
td = TestDemo()
# 在类中方法的形参中,self参数一定要定义,但是在调用时会自动传入。
td.testFn()

执行结果如下:

谁调用我我就是谁,此时调用我的是<__main__.TestDemo object at 0x00000000028836C8>
我是按照<class '__main__.TestDemo'>创建出来的

说明:

<__main__.TestDemo object at 0x00000000028836C8>表示:
self是一个TestDemo类型的object(对象),对象在内存的地址为0x00000000028836C8。

为什么self指的是类的实例对象,而不是类本身。

如果self指向类本身,那么当一个类有多个实例对象时,self指向哪一个呢?

2、self不必非写成self,只是一种规范。

有很多人先学习别的语言,如Java,然后再学习Python的,所以总觉得self怪怪的,想写成this,可以吗?
当然可以,换成任何标识符都可以,把上面的代码改写一下。

# self代表类的实例,而非类
class TestDemo():
    # 可将self理解为实例td
    def testFn(this):  
        print(f"谁调用我我就是谁,此时调用我的是{this}")
        # 实例调用__class__属性时会指向该实例对应的类
        print(f"我是按照{this.__class__}创建出来的")
# td为TestDemo的实例
td = TestDemo()
# 在类中方法的形参中,self参数一定要定义,但是在调用时会自动传入。
td.testFn()

执行结果如下:

谁调用我我就是谁,此时调用我的是<__main__.TestDemo object at 0x00000000028836C8>
我是按照<class '__main__.TestDemo'>创建出来的

改成this后,运行结果完全一样。
当然,最好还是尊重约定俗成的习惯,使用self。(不是最好,是一定。)

3、类中方法的形参中一定要写self,包括内置函数

# 如果类中的方法不写self形参,
# 则不能使用对象.方法名()来调用方法,
# 只能使用类名.方法名()的方式来调用该方法,
# 类似与Java中的静态方法。

class TestDemo():
    # 定义一个方法,不定义self形参
    def testFn():
        print(f"不定义形参self,依旧可以调用我")
        print(__class__)

# 创建对象,用对象.方法名()来调用方法
td = TestDemo()
# 报错
# TypeError: testFn() takes 0 positional arguments but 1 was given
td.testFn()

# 只能使用类名.方法名()的方式来调用该方法。
TestDemo.testFn()

4、__init__函数中,要把接收到的参数赋值到self中,提供全类使用

关于__init__函数,可以查看Python的魔法方法来了解。

class Student():
    def __init__(self, name, age, addr):
        # self的作用主要表示这个变量是类中的公共变量
        # 定义在self中的属性,整个类内都可以使用
        # 普通方法同理
        self.name = name
        self.age = age
        # 没有定义在self中的属性,只能在当前方法内使用
        addr = addr
    # 标准用法
    def tellMeName(self):
        # 如果去掉此处的self,会提示name 'name' is not defined
        print(f'我不叫孙悟空,我叫{self.name}')
    # 方法形参没有定义self,则报错
    # TypeError: tellMeAge() takes 0 positional arguments but 1 was given
    # def tellMeAge():
    def tellMeAge(self):
        # 如果获取age的值不加self,则获取不到,会报错
        # NameError: name 'age' is not defined
        # print(f'我今年{age}啦')
        print(f'我今年{self.age}啦')
    def tellMeAddr(self):
        # 因为__init__函数汇总没有把addr变量定义在self对象中
        # 所以addr变量的作用域只在__init__函数内,
        # 其他函数无法调用。
        # 添加在self对象内的属性为全局属性。
        print(f'我现居住在{self.addr}')
s = Student('美猴王', 18, addr='北京')
s.tellMeName()
s.tellMeAge()
s.tellMeAddr()

5、同一个类中调用其他的方法时需要加self

class Student():
    def __init__(self, name, age, addr):
        # self的作用主要表示这个变量是类中的公共变量
        # 定义在self中的属性,整个类内都可以使用
        # 普通方法同理
        self.name = name
        self.age = age
        self.addr = addr
    def tellMeName(self):
        print(f'我不叫孙悟空,我叫{self.name}')
    def tellMeAge(self):
        print(f'我今年{self.age}啦')
    def tellMeAddr(self):
        print(f'我现居住在{self.addr}')
    def tellAll(self):
        # 如果调用类中的其他函数时,不用self调用,则会报错
        # NameError: name 'tellMeName' is not defined
        # tellMeName()
        self.tellMeName()
        self.tellMeAge()
        self.tellMeAddr()

s = Student('美猴王', 18, addr='北京')
s.tellAll()
"""
输出结果:
我不叫孙悟空,我叫美猴王
我今年18啦
我现居住在北京
"""

6、self总是指调用时的类的实例,在继承时中也一样。

看完面向对象继承的内容可以回来再看。

# 定义一个父类
class Parent():
    def pFn(self):
        print(self)
# 定义一个子类
class Child(Parent):
    def cFn(self):
        print(self)
# 创建子类对象
child = Child()
# 调用子类方法
# <__main__.Child object at 0x00000000025A38C8>
child.cFn()
# 子类调用父类方法
# <__main__.Child object at 0x00000000025A38C8>
child.pFn()
# 创建父类
parent = Parent()
# 调用自己方法
# <__main__.Parent object at 0x00000000025A3908>
parent.pFn()

7、self与私有变量的用法

看完私有变量的内容可以回来再看。

# self的属性名称前加上两个下划线,就变成了一个私有变量(private)
# 只有类内部可以访问,外部不能直接访问
class Student():
    def setname(self, name1, name2):
        self.name1 = name1
        self.__name2 = name2

    def getname(self):
        print(f'我的第一个名字是{self.name1},我的第二个名字是{self.__name2}')
stu = Student()
stu .setname("齐天大圣", "美猴王")
# 结果:我的第一个名字是齐天大圣,我的第二个名字是美猴王
stu .getname()

# 结果:齐天大圣
print(stu.name1)
# 结果报错:AttributeError: 'Student' object has no attribute 'name2'
# 说明私有变量并不能获取到
print(stu.name2)

8、总结:

self总是指调用时的类的实例。

self的名字并不是规定死的,但是在Python中self不是关键词,你可以定义成this、abc或其它名字都可以。但是约定成俗,减少代码理解难度。

在类中方法的形参中,self参数一定要定义,但是在调用时会自动传入。

9、综合练习

一个类可以创建多个对象。

# 需求:洗衣机,功能:能洗衣服

# 1. 定义洗衣机类
class Washer():
    def wash(self):
        print('我会洗衣服')
        print(self)

# 2. 创建对象
haier1 = Washer()
# <__main__.Washer object at 0x0000000002553508>
print(haier1)
# haier1对象调用实例方法(对象方法)
haier1.wash()

# 3.创建第二个对象
haier2 = Washer()
# <__main__.Washer object at 0x0000000002553608>
print(haier2)
haier2.wash()
"""
输出结果:
<__main__.Washer object at 0x00000000025A3688>
我会洗衣服
<__main__.Washer object at 0x00000000025A3688>
<__main__.Washer object at 0x00000000025A3788>
我会洗衣服
<__main__.Washer object at 0x00000000025A3788>
"""
"""
可以看到每创建一个新的对象,都是有独立空间的对象,
所以每个对象的地址是不同的。
"""

注意:

可以看到每创建一个新的对象,都是有独立空间的对象,因为每个对象的地址是不同的。

每次打印对象和self得到的结果是一致的,说明self指向了对象的实例

相关推荐

python创建文件夹,轻松搞定,喝咖啡去了

最近经常在录视频课程,一个课程下面往往有许多小课,需要分多个文件夹来放视频、PPT和案例,这下可好了,一个一个手工创建,手酸了都做不完。别急,来段PYTHON代码,轻松搞定,喝咖啡去了!import...

如何编写第一个Python程序_pycharm写第一个python程序

一、第一个python程序[掌握]python:python解释器,将python代码解释成计算机认识的语言pycharm:IDE(集成开发环境),写代码的一个软件,集成了写代码,...

Python文件怎么打包为exe程序?_python3.8打包成exe文件

PyInstaller是一个Python应用程序打包工具,它可以将Python程序打包为单个独立可执行文件。要使用PyInstaller打包Python程序,需要在命令行中使用py...

官方的Python环境_python环境版本

Python是一种解释型编程开发语言,根据Python语法编写出来的程序,需要经过Python解释器来进行执行。打开Python官网(https://www.python.org),找到下载页面,选择...

[编程基础] Python配置文件读取库ConfigParser总结

PythonConfigParser教程显示了如何使用ConfigParser在Python中使用配置文件。文章目录1介绍1.1PythonConfigParser读取文件1.2Python...

Python打包exe软件,用这个库真的很容易

初学Python的人会觉得开发一个exe软件非常复杂,其实不然,从.py到.exe文件的过程很简单。你甚至可以在一天之内用Python开发一个能正常运行的exe软件,因为Python有专门exe打包库...

2025 PyInstaller 打包说明(中文指南),python 打包成exe 都在这里

点赞标记,明天就能用上这几个技巧!linux运维、shell、python、网络爬虫、数据采集等定定做,请私信。。。PyInstaller打包说明(中文指南)下面按准备→基本使用→常用...

Python自动化办公应用学习笔记40—文件路径2

4.特殊路径操作用户主目录·获取当前用户的主目录路径非常常用:frompathlibimportPathhome_dir=Path.home()#返回当前用户主目录的Path对象...

Python内置tempfile模块: 生成临时文件和目录详解

1.引言在Python开发中,临时文件和目录的创建和管理是一个常见的需求。Python提供了内置模块tempfile,用于生成临时文件和目录。本文将详细介绍tempfile模块的使用方法、原理及相关...

python代码实现读取文件并生成韦恩图

00、背景今天战略解码,有同学用韦恩图展示各个产品线的占比,效果不错。韦恩图(Venndiagram),是在集合论数学分支中,在不太严格的意义下用以表示集合的一种图解。它们用于展示在不同的事物群组之...

Python技术解放双手,一键搞定海量文件重命名,一周工作量秒搞定

摘要:想象一下,周五傍晚,办公室的同事们纷纷准备享受周末,而你,面对着堆积如山的文件,需要将它们的文件名从美国日期格式改为欧洲日期格式,这似乎注定了你将与加班为伍。但别担心,Python自动化办公来...

Python路径操作的一些基础方法_python路径文件

带你走进@机器人时代Discover点击上面蓝色文字,关注我们Python自动化操作文件避开不了路径操作方法,今天我们来学习一下路径操作的一些基础。Pathlib库模块提供的路径操作包括路径的...

Python爬取下载m3u8加密视频,原来这么简单

1.前言爬取视频的时候发现,现在的视频都是经过加密(m3u8),不再是mp4或者avi链接直接在网页显示,都是经过加密形成ts文件分段进行播放。今天就教大家如果通过python爬取下载m3u8加密视频...

探秘 shutil:Python 高级文件操作的得力助手

在Python的标准库中,shutil模块犹如一位技艺精湛的工匠,为我们处理文件和目录提供了一系列高级操作功能。无论是文件的复制、移动、删除,还是归档与解压缩,shutil都能以简洁高效的方式完成...

怎么把 Python + Flet 开发的程序,打包为 exe ?这个方法很简单!

前面用Python+Flet开发的“我的计算器v3”,怎么打包为exe文件呢?这样才能分发给他人,直接“双击”运行使用啊!今天我给大家分享一个简单的、可用的,把Flet开发的程序打包为...