一、单列模式

单例模式(Singleton Pattern)是一种常用的软件设计模式,该模式的主要目的是确保某一个类只有一个实例存在。

如,某个服务器程序的配置信息存放在一个文件中,客户端通过一个 Config 的类来读取配置文件的信息。如果在程序运行期间,有很多地方都需要使用配置文件的内容,也就是说,很多地方都需要创建 Config 对象的实例,这就导致系统中存在多个 Config 的实例对象,而这样会严重浪费内存资源,尤其是在配置文件内容很多的情况下。事实上,类似 Config 这样的类,我们希望在程序运行期间只存在一个实例对象。

二、装饰器实现单列模式

通过python的闭包原理,装饰器能记得外层命名空间装饰器原理参考本文

def single_decorator(cls):
single = {}
def wrapper(*args, **kwargs):
if not single.get(cls.__name__):
single[cls.__name__] = cls(*args, **kwargs)
return single[cls.__name__]
return wrapper @single_decorator
class School:
def __init__(self, name, addr, capital, represent):
self.__name = name #学校名
self.__addr = addr #地址
self.__registered_capital = capital #注册资金
self.__represent = represent #法人代表 def get_info(self):
return "校名: {} 地址: {} 法人代表: {}".format(self.__name, self.__addr, self.__represent) @staticmethod
def show():
print(School._isxx) ss= School('古天乐学校', '大凉山', '100.000', '古天乐')
ss2= School('施明德', '台湾?', '1.000.000', '瞎写的')
print('实例一id: %s'%(id(ss)))
print('实例二id: %s'%(id(ss2)))
print(ss.get_info())
print(ss2.get_info()) #运行结果为:
实例一id: 397109139440
实例二id: 397109139440
校名: 古天乐学校 地址: 大凉山 法人代表: 古天乐
校名: 古天乐学校 地址: 大凉山 法人代表: 古天乐

这样就可以将这个装饰器装饰到任何一个类上,完成单列了

 

三、__new__方法+静态字段实现单列模式
class Single:
_single = None
def __init__(self, value):
self.v = value
print(self.v) def __new__(cls, *args, **kwargs):
if Single._single:
return Single._single
else:
Single._single = super(Single, cls).__new__(cls)
print("只执行一次")
return Single._single s2 = Single(2)
s3 = Single(3)
s4 = Single(4) print(id(s2))
print(id(s3))
print(id(s4)) #输出结果为
只执行一次
2
3
4
s2的id为: 278042421064
s3的id为: 278042421064
s4的id为: 278042421064
  • 和类方法相类似,可以看见,s2/s3/s4 都是同一个id 是同一个实例,尽管实例时传入的值不一样
四、静态方法+静态字段实现单列模式

平时我们在链接数据库时,一个程序,也是只需要一个链接就够了,我们不希望一个程序产生很多实例去连接到数据库,这样每个链接都操作数据库,不但浪费了内存空间,还可能导致数据修改混乱

class Sqlpool:
__single = None #创建一个静态字段
def __init__(self):
#封装数据库链接需要的信息
self.ip = "192.168.1.1"
self.port = 3306
self.pwd = "1234567"
self.username = "xxxxxx" #链接数据库
self.connect = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10] #假如创建了9个链接数据库的线 @staticmethod
def produce():
if Sqlpool.__single: #判断静态字段是否有值
return Sqlpool.__single #如果有值返回静态字段的值
else:
Sqlpool.__single = Sqlpool() #没有就实例一个对象给他
return Sqlpool.__single #将实例返回 #这样,无论创建多少个访问,对象时,都是一个对象,这样不会占用内存
conn1 = Sqlpool.produce() #创建一个访问对象
conn2 = Sqlpool.produce() #创建二个访问对象

原理都差不多。这个没有装饰器好用

五、类/类方法直接实现
class Singleton(object):

    def __init__(self):
pass @classmethod
def get(cls, *args, **kwargs):
if not hasattr(Singleton, "_single"):
Singleton._single = Singleton(*args, **kwargs)
return Singleton._single

类方法这儿实现单列模式还有优化,就不多说了,是多线程问题。

这儿本身原理和静态方法差不多。

单列模式,装饰器、new方法、类/静态方法实现单列模式的更多相关文章

  1. 第7.18节 案例详解:Python类中装饰器@staticmethod定义的静态方法

    第7.18节 案例详解:Python类中装饰器@staticmethod定义的静态方法 上节介绍了Python中类的静态方法,本节将结合案例详细说明相关内容. 一.    案例说明 本节定义了类Sta ...

  2. 第7.17节 Python类中的静态方法装饰器staticmethod 定义的静态方法深入剖析

    第7.17节  Python类中的静态方法装饰器staticmethod 定义的静态方法深入剖析 静态方法也是通过类定义的一种方法,一般将不需要访问类属性但是类需要具有的一些能力可以静态方法提供. 一 ...

  3. python装饰器2:类装饰器

    装饰器1:函数装饰器 装饰器2:类装饰器 装饰器3:进阶 本文是装饰器相关内容的第二篇,关于类装饰器. "类装饰器"有两种解读方式:用来装饰类的装饰器:类作为装饰器装饰其它东西.你 ...

  4. 类装饰器,元类,垃圾回收GC,内建属性、内建方法,集合,functools模块,常见模块

    '''''''''类装饰器'''class Test(): def __init__(self,func): print('---初始化---') print('func name is %s'%fu ...

  5. Python装饰器实现几类验证功能做法

    最近新需求来了,要给系统增加几个资源权限.尽量减少代码的改动和程序的复杂程度.所以还是使用装饰器比较科学 之前用了一些登录验证的现成装饰器模块.然后仿写一些用户管理部分的权限装饰器.比如下面这种 de ...

  6. Decorator模式 装饰器模式

    Android 使用了装饰器模式 1. 概述 若你从事过面向对象开发,实现给一个类或对象增加行为,使用继承机制,这是所有面向对象语言的一个基本特性.如果已经存在的一个类缺少某些方法,或者须要给方法添加 ...

  7. Python类中的装饰器在当前类中的声明与调用

    [本文出自天外归云的博客园] 我的Python环境:3.7 在Python类里声明一个装饰器,并在这个类里调用这个装饰器.代码如下: class Test(): xx = False def __in ...

  8. Python之路(第二十八篇) 面向对象进阶:类的装饰器、元类

    一.类的装饰器 类作为一个对象,也可以被装饰. 例子 def wrap(obj): print("装饰器-----") obj.x = 1 obj.y = 3 obj.z = 5 ...

  9. Python装饰器实现几类验证功能做法(续)

    :昨天聊了一下构造.今天试了一下.感觉昨天聊的还是不够细化.今天结合代码实现,加以一点补充. 首先观察下面这个例子 from functools import wrapsdef decorator(f ...

随机推荐

  1. Windows Server 2016-Active Directory复制概念(二)

    本章继续补充有关Active Directory复制概念,具体内容如下: 连接对象: 连接对象是一个Active Directory对象,表示从源域控制器到目标域控制器的复制连接.域控制器是单个站点的 ...

  2. python函数的用法

    python函数的用法 目录: 1.定义.使用函数 1.函数定义:def 2.函数调用:例:myprint() 3.函数可以当作一个值赋值给一个变量 例:a=myprint()    a() 4.写r ...

  3. linux ubuntu 关于vim得一些基本命令

    1.vim显示行号 :set number 2. 快捷键 J 向下 K 往上 H 向左 L 向右 ctrl+shift+T 打开新窗口 ctrl+Page Down 所有vim窗口向下切换 ctrl+ ...

  4. LeetCode算法题-Move Zeroes(Java实现-三种解法)

    这是悦乐书的第201次更新,第211篇原创 01 看题和准备 今天介绍的是LeetCode算法题中Easy级别的第67题(顺位题号是283).给定一个数组nums,写一个函数将所有0移动到它的末尾,同 ...

  5. #004 C语言大作业学生管理系统试着做

    链表不会用 文件不会使 在这种情况下就边写边做 希望这个月能做完这个作业 #include<stdio.h> #include<stdlib.h> #include<st ...

  6. 为什么一定要学习linux系统?

    为什么一定要学习linux系统? linux诞生了这么多年,以前还喊着如何能取代windows系统,现在这个口号已经小多了,任何事物发展都有其局限性都有其天花板.就如同在国内再搞一个社交软件取代腾讯一 ...

  7. Springboot 实现api校验和登录验证

    https://blog.csdn.net/qq_36085004/article/details/83348144 文章目录 API校验 场景 实现思路 代码 拦截器: 拦截器注册: 登录token ...

  8. 【angularjs】使用ionic+angular 搭建移动端项目,字体适配

    解析: 首先,rem是以html为基准. 一般的,各大主流浏览器的font-size默认值为16px,此时1rem=16px.如果此时将rem与px进行换算很麻烦,比如0.75rem=12px. 为了 ...

  9. mysql之grant权限说明

    mysql中给一个用户授权如select,insert,update,delete等其中的一个或者多个权限,主要使用grant命令,格式为: 给没有用户授权 grant 权限 on 数据库对象 to ...

  10. 盘点 Oracle 11g 中新特性带来的10大性能影响

    Oracle的任何一个新版本,总是会带来大量引人瞩目的新特性,但是往往在这些新特性引入之初,首先引起的是一些麻烦,因为对于新技术的不了解.因为对于旧环境的不适应,从Oracle产品到技术服务运维,总是 ...