单例设计模式:单例模式提供这样一个机制,确保类有且只有一个特定类型的对象,并提供全局访问点。一般单例模式使用时,需要注意以下几点:

  • 确保该类有且只有一个对象被创建。
  • 需要为该对象提供一个全局访问点,以便程序可以全局访问该对象(必须保证这个访问点返回的都是同一个对象)。
  • 需要注意控制共享资源的访问。

单例模式优点:如果在程序中某个对象只需要创建一次,就可以考虑使用单例模式,比如日志操作,数据库操作等,这种对象创建多个就会非常浪费资源,也不能很好的实现这些操作的同步,所以单例模式是一个非常不错的选择。

单例模式缺点:它的缺点也很明显,因为单例是全局变量,所以使用时它可能在某个地方已经被修改了,但是开发人员可能并不知道。而且所有使用到这个单例的类,会因为这个单例成为紧密耦合的关系,从而某个类的改变可能会影响到其他的类。

实现方式:一般有两种方式,一种是使用元类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学习笔记:单例模式的更多相关文章

  1. python学习笔记整理——字典

    python学习笔记整理 数据结构--字典 无序的 {键:值} 对集合 用于查询的方法 len(d) Return the number of items in the dictionary d. 返 ...

  2. VS2013中Python学习笔记[Django Web的第一个网页]

    前言 前面我简单介绍了Python的Hello World.看到有人问我搞搞Python的Web,一时兴起,就来试试看. 第一篇 VS2013中Python学习笔记[环境搭建] 简单介绍Python环 ...

  3. python学习笔记之module && package

    个人总结: import module,module就是文件名,导入那个python文件 import package,package就是一个文件夹,导入的文件夹下有一个__init__.py的文件, ...

  4. python学习笔记(六)文件夹遍历,异常处理

    python学习笔记(六) 文件夹遍历 1.递归遍历 import os allfile = [] def dirList(path): filelist = os.listdir(path) for ...

  5. python学习笔记--Django入门四 管理站点--二

    接上一节  python学习笔记--Django入门四 管理站点 设置字段可选 编辑Book模块在email字段上加上blank=True,指定email字段为可选,代码如下: class Autho ...

  6. python学习笔记--Django入门0 安装dangjo

    经过这几天的折腾,经历了Django的各种报错,翻译的内容虽然不错,但是与实际的版本有差别,会出现各种奇葩的错误.现在终于找到了解决方法:查看英文原版内容:http://djangobook.com/ ...

  7. python学习笔记(一)元组,序列,字典

    python学习笔记(一)元组,序列,字典

  8. Pythoner | 你像从前一样的Python学习笔记

    Pythoner | 你像从前一样的Python学习笔记 Pythoner

  9. OpenCV之Python学习笔记

    OpenCV之Python学习笔记 直都在用Python+OpenCV做一些算法的原型.本来想留下发布一些文章的,可是整理一下就有点无奈了,都是写零散不成系统的小片段.现在看 到一本国外的新书< ...

  10. python学习笔记(五岁以下儿童)深深浅浅的副本复印件,文件和文件夹

    python学习笔记(五岁以下儿童) 深拷贝-浅拷贝 浅拷贝就是对引用的拷贝(仅仅拷贝父对象) 深拷贝就是对对象的资源拷贝 普通的复制,仅仅是添加了一个指向同一个地址空间的"标签" ...

随机推荐

  1. maven课程 项目管理利器-maven 3-4 eclipse安装maven插件和新建maven项目

    本节主要讲了两个主要内容, 1       eclipse安装maven插件 2 新建maven项目 3 本人实操 1       eclipse安装maven插件 eclipse4.0以上和myec ...

  2. Swift自适应布局(Adaptive Layout)教程(一)

    通用的stroyboard文件是通向自适应布局光明大道的第一步.在一个storyboard文件中适配iPad和iPhone的布局在iOS8中已不再是梦想.我们不必再为不同尺寸的Apple移动设备创建不 ...

  3. count group by 组合用法

    1 需求是 求订单表1个月内 订单累计费用超过500的有多少人 根据题意 最先写出的sql是这样的 SELECT SUM(totalfee)AS n FROM sendorder WHERE `add ...

  4. yii2.0安装ElasticSearch及使用

    yii2.0安装ElasticSearch安装及使用教程:https://www.yiichina.com/tutorial/1046 Elasticsearch 权威指南(中文版):https:// ...

  5. Request processing failed; nested exception is org.apache.ibatis.binding.BindingException: Invalid b

    Request processing failed; nested exception is org.apache.ibatis.binding.BindingException: Invalid b ...

  6. 使用sqlyog连接ubuntu mysql server错误解决方案

    现在很多服务都部署在linux环境中,但是在开发阶段,使用windows远程连接工具,直观,这对开发人员更友好. 下面是我在ubuntu16.04使用mysql- server时,遇到了一下的问题,以 ...

  7. mysql:数据库保存时间的类型——int和datetime的区别

    我们都知道,时间保存在数据库中,可以选择使用两种类型,一种是int,一种是datetime 那么,它们两个有什么区别呢?要怎么用呢? 现在和小仓鼠一起来探讨一下 1.int和datetime的使用区别 ...

  8. 【LOJ6029】「雅礼集训 2017 Day1」市场(线段树裸题)

    点此看题面 大致题意: 维护序列,支持区间加法,区间除法(向下取整),区间求\(min\)和区间求和. 线段树维护区间除法 区间加法.区间求\(min\)和区间求和都是线段树基本操作,因此略过不提. ...

  9. c# base new 等关键字基础

    base关键字 不仅可以  调用父类的 实例方法,也能狗调用父类的 构造方法 https://www.cnblogs.com/aehyok/p/3519599.html

  10. git优点缺点(简单介绍)

    什么是Git Git是目前世界上最先进的分布式版本控制系统. Git是免费.开源的 最初Git是为辅助 Linux 内核开发的,来替代 BitKeeper 作者 Linux和Git之父李纳斯·托沃兹( ...