python设计模式之单例模式(二)
上次我们简单了解了一下什么是单例模式,今天我们继续探究。
上次的内容点这
上次们讨论的是GoF的单例设计模式,该模式是指:一个类有且只有一个对象。通常我们需要的是让实例共享一个相同的装态 比如数据库连接。Alex Martelli的建议的是开发人员应该关注状态和行为,而不是同一性,因此它也被称为Monostate(单态)模式。
Monostate单例模式的概念
先说说 MonoState 这个单词的意思,设计模式的名字都是很有意思的,因为为了方便交流、记忆,所以设计模式的命名都对 该模式的意图进行了表述。 Mono 是一个词根,英语中 Mono,Mon 都表示的是 1 的意思, state 意思为 " 状态 " 。 MonoState 的意思就是 " 单一的状态 ",MonoState 并不限制创建对象的个数,但是它的状态却只有一个状态
Monostate单例模式的使用
我们知道在python中,__dict__是用来存储对象属性的一个字典,其键为属性名,值为属性的值。所以下面我们可以使用dict来存储一个类所有对象的状态。来看下面这个例子:
class useDict:
__state={"name":"cxa"}
def __init__(self):
self.age=27
self.__dict__=self.__state
m=useDict()
m1=useDict()
m.age=23
print(m)
print(m1)
print(m.__dict__)
print(m1.__dict__)
运行以后输出结果。
<__builtin__.useDict instance at 0x7f78ceacc098>
<__builtin__.useDict instance at 0x7f78ceacc128>
{'age': 23, 'name': 'cxa'}
{'age': 23, 'name': 'cxa'}
首先我们看结果 我们发现每次对useDict实例化都会创建一个新的对象,然后我们通过m修改了age属性的值后,m1的age属性值也发生了变化。
除此之外我们还可以使用__new__方法本身来实现。
class useNew(object):
_state={}
def __new__(cls,*args,**kwargs):
obj=super(useNew,cls).__new__(cls,*args,**kwargs)
obj.__dict__=cls._state
return obj
a=useNew()
a1=useNew()
a.x=3
print(a)
print(a1)
print(a.__dict__)
print(a1.__dict__)
下面是输出结果
<useNew object at 0x7f78ceb873d0>
<useNew object at 0x7f78ceb87b50>
{'x': 3}
{'x': 3}
两种写法的效果是一样的。
元类的实现方式
元类是一个类的类,这就意味着该类是它的元类的实例。对于已经存在的类来说,当需要创建对象的时候,将调用python的特殊方法__call__,我们可以通过使用元类的__call__方法,来控制一个对象的实例化,具体看下面的例子
一个数据库连接的例子
import pymysql
class MetaSingleton(type):
_inst={}
def __call__(cls,*args,**kwargs):
if cls not in cls._inst:
cls._inst[cls]=super(MetaSingleton,cls).__call__(*args,**kwargs)
return cls._inst[cls]
class MysqlDb(metaclass=MetaSingleton):
connection=None
def conn(self):
if self.connection is None:
self.connection= pymysql.connect(host='127.0.0.1', port=3306, user='root',
passwd='Aa1234', db='user', charset='utf8mb4')
self.cursor=self.connection.cursor()
return self.cursor
d1=MysqlDb().conn()
d2=MysqlDb().conn()
单例模式的缺点
介绍了这么长时间的单例模式,也许你会有疑问,那单例模式有什么缺点呢?
虽然单例模式的效果很好但是依然存在一些问题,因为单例具有全局访问权限,所以可能出现以下问题:
- 全局变量可能在某个地方被改了但是不知道。(比如之前的那个使用__dict__的例子)。
- 可能会对同一个对象创建多个引用。由于单例只创建一个对象,因此这种情况下会对一个对象创建对个引用。
- 所有依赖于全局变量的类都会由于一个类的改变而紧密耦合为全局数据,从而可能在无意中影响另一个类。
到此为止,我们就把单例模式的相关内容介绍完了,后续也许会写关于其他设计模式的相关文章,如果你喜欢的话,转发到朋友圈和小伙儿们一起分享吧。

低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
python设计模式之单例模式(一)
单例设计模式的概念: 单例设计模式即确保类有且只有一个特定类型的对象,并提供全局访问点。一般我们操作数据库的时候为了避免统一资源产生互相冲突,创建单例模式可以维护数据的唯一性。 单例模式的特性: 确保类有且只有一个对象被创建。 为对象提供一个访问点,以使程序可以全局访问对象。 控制共享资源的并行访问。 下面是单例模式的UML图。(注:UML-Unified Model Language 统一建模语言,又称标准建模语言。是用来对软件密集系统进行可视化建模的一种语) 传统的单例模式的实现方法是,使构造函数私有化,并创建一个静态方法来完成对象的初始化,对象在第一次调用时创建,以后这个类将返回同一个对象. 单例模式的实现 网上找了一个C#的例子(其中考虑到多线程的问题)可以了解下,下面我们主要介绍Python的单例模式的使用。 ///<summary> ///单例模式的实现 ///</summary> publicclassSingleton { //定义一个静态变量来保存类的实例 privatestaticSingletonuniqueInstance; //定义一个...
- 下一篇
异步网络模块之aiohttp(一)
平时我们也许用的更多的是requests模块,或者是requests_hml模块,但是他们都属于阻塞类型的不支持异步,速度很难提高,于是后来出现了异步的grequests,开始了异步网络请求,速度得到了大大的提升,但是今天我们要说的另外的一个比较异步网络请求模块-aiohttp。 什么是aiohhtp? 要学习一个模块,首先要知道这个模块都能做什么,我们从官网上找到了关于aiohttp的解释。 Asynchronous HTTP Client/Server for asyncio and Python. Supports both Client and HTTP Server. Supports both Server WebSockets and Client WebSockets out-of-the-box without the
相关文章
文章评论
共有0条评论来说两句吧...
文章二维码
点击排行
推荐阅读
最新文章
- SpringBoot2整合Thymeleaf,官方推荐html解决方案
- CentOS7设置SWAP分区,小内存服务器的救世主
- Hadoop3单机部署,实现最简伪集群
- CentOS8安装Docker,最新的服务器搭配容器使用
- CentOS7安装Docker,走上虚拟化容器引擎之路
- CentOS8安装MyCat,轻松搞定数据库的读写分离、垂直分库、水平分库
- CentOS7编译安装Gcc9.2.0,解决mysql等软件编译问题
- Docker使用Oracle官方镜像安装(12C,18C,19C)
- CentOS7,8上快速安装Gitea,搭建Git服务器
- CentOS7编译安装Cmake3.16.3,解决mysql等软件编译问题