python全栈学习记录(二十二)多态性、封装、绑定方法与非绑定方法

server/2024/10/20 10:15:01/

多态性、封装、绑定方法与非绑定方法

文章目录

  • 多态性、封装、绑定方法与非绑定方法
  • 一、多态性
  • 二、封装
  • 三、绑定方法与非绑定方法

一、多态性

多态指的是同一种事物的多种形态,如水:冰、水蒸气、液态水,又如动物:猫、狗、猪。
多态性是指不考虑对象类型的情况下直接使用对象的方法,这就要求设计者把对象的方法统一为一种,例如:

python">#猫狗猪都是可以叫,那就可以使用一个接口来调用叫这个功能
class Aminal():passclass Cat(Aminal):def talk(self):print('喵喵喵')class Dog(Aminal):def talk(self):print('汪汪汪')class Pig(Aminal):def talk(self):print('哼哼哼')#接口函数
def talk(aminal):aminal.talk()c=Cat()
d=Dog()
p=Pig()
talk(c)
talk(d)
talk(p)
<<<喵喵喵
<<<汪汪汪
<<<哼哼哼

python本身也是支持多态性的,len函数就是对多种容器类型封装的接口函数,所以使用len函数时只需要将容器的实例传入就可以得到容器的长度值。

由上面的例子我们可以知道多态性的本质就是在不同的类型中定义相同的方法,这样就可以不考虑具体的类型直接调用统一的方法了。
python中还可以使用抽象类来实现多态性。

python">import abc# Animal是抽象类,用来指定子类的标准,不能被实例化
class Animal(metaclass=abc.ABCMeta):@abc.abstractmethod #这个装饰器用来约束子类必须定义speak方法def speak(self):pass       class People(Animal):#继承抽象类的子类不定义speak方法时会报错def speak(self):print('say hello')class Dog(Animal):def speak(self):print('汪汪汪')p=People()
d=Dog()
p.speak()
d.speak()
<<<say hello
<<<汪汪汪

python中还存在一种名为鸭子类型的规范,也就是叫的声音像鸭子,并且你走路的样子也像鸭子,那你就是鸭子。

python">class Disk:def read(self):print('disk read')def write(self):print('disk wirte')class Process:def read(self):print('process read')def write(self):print('process wirte')class File:def read(self):print('file read')def write(self):print('file wirte')

上述代码定义了三个类型鸭子、管道、文件,虽然这三者之间没有任何的联系,但是他们都有read和write方法,所以可以将他们都视为文件类型来使用。通过这种约定式的方式实现多态可以减少程序的耦合度。

二、封装

封指属性对外是隐藏的,但对内是开放的。装指申请一个名称空间,往里装入一系列名字/属性。所以封装指使用类存放一系列属性方法,并且这些属性方法只对类内部开放。

封装属性的目的: 隐藏之后是为了不让外部使用直接使用,需要类内部开辟一个接口然后让类外部的使用通过接口来间接地操作隐藏的属性。精髓在于:我们可以在接口之上附加任意逻辑,从而严格控制使用者对属性的操作。

封装方法的目的:隐藏方法是为了不让外不直接使用,需要类内部开辟一个接口然后在接口内去调用隐藏的功能。精髓在于:隔离了复杂度。

python中隐藏属性和方法的方式是在变量前面加上__。但是

  • 这种隐藏仅仅只是一种语法上的变形操作(变量在类定义阶段被修改为_类名__属性/方法)
  • 这种语法上的变形只在类定义阶段发生一次,因为类体代码仅仅只在类定义阶段检测一次
  • 这种隐藏是对外不对内的,即在类的内部可以直接访问,而在类的外则无法直接访问,原因是在类定义阶段,类体内代码统一发生了一次变形
  • 如果不想让子类的方法覆盖父类的,可以将该方法名前加一个__开头
python">class A():__a=1def __b(self):#类内部可以直接访问隐藏属性print(self.__a)a=A()
print(A.__dict__)
#在类外部可以通过_类名__属性/方法的方式访问
a._A__b()
#类外部加入的__属性不会再次改名
a.__c=2
print(a.__dict__)
<<< {...,'_A__a': 1, '_A__b': <function A.__b at 0x0000018C8E8E9160>,...}
<<<1
<<< {'__c': 2}

将属性隐藏起来的目的是通过设置相应的接口来限制用户对隐藏属性的修改。

python">#控制名字的输入
class TellName():def __init__(self,name):self.__tell_name(name)self.__name=namedef __tell_name(self,name):if not isinstance(name,str): #判断name是否为str类型的对象raise TypeError('姓名必须为字符串') #抛出TypeError('姓名必须为字符串')的报错信息def check_name(self):print(self.__name)t=TellName(111)
<<< TypeError: 姓名必须为字符串t=TellName('李四')
t.check_name()
<<<李四

将方法隐藏起来的目的是为了减少复杂度。

python">class Atm():def __card(self):passdef __auth(self):passdef __input(self):passdef __take_money(self):passdef withdraw(self):self.__card()self.__auth()self.__input()self.__take_money()a=Atm()
a.withdraw()

上述代码只需要在类外调用withdraw方法就可以完成整个取款流程了,大大降低了操作的复杂度。

python中property装饰器用于将被装饰的方法伪装成一个数据属性,在使用时可以不用加括号而直接引用。

python">class People:def __init__(self,name):self.__name=name@property #伪装成obj.namedef name(self):#返回值就是伪装的obj.name的值return '<名字是:%s>' %self.__name@name.setter #伪装name的修改方式obj.name=值def name(self,name):if type(name) is not str:raise TypeError('名字必须是str类型')self.__name=name@name.deleter #伪装name的删除方式del obj.namedef name(self):del self.__namepeo1=People('111')
print(peo1.name)
peo1.name='222'
print(peo1.name)
del peo1.name
print(peo1.name)<<< <名字是:111>
<<< <名字是:222>
<<< AttributeError: 'People' object has no attribute '_People__name'

三、绑定方法与非绑定方法

绑定方法:绑定给谁就应该由谁来调用,谁来调用就会将谁当作第一个参数自动传入
绑定方法分为两类:

  • 绑定给对象方法:在类内部定义的函数(没有被任何装饰器修饰的),默认就是绑定给对象用的
  • 绑定给类的方法:在类内部定义的函数如果被装饰器@classmethod装饰,那么则是绑定给类的,应该由类来调用,类来调用就自动将类当作第一个参数自动传入。

非绑定方法:类中定义的函数如果被装饰器@staticmethod装饰,那么该函数就变成非绑定方法,既不与类绑定,又不与对象绑定,意味着类与对象都可以来调用,但是无论谁来调用,都没有任何自动传值的效果,就是一个普通函数。

python">class Mysql:def __init__(self, ip):self.ip=ip#绑定给对象的方法,第一个值为self,默认是给对象调用的def tell_info(self):print('%s'%self.ip)#绑定给类的方法,第一个值为cls,默认是给类调用的@classmethoddef from_conf(cls):#cls就是Mysql,cls('127.0.0.1')这是一种在类内部完成实例化的方式return cls('127.0.0.1')#不绑定的方法,对象和类都可以调用@staticmethoddef func():print('不与任何人绑定')m=Mysql('172.30.156.0')
m.tell_info()
#绑定给对象的方法一般都是通过类调用的
#若是通过对象调用m.from_conf().tell_info(),会将对象对应的类传给cls
Mysql.from_conf().tell_info()
m.func()
Mysql.func()<<<172.30.156.0
<<<127.0.0.1
<<<不与任何人绑定
<<<不与任何人绑定

http://www.ppmy.cn/server/127391.html

相关文章

Linux:无法为立即文档创建临时文件: 设备上没有空间

虚拟机磁盘空间不足解决记录 1、问题描述2、问题解决 1、问题描述 在命令行输入命令按Tab键时出现如下报错&#xff1a; 很明显&#xff0c;设备上没有空间&#xff0c;即磁盘空间不足。通过命令查看具体情况如下&#xff1a; df -h2、问题解决 首先想到的是虚拟机扩容。关机虚…

STM32 通用定时器

一、概述 STM32内部集成了多个定时/计数器&#xff0c;根据型号不同&#xff0c;STM32系列芯片最多包含8个定时/计数器。其中&#xff0c;TIM6、TIM7为基本定时器&#xff0c;TIM2~TIM5为通用定时器&#xff0c;TIM1、TIM8为高级控制定时器。 1.定时器的类型 基本定时器通用定…

以串口接口为例介绍关于BSP底层架构开发的迭代过程

以串口接口为例介绍关于BSP底层架构开发的迭代过程 文章目录 以串口接口为例介绍关于BSP底层架构开发的迭代过程架构概述初代BSP二代BSP:三代BSP:四代BSP:架构概述 单片机开发有四个阶段: 阶段一:单一单片机的功能实现阶段 此阶段你开始熟悉STM32F1系列的单片机,并利用…

探索Python的工业通信之光:pymodbus的奇妙之旅

文章目录 探索Python的工业通信之光&#xff1a;pymodbus的奇妙之旅背景&#xff1a;为何选择pymodbus&#xff1f;pymodbus是什么&#xff1f;如何安装pymodbus&#xff1f;5个简单的库函数使用方法3个场景使用示例常见bug及解决方案总结 探索Python的工业通信之光&#xff1a…

Linux下Socket编程

1. Socket简介 Socket是什么&#xff1f; Socket是一种进程间通信的机制&#xff0c;通过它应用程序可以通过网络进行数据传输。Socket提供了一种跨平台的接口&#xff0c;使得同样的代码可以在不同的操作系统上运行。Socket类型 流式套接字&#xff08;SOCK_STREAM&#xff0…

Linux中的 `vi` 与 `vim` 使用详解

文章目录 Linux中的 vi 与 vim 使用详解1. vi 编辑器1.1 什么是 vi1.2 vi 的基本用法1.2.1 启动 vi1.2.2 模式1.2.3 基本操作1.2.4 常用命令 1.3 vi 的特点 2. vim 编辑器2.1 什么是 vim2.2 vim 的基本用法2.2.1 启动 vim2.2.2 模式2.2.3 vim 的增强功能2.2.4 vim 的基本操作 2…

Oracle 配置恢复目录catalog

一.介绍 Oracle中使用RMAN备份的数据我们分为两类 RMAN知识库数据库的数据块 Oracle默认把 RMAN知识库 放在目标数据库的控制文件中&#xff0c;在以后进行恢复的时候 我们要先读知识库的信息然后才能恢复。 但这样就产生了一个问题&#xff0c;知识库放在了控制文件上&#xf…

国庆更新|芒果YOLOv8改进181:即插即用,最新注意力机制EMA:具有跨空间学习的高效多尺度注意力模块,ICCASSP论文

💡本篇内容:芒果YOLOv8改进135:最新注意力机制EMA:即插即用,具有跨空间学习的高效多尺度注意力模块,ICCASSP 论文 **EMA|具有跨空间学习的高效多尺度注意力模块 | 即插即用 该模块通常包括多个并行的注意力子模块,每个子模块关注于输入数据的不同尺度或分辨率。这些子模块…