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 相关文章推荐
Python中logging模块的用法实例
Sep 29 Python
Python爬取读者并制作成PDF
Mar 10 Python
python使用Queue在多个子进程间交换数据的方法
Apr 18 Python
解读Python中degrees()方法的使用
May 18 Python
Python中有趣在__call__函数
Jun 21 Python
matplotlib在python上绘制3D散点图实例详解
Dec 09 Python
python调用staf自动化框架的方法
Dec 26 Python
windows系统中Python多版本与jupyter notebook使用虚拟环境的过程
May 15 Python
Python之Class&amp;Object用法详解
Dec 25 Python
pytorch dataloader 取batch_size时候出现bug的解决方式
Feb 20 Python
利用OpenCV中对图像数据进行64F和8U转换的方式
Jun 03 Python
Python first-order-model实现让照片动起来
Jun 25 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
封装一个PDO数据库操作类代码
2009/09/09 PHP
PHP 抓取新浪读书频道的小说并生成txt电子书的代码
2009/12/18 PHP
PHP错误提示的关闭方法详解
2013/06/23 PHP
php+MySQL判断update语句是否执行成功的方法
2014/08/28 PHP
php+ajax实现商品对比功能示例
2019/04/13 PHP
Laravel使用模型实现like模糊查询的例子
2019/10/24 PHP
用Laravel轻松处理千万级数据的方法实现
2020/12/25 PHP
jQuery学习总结之元素的相对定位和选择器(持续更新)
2011/04/26 Javascript
JS写的贪吃蛇游戏(个人练习)
2013/07/08 Javascript
我用的一些Node.js开发工具、开发包、框架等总结
2014/09/25 Javascript
JS组件Bootstrap Table使用方法详解
2016/02/02 Javascript
js停止冒泡和阻止浏览器默认行为的简单方法
2016/05/15 Javascript
JS排序之选择排序详解
2017/04/08 Javascript
Centos6.8下Node.js安装教程
2017/05/12 Javascript
vue监听scroll的坑的解决方法
2017/09/07 Javascript
微信小程序下拉刷新界面的实现
2017/09/28 Javascript
使用Vue写一个datepicker的示例
2018/01/27 Javascript
手把手教你vue-cli单页到多页应用的方法
2018/05/31 Javascript
小程序分享模块超级详解(推荐)
2019/04/10 Javascript
Python中处理时间的几种方法小结
2015/04/09 Python
python输出指定月份日历的方法
2015/04/23 Python
Pycharm中切换pytorch的环境和配置的教程详解
2020/03/13 Python
python使用Thread的setDaemon启动后台线程教程
2020/04/25 Python
PythonPC客户端自动化实现原理(pywinauto)
2020/05/28 Python
python语音识别指南终极版(有这一篇足矣)
2020/09/09 Python
用html5绘制折线图的实例代码
2016/03/25 HTML / CSS
Sunglasses Shop德国站:欧洲排名第一的太阳镜网站
2017/08/01 全球购物
贝尔帐篷精品店:Bell Tent Boutique
2019/06/12 全球购物
财务人员个人求职信范文
2013/12/04 职场文书
日化店促销方案
2014/03/26 职场文书
超市仓管员岗位职责
2014/04/07 职场文书
建筑工地门卫岗位职责
2014/04/30 职场文书
中学教师师德师风承诺书
2015/04/28 职场文书
致短跑运动员加油稿
2015/07/21 职场文书
给领导敬酒词
2015/08/12 职场文书
2016年学习雷锋精神广播稿
2015/12/17 职场文书