Python 用__new__方法实现单例的操作


Posted in Python onDecember 11, 2020

介绍

init 方法通常用在初始化一个类实例时候,但其实它不是实例化一个类的时候第一个被调用 的方法。当使用 Student(id, name) 这样的表达式来实例化一个类时,最先被调用的方法 其实是 new 方法。

new方法接受的参数虽然也是和init一样,但init是在类实例创建之后调用,而 new方法正是创建这个类实例的方法。

new为对象分配空间,是内置的静态方法,new在内存中为对象分配了空间也返回了对象的引用,init获得了这个引用才初始化这个实例。

示例

一个非常简单的单例

class A:
 instance = None
 def __new__(cls, *args, **kwargs):
  if cls.instance is None:
   cls.instance = super().__new__(cls)
  return cls.instance

因为new方法是一个静态方法(也就是在定义的时候就没有cls参数),所以在这里要传入一个cls参数,而且这里的new你改造过了,所以要返回爸爸的new方法。

按造这个方法改造的单例怎么new都是同一个实例,但init仍然会被执行多次,也就是创建了几个对象就调用几次初始化方法。所以还要对init再进行一些判断。

class A:
 instance = None
 init_flag = False # 初始化标记

 def __new__(cls, *args, **kwargs):
  if cls.instance is None:
   cls.instance = super().__new__(cls)
  return cls.instance

 def __init__(self):
  if A.init_flag:
   return
  print('执行了初始化方法')
  A.init_flag = True

if __name__ == '__main__':
 a = A()
 b = A()
 print(a)
 print(b)

输出结果:

执行了初始化方法

<main.A object at 0x00000210E6F09320>

<main.A object at 0x00000210E6F09320>

总结

通过重载new方法,可以比较简单地实现单例,Python还有很多有趣的内置函数,有空可以再研究研究。

补充知识:Python饿汉式和懒汉式单例模式的实现

看代码吧~

# 饿汉式
class Singleton(object):
 # 重写创建实例的__new__方法
 def __new__(cls):
  # 如果类没有实例属性,进行实例化,否则返回实例
  if not hasattr(cls, 'instance'):
   cls.instance = super(Singleton, cls).__new__(cls)
  return cls.instance

饿汉式在创建的时候就会生成实例

# 懒汉式
class Singleton(object):
 __instance = None
 def __init__(self):
  if not self.__instance:
   print('调用__init__, 实例未创建')
  else:
   print('调用__init__,实例已经创建过了:', __instance)

 @classmethod
 def get_instance(cls):
  # 调用get_instance类方法的时候才会生成Singleton实例
  if not cls.__instance:
   cls.__instance = Singleton()
  return cls.__instance

以上这篇Python 用__new__方法实现单例的操作就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持三水点靠木。

Python 相关文章推荐
Python3读取文件常用方法实例分析
May 22 Python
Python压缩解压缩zip文件及破解zip文件密码的方法
Nov 04 Python
Python 制作糗事百科爬虫实例
Sep 22 Python
Windows安装Python、pip、easy_install的方法
Mar 05 Python
Python外星人入侵游戏编程完整版
Mar 30 Python
scikit-learn线性回归,多元回归,多项式回归的实现
Aug 29 Python
python验证码图片处理(二值化)
Nov 01 Python
Python 通过爬虫实现GitHub网页的模拟登录的示例代码
Aug 17 Python
python使用布隆过滤器的实现示例
Aug 20 Python
python request 模块详细介绍
Nov 10 Python
python opencv人脸识别考勤系统的完整源码
Apr 26 Python
python函数的两种嵌套方法使用
Apr 02 Python
python实现图像高斯金字塔的示例代码
Dec 11 #Python
Pycharm plot独立窗口显示的操作
Dec 11 #Python
Python OpenCV中的numpy与图像类型转换操作
Dec 11 #Python
使用python操作lmdb对数据读取的实例
Dec 11 #Python
PyTorch 中的傅里叶卷积实现示例
Dec 11 #Python
python中append函数用法讲解
Dec 11 #Python
python实现图像随机裁剪的示例代码
Dec 10 #Python
You might like
PHP mb_convert_encoding 获取字符串编码类型实现代码
2009/04/26 PHP
PHP备份/还原MySQL数据库的代码
2011/01/06 PHP
PHP显示今天、今月、上月、今年的起点/终点时间戳的代码
2011/05/25 PHP
php gzip压缩输出的实现方法
2013/04/27 PHP
php生成html文件方法总结
2014/12/01 PHP
PHP+Ajax实现无刷新分页实例详解(附demo源码下载)
2016/04/07 PHP
微信网页授权(OAuth2.0) PHP 源码简单实现
2016/08/29 PHP
PHP getNamespaces()函数讲解
2019/02/03 PHP
javascript 设为首页与加入收藏兼容多浏览器代码
2011/01/11 Javascript
js获取浏览器的可视区域尺寸的实现代码
2011/11/30 Javascript
在浏览器窗口上添加遮罩层的方法
2012/11/12 Javascript
详解JavaScript中undefined与null的区别
2014/03/29 Javascript
Jquery Ajax方法传值到action的方法
2014/05/11 Javascript
判断window.onload是否多次使用的方法
2014/09/21 Javascript
JS实现让网页背景图片斜向移动的方法
2015/02/25 Javascript
JS实现的数组全排列输出算法
2015/03/19 Javascript
AngularJS 基础ng-class-even指令用法
2016/08/01 Javascript
javascript自执行函数
2017/02/10 Javascript
基于Bootstrap下拉框插件bootstrap-select使用方法详解
2018/08/07 Javascript
[11:33]DAC2018 4.5SOLO赛决赛 MidOne vs Paparazi第二场
2018/04/06 DOTA
详解Python各大聊天系统的屏蔽脏话功能原理
2016/12/01 Python
matplotlib绘制动画代码示例
2018/01/02 Python
Python的SimpleHTTPServer模块用处及使用方法简介
2018/01/22 Python
Anaconda下安装mysql-python的包实例
2018/06/11 Python
python 字典 按key值大小 倒序取值的实例
2018/07/06 Python
对Python 窗体(tkinter)树状数据(Treeview)详解
2018/10/11 Python
Python实现获取汉字偏旁部首的方法示例【测试可用】
2018/12/18 Python
Django QuerySet查询集原理及代码实例
2020/06/13 Python
amazeui时间组件的实现示例
2020/08/18 HTML / CSS
佳能英国官方网站:Canon UK
2017/08/08 全球购物
四年大学自我鉴定
2014/02/17 职场文书
校园学雷锋活动月总结
2014/03/09 职场文书
《二泉映月》教学反思
2014/04/15 职场文书
学校食堂管理制度
2015/08/04 职场文书
原生CSS实现文字无限轮播的通用方法
2021/03/30 HTML / CSS
Windows server 2016服务器基本设置
2022/08/14 Servers