python单例的两种实现方法介绍(附代码)


本文摘自php中文网,作者不言,侵删。

本篇文章给大家带来的内容是关于python单例的两种实现方法介绍(附代码),有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。

这两天在看自己之前写的代码,所以正好把用过的东西整理一下,单例模式,在日常的代码工作中也是经常被用到,

所以这里把之前用过的不同方式实现的单例方式整理一下

装饰器的方式

这种方式也是工作中经常用的一种,用起来也比较方便,代码实现如下

1

2

3

4

5

6

7

8

def Singleton(cls):

    _instance = {}

    def _singleton(*args, **kwargs):

        if cls not in _instance:

            _instance[cls] = cls(*args, **kwargs)

        return _instance[cls]

 

    return _singleton

如果我们工作的一个类需要用单例就通过类似下面的方式实现即可:

1

2

3

4

@Singleton

class A(object):   

def __init__(self, x):

     self.x = x

我个人还是挺喜欢这种方式的

类的方式实现

这里其实有一些问题就需要注意了,先看一下可能出现的错误代码

1

2

3

4

5

6

class Member(object):

    @classmethod

    def instance(cls, *args, **kwargs):

        if not hasattr(Member, "_instance"):

            Member._instance = Member(*args, **kwargs)

        return Member._instance

乍一看这个类好像已经实现了单例,但是这里有一个潜在的问题,就是如果是多线程的情况,这样写就会有问题了,尤其是在当前类的初始化对象里有一些耗时操作时候

例如下面代码:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

#! /usr/bin/env python3

# .-*- coding:utf-8 .-*-

import time

import threading

import random

class Member(object):

    def __init__(self):

        time.sleep(random.randint(1,3))

    @classmethod

    def instance(cls, *args, **kwargs):

        if not hasattr(Member, "_instance"):

            Member._instance = Member(*args, **kwargs)

        return Member._instance

def task(arg):

    obj = Member.instance()

    print(obj)

for i in range(5):

    t = threading.Thread(target=task, args=[i,])

    t.start()

这段代码的执行结果会出现实例化了多个对象,导致你写的单例就没起到作用

当然自然而然我们会想起加锁,通过锁来控制,所以我们将上面代码进行更改:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

#! /usr/bin/env python3

# .-*- coding:utf-8 .-*-

import time

import threading

import random

class Member(object):

    _instance_lock = threading.Lock()

    def __init__(self):

        i = random.randint(1, 3)

        print(i)

        time.sleep(i)

    @classmethod

    def instance(cls, *args, **kwargs):

        with Member._instance_lock:

            if not hasattr(Member, "_instance"):

                Member._instance = Member(*args, **kwargs)

        return Member._instance

def task():

    obj = Member.instance()

    print(obj)

 

for i in range(5):

    threading.Thread(target=task,).start()

但是上面的代码还有一个问题,就是当我们已经实例化过之后每次调用instance都会去请求锁,所以这点并不好,所以我们将这部分代码再次更改:

1

2

3

4

5

6

7

@classmethod

    def instance(cls, *args, **kwargs):

        if not hasattr(Member, "_instance"):

            with Member._instance_lock:

                if not hasattr(Member, "_instance"):

                    Member._instance = Member(*args, **kwargs)

        return Member._instance

这样就很好的实现一个可以多线程使用的单例

以上就是本篇文章的全部内容,关于python更多精彩内容大家可以关注php中文网的Python视频教程和python文章教程栏目!!!

以上就是python单例的两种实现方法介绍(附代码)的详细内容,更多文章请关注木庄网络博客!!

相关阅读 >>

Python3读取excel数据存入mysql的方法

学习Python使用自定义钉钉机器人的示例代码

Python怎么查看变量类型

Python中定义字符串

Python numpy中nonzero()应该如何使用

Python常用类型转换实现

pycharm自带Python

Python 爬虫网页登录的实现

Python 限制函数调用次数

实例详解Python人脸识别

更多相关阅读请进入《Python》频道 >>




打赏

取消

感谢您的支持,我会继续努力的!

扫码支持
扫码打赏,您说多少就多少

打开支付宝扫一扫,即可进行扫码打赏哦

分享从这里开始,精彩与您同在

评论

管理员已关闭评论功能...