Python之面向对象:孔乙己,实现单例模式,我有4种方法
off999 2024-09-23 11:31 63 浏览 0 评论
引言
前面的一系列文章中,已经花了很大的篇幅来介绍Python中面向对象的内容,实际场景中高频使用的、不那么高频使用的,基本都介绍到了。我打算把今天这篇文章作为面向对象模块的收尾。
本文将首先回顾一下面向对象模块的主要内容,然后重点介绍基于前面介绍的知识,实现单例模式。
面向对象内容梳理
关于面相对象的内容,其实我们主要介绍了三大特性、四种数据封装、三种方法、三种类。其中:
三大特性:
1、封装:将数据与算法封装为一个整体,也可以理解为状态与操作的封装。
2、继承:更高级别的代码复用,子类直接拿来即用,强调一组类的共性的部分。
3、多态:子类对父类的方法的不同实现,强调子类可以丰富多彩的个性的部分。
四种数据封装:
1、实例属性:每个实例对象相互独立,可以进行私有化,虽然是“名称混淆”的实现机制。
2、类属性:归属于类对象,所有实例对象共享该实例,同样可以私有化。
3、property:实现数据保护、添加新的校验、处理逻辑,同时兼容直接基于属性名访问的历史接口。
4、属性描述符:更加底层的一种数据封装机制,适用于很多个属性都要进行类似的封装逻辑的复用。
三种方法:
1、实例方法:通过实例对象进行调用,默认将实例对象作为方法的第一个参数进行自动传递,通常以self作为第一个形参的名字。
2、类方法:@classmethod,通过类名进行调用,与实例对象无关,不存在实例时,也是可以调用的,默认将类对象作为方法的第一个参数进行自动传递,通常以cls作为第一个形参的名字。
3、静态方法:从抽象、设计的角度,与类本身无太大关联,更多的是作为工具方法的一种抽离,所以,不会自动传递实例对象或者类对象,可以理解为类中定义的普通函数。
三种类:
1、普通类:通常定义的类,都是普通的类,可以随意进行实例化对象的创建,完成基础的数据和算法逻辑的封装。
2、抽象基类(abc):不能进行实例化操作,通常作为基类使用,定义相关协议,具体实现延迟到各个子类中。
3、元类:继承自type及其子类的类,用于创建类对象的类,通过元类,可以规范类定义以及完成类对象的动态增强。
其实,大部分内容都是在以不同的方式帮我们完成代码的复用,只是复用的颗粒度有所不同。毕竟,编程本身在于将重复工作的自动化转换,是帮助人们更好地偷懒。而编程语言中的各种代码复用的思想,其实是对如何偷懒的偷懒实现,似乎有点“元偷懒”的意味。
虽然花了很多的篇幅去讲解面向对象的内容,但是,真正梳理一下,也就这些东西。
单例模式的4种实现方法
抽象基类不能进行实例化,普通类可以随意进行实例化。其实,关于实例化,现实场景中,还有一种只能实例化一个对象的需求,应运而生的设计模式就是“单例模式”。
所谓“单例模式”,简单来说,就是一种设计模式,确保某个类中只能有一个实例对象,并提供一个全局访问点。
单例模式通常在资源管理、全局配置、日志记录、缓存等场景中被频繁使用,更多的是可以有效节省计算及存储资源。
方法1:使用类属性
直接看代码:
class King:
_instance = None
def __new__(cls, *args, **kwargs):
if cls._instance is None:
cls._instance = super().__new__(cls, *args, **kwargs)
return cls._instance
if __name__ == '__main__':
king1 = King()
king2 = King()
print(king1)
print(king2)
print(king1 is king2)
执行结果:
从执行结果,可以看到每次实例化,确实都是返回同一个实例化对象。
这种方式,其实是基于实例化对象之前,首先是需要调用__new__()方法的机制进行单例模式的,所以,需要对__new__()和__init__()方法的区别和触发机制有一个比较清晰的理解。如果单例的实例对象有状态(属性),稍不留意,可能会导致奇怪的事情的发生。
直接看代码:
class King:
_instance = None
def __new__(cls, *args, **kwargs):
if cls._instance is None:
cls._instance = super().__new__(cls, *args, **kwargs)
return cls._instance
def __init__(self):
print(f"虽然是同一个实例,不会被重复创建,但是初始化操作会重复")
self.name = '老虎'
if __name__ == '__main__':
king1 = King()
print(king1)
king1.name = '猴子'
print(king1.name)
king2 = King()
print(king2)
print(king1.name)
执行结果:
代码的本意是,既然是同一个实例,我如果修改了实例中的属性,返回的同一个实例,应该是修改后的属性。但是,__init__()方法会被重复调用的。
可以调整为以下的做法,从而避免重复初始化导致的状态丢失:
class King:
_instance = None
def __new__(cls, *args, **kwargs):
if cls._instance is None:
cls._instance = super().__new__(cls, *args, **kwargs)
return cls._instance
def __init__(self):
if not hasattr(self, 'initialized'):
self.initialized = True
print(f"虽然是同一个实例,不会被重复创建,但是初始化操作会重复")
self.name = '老虎'
if __name__ == '__main__':
king1 = King()
print(king1)
king1.name = '猴子'
print(king1.name)
king2 = King()
print(king2)
print(king1.name)
执行结果:
通过在实例中维持一个实例化的状态属性,从而确保实例对象只被创建一次的同时,也只实例化一次,从而避免实例对象状态的丢失。
方法2:使用元类
使用类属性的方法实现元类,比较清晰,但是,每次创建一个单例模式的类,都要重复写单例的处理逻辑,为了更好的复用,可以使用元类来介入类对象的创建过程,从而复用单例的逻辑。
直接看代码:
class SingletonMeta(type):
_instances = {}
def __call__(cls, *args, **kwargs):
if cls not in cls._instances:
cls._instances[cls] = super().__call__(*args, **kwargs)
return cls._instances[cls]
class King(metaclass=SingletonMeta):
pass
if __name__ == '__main__':
king1 = King()
print(king1)
king1.name = '猴子'
print(king1.name)
king2 = King()
print(king2)
print(king1.name)
执行结果:
通过以上两种实现方法,可以顺便回顾一下实例化对象的创建机制及元类的使用。
接下来的两种方法,涉及到我们暂未提及的Python中的内容,对新手看,可能有点超纲,所以,只是演示代码,不做过多的展开了。
方法3:使用装饰器
直接看代码:
def singleton(cls):
instances = {}
def get_instance(*args, **kwargs):
if cls not in instances:
instances[cls] = cls(*args, **kwargs)
return instances[cls]
return get_instance
@singleton
class King:
pass
@singleton
class Queen:
pass
if __name__ == '__main__':
king1 = King()
print(king1)
king2 = King()
print(king2)
queen1 = Queen()
print(queen1)
queen2 = Queen()
print(queen2)
执行结果:
方法4:使用模块的加载特性
首先在模块中定义King类,并实例化一个对象,假设模块名是singletong_module.py:
class King:
pass
king_instance = King()
print(f"模块中的king实例: {king_instance}")
然后,在需要使用该实例对象的地方进行导入:
from singleton_module import king_instance
print(f"导入的king实例: {king_instance}")
执行结果:
需要说明的是,后面的两种实现方法,分别涉及到装饰器和模块,之所以在这里也提及一下,其实,是作为后面内容的一个提前引入。
总结
本文首先回顾了Python中面向对象的相关内容,可以简单概括为:三大特性、四种数据封装、三种方法、三种类。其实,这些内容始终围绕着一个基本点:代码复用(如何更好地偷懒)。其次,通过介绍单例模式的实现方法,对面向对象的实例化对象构建及元类的核心内容进行了复习,并对后续要介绍的装饰器、模块的内容做了一个引入。
感谢您的拨冗阅读,如果对您学习Python有所帮助,欢迎点赞、关注。
相关推荐
- 安全教育登录入口平台(安全教育登录入口平台官网)
-
122交通安全教育怎么登录:122交通网的注册方法是首先登录网址http://www.122.cn/,接着打开网页后,点击右上角的“个人登录”;其次进入邮箱注册,然后进入到注册页面,输入相关信息即可完...
- 大鱼吃小鱼经典版(大鱼吃小鱼经典版(经典版)官方版)
-
大鱼吃小鱼小鱼吃虾是于谦跟郭麒麟的《我的棒儿呢?》郭德纲说于思洋郭麒麟作诗的相声,最后郭麒麟做了一首,师傅躺在师母身上大鱼吃小鱼小鱼吃虾虾吃水水落石出师傅压师娘师娘压床床压地地动山摇。...
-
- 哪个软件可以免费pdf转ppt(免费的pdf转ppt软件哪个好)
-
要想将ppt免费转换为pdf的话,我们建议大家可以下一个那个wps,如果你是会员的话,可以注册为会员,这样的话,在wps里面的话,就可以免费将ppt呢转换为pdfpdf之后呢,我们就可以直接使用,不需要去直接不需要去另外保存,为什么格式转...
-
2026-02-04 09:03 off999
- 电信宽带测速官网入口(电信宽带测速官网入口app)
-
这个网站看看http://www.swok.cn/pcindex.jsp1.登录中国电信网上营业厅,宽带光纤,贴心服务,宽带测速2.下载第三方软件,如360等。进行在线测速进行宽带测速时,尽...
- 植物大战僵尸95版手机下载(植物大战僵尸95 版下载)
-
1可以在应用商店或者游戏平台上下载植物大战僵尸95版手机游戏。2下载教程:打开应用商店或者游戏平台,搜索“植物大战僵尸95版”,找到游戏后点击下载按钮,等待下载完成即可安装并开始游戏。3注意:确...
- 免费下载ppt成品的网站(ppt成品免费下载的网站有哪些)
-
1、Chuangkit(chuangkit.com)直达地址:chuangkit.com2、Woodo幻灯片(woodo.cn)直达链接:woodo.cn3、OfficePlus(officeplu...
- 2025世界杯赛程表(2025世界杯在哪个国家)
-
2022年卡塔尔世界杯赛程公布,全部比赛在卡塔尔境内8座球场举行,2022年,决赛阶段球队全部确定。揭幕战于当地时间11月20日19时进行,由东道主卡塔尔对阵厄瓜多尔,决赛于当地时间12月18日...
- 下载搜狐视频电视剧(搜狐电视剧下载安装)
-
搜狐视频APP下载好的视频想要导出到手机相册里方法如下1、打开手机搜狐视频软件,进入搜狐视频后我们点击右上角的“查找”,找到自已喜欢的视频。2、在“浏览器页面搜索”窗口中,输入要下载的视频的名称,然后...
- 永久免费听歌网站(丫丫音乐网)
-
可以到《我爱音乐网》《好听音乐网》《一听音乐网》《YYMP3音乐网》还可以到《九天音乐网》永久免费听歌软件有酷狗音乐和天猫精灵,以前要跳舞经常要下载舞曲,我从QQ上找不到舞曲下载就从酷狗音乐上找,大多...
- 音乐格式转换mp3软件(音乐格式转换器免费版)
-
有两种方法:方法一在手机上操作:1、进入手机中的文件管理。2、在其中选择“音乐”,将显示出手机中的全部音乐。3、点击“全选”,选中所有音乐文件。4、点击屏幕右下方的省略号图标,在弹出菜单中选择“...
- 电子书txt下载(免费的最全的小说阅读器)
-
1.Z-library里面收录了近千万本电子书籍,需求量大。2.苦瓜书盘没有广告,不需要账号注册,使用起来非常简单,直接搜索预览下载即可。3.鸠摩搜书整体风格简洁清晰,书籍资源丰富。4.亚马逊图书书籍...
- 最好免费观看高清电影(播放免费的最好看的电影)
-
在目前的网上选择中,IMDb(互联网电影数据库)被认为是最全的电影网站之一。这个网站提供了各种类型的电影和电视节目的海量信息,包括剧情介绍、演员表、评价、评论等。其还提供了有关电影制作背后的详细信息,...
- 孤单枪手2简体中文版(孤单枪手2简体中文版官方下载)
-
要将《孤胆枪手2》游戏的征兵秘籍切换为中文,您可以按照以下步骤进行操作:首先,打开游戏设置选项,通常可以在游戏主菜单或游戏内部找到。然后,寻找语言选项或界面选项,点击进入。在语言选项中,选择中文作为游...
欢迎 你 发表评论:
- 一周热门
-
-
抖音上好看的小姐姐,Python给你都下载了
-
全网最简单易懂!495页Python漫画教程,高清PDF版免费下载
-
飞牛NAS部署TVGate Docker项目,实现内网一键转发、代理、jx
-
win7系统还原步骤图解(win7还原电脑系统的步骤)
-
Python 3.14 的 UUIDv6/v7/v8 上新,别再用 uuid4 () 啦!
-
python入门到脱坑 输入与输出—str()函数
-
16949认证费用是多少(16949审核员太难考了)
-
linux软件(linux软件图标)
-
Python三目运算基础与进阶_python三目运算符判断三个变量
-
windows7旗舰版多少钱(win7旗舰版要多少钱)
-
- 最近发表
- 标签列表
-
- python计时 (73)
- python安装路径 (56)
- python类型转换 (93)
- python进度条 (67)
- python吧 (67)
- python的for循环 (65)
- python格式化字符串 (61)
- python静态方法 (57)
- python列表切片 (59)
- python面向对象编程 (60)
- python 代码加密 (65)
- python串口编程 (77)
- python封装 (57)
- python写入txt (66)
- python读取文件夹下所有文件 (59)
- python操作mysql数据库 (66)
- python获取列表的长度 (64)
- python接口 (63)
- python调用函数 (57)
- python多态 (60)
- python匿名函数 (59)
- python打印九九乘法表 (65)
- python赋值 (62)
- python异常 (69)
- python元祖 (57)
