零基础学习Python(三)

1. 多重继承

一个子类可以继承多个父类,这与一些编程语言的规则不通。

如果多个父类中有同名的变量和方法,子类访问的顺序是按照继承时小括号里书写的顺序进行访问的。

可以用issubclass(B, A)方法判断B是否为A的子类。

2. 绑定

类中的方法通过参数self与对象绑定,通过参数cls与类绑定,如果不用self或者cls来访问类中的变量或者方法,会报错找不到变量或者方法,或者达不到预期效果。

class C:x = 100def set_x(self, v):x = vc = C()
c.set_x(520)

上述代码既不会改变对象c的属性x,也不会改变类C的属性x,set_x方法只是在函数内部建立了一个临时变量。

3. 重写与钻石继承

先定义父类:

class C:def __init__(self, x, y):self.x = xself.y =ydef add():return self.x + self.ydef mul():return self.x * self.y

定义子类,并且重写父类的方法,同时调用父类的方法:

class D(C):def __init__(self, x, y, z):C.__init__(self, x, y)self.z = zdef add():return C.add(self) + self.zdef mul():return C.mul(self) * self.z

如果是钻石继承,这种直接通过类名调用类里面方法的方式可能会出现问题。什么是钻石继承?就是有两个类同时继承了同一个父类,然后另外一个类继承了这两个类,继承关系拓扑图类似于一个钻石(菱形):

钻石继承的问题如下:

可以发现,类A初始化了两次!使用super()函数可以解决上述问题,类B1、类B2、类C的代码修改如下:

class B1(A):def __init__(self):super().__init__()print("哈喽,我是B1~")class B2(A):def __init__(self):super().__init__()print("哈喽,我是B2~")class C(B1, B2):def __init__(self):super().__init__()print("哈喽,我是C~")

使用super()方法调用__init__方法,不用传递self参数,因为super方法会自动解决。super函数的原理依赖于Python的MRO(Method Resolution Order:方法解析顺序),这可以通过类的方法mro()或者变量__mro__查看:

关于MRO顺序的一个案例:

class Displayer:def display(self, msg):print(msg)class LoggerMixin:def log(self, msg, filename):with open(filename, 'a') as f:f.write(msg)def display(self, msg):super().display(msg)self.log(msg)class MySubClass(LoggerMixin, Displayer):def log(self, msg):super().log(msg, filename="subclasslog.txt")subcls = MySubClass()
subcls.display("This is a test.")

运行结果是打印This is a test,并且生成subclasslog.txt,里面内容是This is a test。查看MySubClass的MRO顺序:

 

那么调用subcls的display方法,先去LoggerMixin里面找display方法,找到了,但是第一句是super().display(msg),此时不要以为会去LoggerMixin类的父类object里面找display方法,而是按照MRO顺序去Displayer里面找display方法,从而打印出This is a test,然后执行LoggerMixin力的display方法的第二句:self.log(msg),即调用MySubClass里面的log方法,发现没有,按照MRO顺序去LoggerMixin类里找log方法,找到了,即写入文件内容为msg。

4. __slots__属性

对象的属性除了直接obj.x = valueX这种方式设置外,还可以通过其字典__dict__的方式设置,比如:

c.__dict__['z'] = 666

 

Python中对象是可以随意添加任何属性的,这些属性存放在字典中,虽然字典访问效率高,但是浪费了很多空间,为了避免空间浪费,Python引入__slots__属性,通过__slots__属性设置一个类的对象只能拥有固定属性:

class C:__slots__ = ["x", "y"]def __init__(self, x)self.x = x

 

除了动态添加属性不行,在类的方法(包括构造方法)内添加属性也不行:

父类中的slots属性是不会在子类中生效的(但是子类仍然继承了父类的slots属性,只是不起作用了):

5. 魔法方法

__new__方法是在__init__方法之前调用的,常见的self对象就是它返回的。创建一个类,使得传给他的字符串始终为大写:

class CapStr(str):def __new__(cls, string):string = string.upper()return super().__new__(cls, string)

 

因为CapStr继承自str,所以str有的方法它也有:

对象被销毁时会调用__del__方法:

class C:def __init__(self):print("我来了~")def __del__(self):print("我走了~")

 

注意,不是使用了del语句就会调用__del__这个魔法方法,而是对象被销毁前才会调用,也就是说调用del语句并不会立即销毁对象,只是销毁对象的引用,只有当对象的引用都被销毁时,才会去销毁对象。比如将c再赋值给d,调用del c不会触发__del__魔法方法,接着调用del d才会触发。 

既然__del__方法可以在对象被销毁前动手动脚,那么可以通过将即将要销毁的对象赋值给全局变量的方式,实现对象的重生。但是使用全局变量可能会污染命名空间,那么可以通过闭包的形式将对象传递给函数的参数,从而实现永久保存:

class E:def __init__(self, name, func):self.name = nameself.func = funcdef __del__(self):self.func(self)def outer():x = 0def inner(y=None):nonlocal xif y:x = yelsereturn xreturn inner

 

重写add方法可以实现自定义的加法功能,比如字符串的相加不再是拼接,而是字符长度的相加:

 

注意,加号调用的是左边对象的__add__方法,s1 + s2相当于s1.__add__(s2)。

__radd__方法的调用原则:如果加号两边的数据类型不同,并且左侧对象没有定义__add__方法,或者__add__方法实现为NotImplemented,那么Python就会去右侧对象找__radd__方法。

__iadd__方法不仅进行加法运算,还会将运算后的结果赋值给左侧对象,调用时机是使用了运算符+=:

__index__魔法方法是当对象作为索引值才会去调用,而不是对象的索引访问触发:

class C:def __index__(self):print("被拦截了~")return 3

关于属性的方法:hasattr、getattr:

class C:def __init(self, name, age):self.name = nameself.__age = agec = C("小甲鱼", 18)
hasattr(c, "name") # 返回True
getattr(c, "name") # 返回小甲鱼

对于私有属性,依然可以访问:

getattr(c, "_c__age") # 返回18

设置属性:

setattr(c, "_c__age", 19) # 返回18

删除属性:

delattr(c, "_c__age")

魔法方法__getattribute__会拦截getattr方法: 

class C:def __init(self, name, age):self.name = nameself.__age = agedef __getattribute__(self, attrname):print("拿来吧你~")return super().getattribute__(attrname)c = C("小甲鱼", 18)
getattr(c, "name")

 

魔法方法__getattr__只有尝试去获取不存在的属性时才会去触发:

class C:def __init(self, name, age):self.name = nameself.__age = agedef __getattribute__(self, attrname):print("拿来吧你~")return super().getattribute__(attrname)def __getattr__(self, attrname):if attrname == 'FishC':print("I love FishC")elseraise AttributeError(attrname)c = C("小甲鱼", 18)

 

给属性赋值对应的魔法方法是__setattr__,但是这个方法里面的实现不能是self.attrname=value,因为这个语句又会调用魔法方法__setattr__,所以会死循环,正确做法是:

class D:def __setattr(self, attrname, value):self.__dict__[attrname] = value

同理,del语句删除属性时,也不能在魔法方法__del__中直接使用del self.attrname,否则也会无限循环,正确做法依然是操作self.__dict__这个字典:

class D:def __setattr(self, attrname, value):self.__dict__[attrname] = valuedef __delattr(self, attrname):self.__dict__[attrname]

​​​​​​​ 

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

《TF2.x强化学习手册》P59-P65-SARSA-Q-learning

文章目录 实现SARSA算法和对应的强化学习智能体前期准备实现步骤工作原理初始化算法流程 构建基于Q学习的智能体前期准备实现步骤工作原理SARSA 算法的收敛性:SARSA 适合在线学习和真实系统:Q 学习算法的适用性: 实现SARSA算法和对应的强化学…

HDC使用常见命令

HDC(HarmonyOS Device Connector)是为开发人员提供的用于调试的命令行工具,通过该工具可以在windows/linux/mac系统上与真实设备进行交互。 使用HDC前,需要配置相关环境变量: 在此电脑 > 属性 > 高级系统设置 &g…

Git常用命令以及使用IDEA集成Gitee

目录 一、设置用户签名 二、初始化本地库 三、查看本地库状态 四、添加文件到暂存区 五、提交本地库 六、修改文件 七、版本穿梭 八、Git分支 九、分支的操作 9.1、查看分支 9.2、创建分支 9.3、切换分支 9.4、合并分支 十、团队协作 十一、Idea集成Git 11.1、配…

Github 2024-07-15 开源项目周报 Top15

根据Github Trendings的统计,本周(2024-07-15统计)共有15个项目上榜。根据开发语言中项目的数量,汇总情况如下: 开发语言项目数量Python项目5非开发语言项目4JavaScript项目3TypeScript项目2Go项目1Solidity项目1Java项目1Rust项目1免费编程学习平台:freeCodeCamp.org 创建…

3.1-RNN存在的问题以及LSTM的结构

文章目录 1 RNN存在的问题1.1梯度消失问题1.2梯度爆炸问题1.3梯度爆炸的对策 2梯度消失的对策——LSTM2.1输出门2.2遗忘门2.3输入门2.4总结2.5 LSTM梯度的流动 1 RNN存在的问题 RNN存在梯度消失和梯度爆炸的问题。 书上以下图的这句话为例,进行说明;为了…

前瞻断言与后瞻断言:JavaScript 正则表达式的秘密武器

JavaScript 中的前瞻断言(lookahead)和后瞻断言(lookbehind)相信用过的小伙伴就知道它的威力了,在一些特定的需求场景下,可以做到四两拨千斤的作用,今天让我们来盘点一下在 JavaScript 正则表达…

昇思25天学习打卡营第14天|munger85

基于MindNLPMusicGen生成自己的个性化音乐 这个所谓的个性化的音乐就是指你输入一段文字它会根据这个文字输出一段音乐这个音乐是贴近于那段文字的所以叫做文生成音乐, 如果网络正常的话就可以直接从下载这个模型。 那么音乐生成的有两种方式呢有两种方式&#xff…

【C++初阶】C/C++内存管理

【C初阶】C/C内存管理 🥕个人主页:开敲🍉 🔥所属专栏:C🥭 🌼文章目录🌼 1. C/C内存分布 2. C语言中动态内存管理方式:malloc/calloc/realloc/free 3. C内存管理方式 3…

拉格朗日乘子法和KKT条件

拉格朗日乘子法(Lagrange Multiplier) 和 KKT(Karush-Kuhn-Tucker) 条件是求解约束优化问题的重要方法,在有等式约束时使用拉格朗日乘子法,在有不等约束时使用 KKT 条件。当然,这两个方法求得的结果只是必要条件,只有当目标函数…

ssrf复习(及ctfshow351-360)

1. SSRF 概述 服务器会根据用户提交的URL发送一个HTTP请求。使用用户指定的URL,Web应用可以获取图片或者文件资源等。典型的例子是百度识图功能。 如果没有对用户提交URL和远端服务器所返回的信息做合适的验证或过滤,就有可能存在“请求伪造"的缺陷…

FPGA学习笔记(一) FPGA最小系统

文章目录 前言一、FPGA最小系统总结 前言 今天学习下FPGA的最小系统一、FPGA最小系统 FPGA最小系统与STM32最小系统类似,由供电电源,时钟电路晶振,复位和调试接口JTAG以及FLASH配置芯片组成,其与STM32最大的不同之处就是必须要有…

链表面试练习习题集(Java)

1. 思路&#xff1a; 因为杨辉三角是由二维数组构成&#xff0c;所以要先创建一个二维数组&#xff0c;如何用顺序表表示二维数组&#xff0c;可以通过List<List<Interger>>来表示一个二维数组&#xff0c;可以这样理解&#xff1a;先创建一个一维数组List&#x…

modbus slave 设备通过 网关thingsboard-gateway 将数据上传到thingsboard云平台

搭建thingsboard物联网云平台花了大量时间&#xff0c;从小白到最后搭建成功&#xff0c;折磨了好几天&#xff0c;也感谢网友的帮助&#xff0c;提供了思路最终成功搞定&#xff0c;特此记录。 一、thingsboard环境搭建&#xff08;Ubuntu20.04LTS&#xff09; 参考官方文档&a…

java之 junit单元测试案例【经典版】

一 junit单元测试 1.1 单元测试作用 单元测试要满足AIR原则&#xff0c;即 A&#xff1a; automatic 自动化&#xff1b; I: Independent 独立性&#xff1b; R&#xff1a;Repeatable 可重复&#xff1b; 2.单元测试必须使用assert来验证 1.2 案例1 常规单元测试 1.…

PSINS工具箱函数介绍——r2d

介绍工具箱里面r2d这个小函数的作用。 程序源码 function deg r2d(rad) % Convert angle unit from radian to degree % % Prototype: deg r2d(rad) % Input: rad - angle in radian(s) % Output: deg - angle in degree(s) % % See also r2dm, r2dms, d2r, dm2r, dms2r% …

多种方式实现 元素高度丝滑的从0-1显示出来

选择合适的方式&#xff0c;给用户更好的体验&#xff0c;多种方式实现 元素高度丝滑的从0-1显示出来。 能用 CSS 实现的动画&#xff0c;就不要采用 JS 去实现。 1、浏览器可以对CSS动画进行优化&#xff0c;其优化原理类似于requestAnimationFrame&#xff0c;会把每一帧的…

appium2.0 执行脚本遇到的问题

遇到的问题&#xff1a; appium 上的日志信息&#xff1a; 配置信息 方法一 之前用1.0的时候 地址默认加的 /wd/hub 在appium2.0上&#xff0c; 服务器默认路径是 / 如果要用/wd/hub 需要通过启动服务时设置基本路径 appium --base-path/wd/hub 这样就能正常执行了 方法二…

mysql的索引事务和存储引擎

一、索引 1、索引 索引的概念 &#xff1a;索引是一个排序的列表&#xff0c;在列表当中存储索引的值以及索引值对应数据所在的物理行。 索引的引用&#xff1a; 使用索引之后&#xff0c;就不需要扫描全表来定位某行的数据。 加快数据库的查询速度。 索引可以是表中的一…

在 K8s 上使用 KubeBlocks 提供的 MySQL operator 部署高可用 WordPress 站点

引言 WordPress WordPress 是全球最流行的内容管理系统&#xff08;CMS&#xff09;&#xff0c;自 2003 年发布以来&#xff0c;已成为网站建设的首选工具。其广泛的插件和主题生态系统使用户能够轻松扩展功能和美化外观。活跃的社区提供丰富的资源和支持&#xff0c;进一步…

[RK3588-Android12] 关于如何取消usb-typec的pd充电功能

问题描述 RK3588取消usb-typec的pd充电功能 解决方案&#xff1a; 在dts中fusb302节点下usb_con: connector子节点下添加如下熟悉&#xff1a; 打上如下2个补丁 diff --git a/drivers/usb/typec/tcpm/tcpm.c b/drivers/usb/typec/tcpm/tcpm.c index c8a4e57c9f9b..173f8cb7…