网易春招笔试问答题之单例模式

题目描述

请列出使用python实现singleton模式, 方法越多越好.

题目分析

题目很短, 但是涉及到方法越多越好就得绞尽脑汁了。
这里写图片描述

What 单例模式?

保证一个类仅有一个实例,并提供一个访问它的全局访问点。

Why 单例模式?

对于系统中的某些类来说,只有一个实例很重要,例如,一个系统中可以存在多个打印任务,但是只能有一个正在工作的任务;一个系统只能有一个窗口管理器或文件系统;一个系统只能有一个计时工具或ID(序号)生成器。如在Windows中就只能打开一个任务管理器。如果不使用机制对窗口对象进行唯一化,将弹出多个窗口,如果这些窗口显示的内容完全一致,则是重复对象,浪费内存资源;如果这些窗口显示的内容不一致,则意味着在某一瞬间系统有多个状态,与实际不符,也会给用户带来误解,不知道哪一个才是真实的状态。因此有时确保系统中某个对象的唯一性即一个类只能有一个实例非常重要。

How 单例模式?

  • 使用模块
  • 使用__new__
  • 使用装饰器(decorator)
  • 使用元类(metaclass)

使用模块

python的模块就是天然的单例模式,因为在模块第一次导入时,会生成.pyc文件, 当第二次导入时, 就会直接加载.pyc文件,而不是再次执行模块代码。

因此, 我们只需把相关的函数和数据定义在一个模块中,就可以获得一个单例的对象

通过装饰器(最好的方法)

#!/usr/bin/env python
# coding:utf-8

"""
@Name: method01.py
@Author: lvah
@Date:4/12/18
@Connect: xc_guofan@163.com
@Desc:




"""


def singleton(class_):  # class_ = MyClass
    """
    单例模式的装饰器
    :param class_:  类名称
    :return: 函数名getinstance
    """
    instances = {}

    def getinstance(*args, **kwargs):
        """
        判断class_<MyClass>是否实例化过;
            该类未曾实例化过, 实例化该类,将类名和实例化对象分别作为key和value存储到instances字典中;
            该类已经实例化过, 返回之前类对应的对象;
        """
        if class_ not in instances:
            instances[class_] = class_(*args, **kwargs)
        return instances[class_]

    return getinstance


@singleton  # MyClass = singleton(MyClass)
class MyClass(object):
    pass


def test():
    a = MyClass()
    b = MyClass()

    print a
    print b

    # 类被singleton装饰器装饰, 返回True;
    # 反之返回False;
    print a is b


if __name__ == "__main__":
    test()
  • 总结
    • 被singleton装饰的类, 实质上是个函数, 而不是类;
    • m=MyClass(); n=MyClass(); o=type(m)(); m==n && m!=o && n!=o;

metaclass

class Singleton(type):
    """
    实现单例模式的元类;

    metaclass的主要任务是: 拦截类, 修改类, 返回类
    """
    _instance = {}

    def __call__(cls, *args, **kwargs):
        if cls not in cls._instance:
            cls._instance[cls] = super(Singleton, cls).__call__(*args, **kwargs)
        return cls._instance[cls]




class MyClass(object):
    # __metaclass__ = Singleton
    pass




def test():
    a = MyClass()
    b = MyClass()

    print a is b



if __name__ == "__main__":
    test()
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值