Python学习笔记:单例模式
单例设计模式:单例模式提供这样一个机制,确保类有且只有一个特定类型的对象,并提供全局访问点。一般单例模式使用时,需要注意以下几点:
- 确保该类有且只有一个对象被创建。
- 需要为该对象提供一个全局访问点,以便程序可以全局访问该对象(必须保证这个访问点返回的都是同一个对象)。
- 需要注意控制共享资源的访问。
单例模式优点:如果在程序中某个对象只需要创建一次,就可以考虑使用单例模式,比如日志操作,数据库操作等,这种对象创建多个就会非常浪费资源,也不能很好的实现这些操作的同步,所以单例模式是一个非常不错的选择。
单例模式缺点:它的缺点也很明显,因为单例是全局变量,所以使用时它可能在某个地方已经被修改了,但是开发人员可能并不知道。而且所有使用到这个单例的类,会因为这个单例成为紧密耦合的关系,从而某个类的改变可能会影响到其他的类。
实现方式:一般有两种方式,一种是使用元类metaclass控制类实例化时的对象,另一种是使用类的__new__方法控制类返回的对象。(关于元类,这里有篇博客讲得很详细:https://www.cnblogs.com/tkqasn/p/6524879.html)
元类实现单例模式(Python3.6):
class Singleton(type):
def __init__(cls, *args, **kwargs):
cls.__instance = None
super().__init__(*args, **kwargs) def __call__(cls, *args, **kwargs):
if cls.__instance is None:
cls.__instance = super().__call__(*args, **kwargs) return cls.__instance class MySingleton(metaclass=Singleton):
def __init__(self, val):
self.val = val
print(self.val) hello = MySingleton('hello')
hi = MySingleton('hi')
print(hello is hi) ----------输出结果----------
hello
True
- metaclass:Python3中metaclass是通过指定metaclass实现的,Python2中是通过指定类变量__metaclass__来实现的,但原理都是一样的。
- type:Python中所有的类都是type类的实例,即一个类(还未实例化)的定义,其实就是type类(Python内建元类)的实例。如下的打印可以更加直观的理解这一点:
>>> int.__class__
<class 'type'>
>>> num = 3
>>> num.__class__
<class 'int'>
>>> num.__class__.__class__
<class 'type'>
>>>
>>>
>>> class A:
pass >>> A.__class__
<class 'type'>
>>> a = A()
>>> a.__class__
<class '__main__.A'>
>>> a.__class__.__class__
<class 'type'>
>>> - __call__:当调用一个实例时,即执行实例加括号的形式,就会调用该实例的__call__方法,如果没有定义(需要自己定义),则会报错。例如a=A(),a()就会调用a的__call__方法。
- 代码执行流程:第一步执行MySingleton时(即没加括号的部分),进行元类的实例化,即MySingleton=Singleton(),Singleton的实例化和普通类一样会先执行__new__返回该类的实例,然后自动执行该实例的__init__方法进行初始化,此示例中的初始化方法给该实例赋予了一个值为None的__instance变量;第二步执行MySingleton('hello')时,进行类的实例化,即MySingleton('hello')=Singleton()('hello'),这里就会调用到Singleton的__call__方法了,而super().__call__即调用type的__call__方法,这时候就和普通类实例化一样会调用MySingleton的__new__和__init__方法了。
- 原理:由于每次实例化MySingleton时都会先调用metaclass中的__call__方法,所以只有第一次实例化时才会执行MySingleton的__new__和__init__,后面的实例化都只会返回第一次实例化好的实例,所以导致的结果就是无论进行多少次实例化,都给你返回同一个实例,当然就只有单例了(所以“输出结果”中就没有打印“hi”了) 。
- cls和self:Singleton的编写,在eclipse中提示需要写成self,在PyCharm中提示需要写成cls,因为参数self是约定代表实例本身,但在这里type的实例就是类,所以推荐写成cls。
__new__实现单例模式(Python3.6):
class MySingleton:
def __init__(self, val):
self.val = val
4 print(self.val)
5 print(self.__dict__) def __new__(cls, *args, **kwargs):
if not hasattr(cls, '_instance'):
cls._instance = super().__new__(cls) return cls._instance hello = MySingleton('hello')
hi = MySingleton('hi')
print(hello is hi) -----------输出结果--------------
hello
{'val': 'hello'}
hi
{'val': 'hi'}
True
- 原理:通过给类定义一个类变量,指向本类的一个实例,每次实例化调用__new__的时候都返回这个类变量,可以看到数据结果打印的是True,所以自然就是单例了。
- 注意:每次实例化虽然都是同一个实例,但是每次实例化都会调用一次__init__方法,导致这个实例会随着每次初始化而改变,所以这种方式来实现单例就看情况和自己的需求来使用了。
- hasattr:类中的私有变量,即加了双下划线的变量,在__dict__中会加上一个“_classname”前缀,所以如果这里使用__instance的话,hasattr(cls, '__instance')会一直返回False,因为这里已经不是__instance了,而是_MySingleton__instance。
Python学习笔记:单例模式的更多相关文章
- python学习笔记整理——字典
python学习笔记整理 数据结构--字典 无序的 {键:值} 对集合 用于查询的方法 len(d) Return the number of items in the dictionary d. 返 ...
- VS2013中Python学习笔记[Django Web的第一个网页]
前言 前面我简单介绍了Python的Hello World.看到有人问我搞搞Python的Web,一时兴起,就来试试看. 第一篇 VS2013中Python学习笔记[环境搭建] 简单介绍Python环 ...
- python学习笔记之module && package
个人总结: import module,module就是文件名,导入那个python文件 import package,package就是一个文件夹,导入的文件夹下有一个__init__.py的文件, ...
- python学习笔记(六)文件夹遍历,异常处理
python学习笔记(六) 文件夹遍历 1.递归遍历 import os allfile = [] def dirList(path): filelist = os.listdir(path) for ...
- python学习笔记--Django入门四 管理站点--二
接上一节 python学习笔记--Django入门四 管理站点 设置字段可选 编辑Book模块在email字段上加上blank=True,指定email字段为可选,代码如下: class Autho ...
- python学习笔记--Django入门0 安装dangjo
经过这几天的折腾,经历了Django的各种报错,翻译的内容虽然不错,但是与实际的版本有差别,会出现各种奇葩的错误.现在终于找到了解决方法:查看英文原版内容:http://djangobook.com/ ...
- python学习笔记(一)元组,序列,字典
python学习笔记(一)元组,序列,字典
- Pythoner | 你像从前一样的Python学习笔记
Pythoner | 你像从前一样的Python学习笔记 Pythoner
- OpenCV之Python学习笔记
OpenCV之Python学习笔记 直都在用Python+OpenCV做一些算法的原型.本来想留下发布一些文章的,可是整理一下就有点无奈了,都是写零散不成系统的小片段.现在看 到一本国外的新书< ...
- python学习笔记(五岁以下儿童)深深浅浅的副本复印件,文件和文件夹
python学习笔记(五岁以下儿童) 深拷贝-浅拷贝 浅拷贝就是对引用的拷贝(仅仅拷贝父对象) 深拷贝就是对对象的资源拷贝 普通的复制,仅仅是添加了一个指向同一个地址空间的"标签" ...
随机推荐
- 处理移动端自适应布局的方法- calc()与vw
在处理移动端自适应布局时,目前前端最流行的方法应该就是使用媒体查询,来设置HTML的字体大小,然后用rem为单位对Dom的宽高进行设置,这个方法的优势在于兼容性方面很好,劣势则在于当前市场上不同的机型 ...
- iDempiere 使用指南 生产插件(Manufacturing)安装过程
Created by 蓝色布鲁斯,QQ32876341,blog http://www.cnblogs.com/zzyan/ iDempiere官方中文wiki主页 http://wiki.idemp ...
- ndk制作so库,ndk-build不是内部或外部命令。。。的错误
想了想大概就需要下面这几步: 1.下载ndk 2.配置ndk的环境变量 3.在android studio添加一些ndk的配置 4.编写c文件 5.生成so库 6.调用so库 上面提到的大部分问题你都 ...
- zabbix web端有数据但是没有图形
zabbix web端有数据但是没有图形 我遇到的情况是,在配置 zabbix 网站目录时,修改了zabbix 目录的所有者和所属组,以使得 zabbix/conf/zabbix.conf.php 文 ...
- selenium产生的垃圾文件清理
C:\Users\XXXX\AppData\Local\Temp\anonymous7822503.webdriver-profile. 这个地址就是我的本地临时文件夹中seleniumdriver的 ...
- sharepoint知识点总结
{ users.Add(value.User); } else { SPGroup group = web.Groups.GetByID(value.LookupId); groups.Add(gro ...
- check_mk检测插件 - raid监控
mk_raidstatus python版本 #!/usr/bin/env python # -*- encoding: utf-8; py-indent-offset: 4 -*- import s ...
- 【Troubleshooting 】Outlook 客户端无法显示电子邮件图像
出于安全原因,Outlook 2013/2016不会在电子邮件中显示图像,但您可以右键单击图像并选择" 下载图片 "选项.我最近注意Outlook停止显示图像,并且没有下载图像的选 ...
- 使用sqlyog连接ubuntu mysql server错误解决方案
现在很多服务都部署在linux环境中,但是在开发阶段,使用windows远程连接工具,直观,这对开发人员更友好. 下面是我在ubuntu16.04使用mysql- server时,遇到了一下的问题,以 ...
- redis网络模型
多路IO复用-非阻塞同步IO模型.见http://www.cnblogs.com/syyong/p/6231326.html 具体结构:http://blog.jobbole.com/100079/ ...