我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Python学习笔记:单例模式

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Python学习笔记:单例模式

单例模式:一个类无论实例化多少次,返回的都是同一个实例,例如:a1=A(), a2=A(), a3=A(),a1、a2和a3其实都是同一个对象,即print(a1 is a2)和print(a2 is a3)都会打印True。

实现方式:有两种方式,一种是使用元类metaclass控制类实例化时的对象,另一种是使用类的__new__方法控制类返回的对象,推荐使用元类的方式,因为__new__通常是用来改变类结构的。

注:关于元类和单例模式,本文只是贴了两个简单的示例代码和自己的一些心得,想要更加深入的学习,这里有一篇博客讲得很详细https://www.cnblogs.com/tkqasn/p/6524879.html

 

元类实现单例模式(Python3.6):

 1 class Singleton(type):
 2     def __init__(cls, *args, **kwargs):
 3         cls.__instance = None
 4         super().__init__(*args, **kwargs)
 5 
 6     def __call__(cls, *args, **kwargs):
 7         if cls.__instance is None:
 8             cls.__instance = super().__call__(*args, **kwargs)
 9 
10         return cls.__instance
11 
12 
13 class MySingleton(metaclass=Singleton):
14     def __init__(self, val):
15         self.val = val
16         print(self.val)
17 
18 
19 hello = MySingleton('hello')
20 hi = MySingleton('hi')
21 print(hello is hi)
22 
23 ----------输出结果----------
24 hello
25 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):

 1 class MySingleton:
 2     def __init__(self, val):
 3         self.val = val 
4 print(self.val)
5 print(self.__dict__) 6 7 def __new__(cls, *args, **kwargs): 8 if not hasattr(cls, '_instance'): 9 cls._instance = super().__new__(cls) 10 11 return cls._instance 12 13 14 hello = MySingleton('hello') 15 hi = MySingleton('hi') 16 print(hello is hi) 17 18 19 -----------输出结果-------------- 20 hello
21 {'val': 'hello'}
22 hi
23 {'val': 'hi'}
24 True
  • 原理:通过给类定义一个类变量,指向本类的一个实例,每次实例化调用__new__的时候都返回这个类变量,可以看到数据结果打印的是True,所以自然就是单例了。
  • 缺点:每次实例化虽然都是同一个实例,但是每次实例化都会调用一次__init__方法,导致这个实例会随着每次初始化而改变,所以不推荐这种方式来实现单例,因为__new__方法一般是用来改变类结构的。
  • hasattr:类中的私有变量,即加了双下划线的变量,在__dict__中会加上一个“_classname”前缀,所以如果这里使用__instance的话,hasattr(cls, '__instance')会一直返回False,因为这里已经不是__instance了,而是_MySingleton__instance。

 

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

Python学习笔记:单例模式

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Python学习笔记:单例模式

单例模式:一个类无论实例化多少次,返回的都是同一个实例,例如:a1=A(), a2=A(), a3=A(),a1、a2和a3其实都是同一个对象,即print(a1 is a2)和print(a2 is a3)都会打印True。实现方式:有两
2023-01-30

python学习笔记(九)、模块

1 模块  使用import 语句从外部导入模块信息,python提供了很大内置模块。当你导入模块时,你会发现其所在目录中,除源代码文件外,还新建了一个名为__pycache__的子目录(在较旧的Python版本中,是扩展名为.pyc 的文
2023-01-31

Python学习 :反射 & 单例模式

反射  什么是反射?  - 反射主要是指程序可以访问、检测和修改它本身状态或行为的一种能力(自省)  面向对象中的反射  - 通过字符串的形式来操作(获取、检查、增加、删除)对象中的成员  - python中的一切事物都是对象(都可以使用反
2023-01-31

python zipfile模块学习笔记

ZIP文件格式是一种常见的存档和压缩标准,这个zipfile模块提供了工具来创建、读取、写入、附加和列出一个ZIP文件。使用ZIP64扩展(即压缩文件大小超过4G),它能解压加密的ZIP文件,解密过程很慢。1、测试是否为ZIP文件is_zi
2023-01-31

python os模块学习笔记

import os os.mkdir(name) #创建目录 os.path.exists(name) #判断文件或者目录是否存在 os.path.isdir(name) #判断指定对象是否为目录。是True,否则Fa
2023-01-31

Python 学习笔记 (4)—— 模块

模块基本上就是一个包含了所有你定义的函数和变量的文件。为了在其他程序中重用模块,模块的文件名必须以.py为扩展名。        模块可以从其他程序 输入 以便利用它的功能。这也是我们使用Python标准库的方法。首先,我们将学习如何使用标
2023-01-31

Python学习笔记5—Python模块

python模块分为系统内置的模块、第三方的模块和用户编写的模块    默认情况下,python第三方的模块安装在python 的安装目录下site-packages下,以文件或者目录的形式存放    用户模块,程序模块化对区分功能和结构,
2023-01-31

Python学习笔记

Python介绍Python是一种解释型、面向对象的语言。官网:www.python.orgPython环境解释器:www.python.org/downloads运行方式:交互模式。在IDLE中运行。脚本模式。文件的后缀名为.py。
2023-01-30

Python 学习笔记

rs=Person.objects.all()all返回的是QuerySet对象,程序并没有真的在数据库中执行SQL语句查询数据,但支持迭代,使用for循环可以获取数据。print rs.query 会打印出原生sql语句rs=Person
2023-01-31

Python学习笔记7——文本、模块

参考书籍:《Learning_Python_5th_Edition.pdf》,一本英文书呢,我上传到百度网盘吧,请点击这里,密码是:kym3文本文件的输入输出Python具有基本的文本文件读写功能。Python的标准库提供有更丰富的读写功能
2023-01-30

python学习笔记(三)-表单处理

表单类 默认情况下,Flask-WTF能保护所有表单免受跨站请求伪造攻击(CSRF)app = Flask(__name__)# 强制性必须填写secret_keyapp.config['SECRET_KEY'] = 'hard to gu
2023-01-31

Python学习笔记:json模块和pi

Python中的json模块和pickle都是用于数据的序列化和反序列化,它们提供的方法也是一样的:dumps,dump,loads,loaddumps(obj):将对象序列化为str。dump(obj, fp):将对象序列化为str,并存
2023-01-30

python学习笔记--趣学Python

由反弹球和球拍构成的游戏。球会在屏幕上飞过来,玩家要用球拍把它弹回去画布和画弹球引入模块#Tkinter -- Python的标准GUI库,Tk 接口,是python 内置的安装包from tkinter import *import ra
2023-01-31

Python学习笔记(1)

1 def sum_args(*args):2 return sum(args))3 4 def run_with_positional_args(func, *args):5 return func(*args)6
2023-01-31

Python 学习笔记 - SQLAlc

继续上一篇SQLAlchemy的学习之旅。多对多表的创建表Host和表HostUser通过表HostToHostUser关联在一起from sqlalchemy import create_enginefrom sqlalchemy.ext
2023-01-31

python scapy学习笔记

1. ubuntu下安装gnuplot 转自:http://blog.163.com/gz_ricky/blog/static/182049118201362501316961/2.安装PyX sudo pip install pyx==0
2023-01-31

python scrapy学习笔记

scrapy是python最有名的爬虫框架之一,可以很方便的进行web抓取,并且提供了很强的定制型。一、安装scrapy# pip install scrapy二、基本使用1、初始化scrapy项目# scrapy startproject
2023-01-31

python egg学习笔记

原文链接:http://www.worldhello.net/2010/12/08/2178.html经常接触Python的同学可能会注意到,当需要安装第三方python包时,可能会用到easy_install命令。easy_install
2023-01-31

Python 学习笔记 - Memcac

Memcached是一个分布式内存对象缓存系统,他把数据缓存在内存里面来减少对数据库的访问,从而提高动态网页的访问速度。他的基本结构是key/value(键值对)。下面看看在Python里面如何使用。首先来安装一下服务器端,豆子直接在一个C
2023-01-31

Python学习笔记(matplotli

Python学习笔记--在Python中如何调整颜色和样式  参靠视频:《Python数据可视化分析 matplotlib教程》链接:https://www.bilibili.com/video/av6989413/?p=6所用的库及环境:
2023-01-30

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录