Python中的单例模式与反射机制详解

目录
  • 单例模式
  • 反射
    • hasatter
    • getattr
    • setattr
  • 总结

单例模式

一般情况下,类可以生成任意个实例,而单例模式只生成一个实例

我们先用单例模式设计一个Rectangle类

然后用__new__方法设计单例模式,代码如下

class Rectangle:
    def __init__(self,length,width):
        self.length=length
        self.width=width
    def __new__(cls, *args, **kwargs):#__new__至少要有一个参数cls,代表要实例化的类
        if not hasattr(cls,'obj'):  #判断类当中有没有实例,如果没有则新建
            cls.obj=object.__new__(cls)  #生成实例对象
        return cls.obj

然后我们来验证下,单例模式下是否只能生成一个实例

rec1 = Rectangle(10,8)#判断类中没有实例,则新建实例rec1
print(rec1.length,rec1.width)#这时候我们打印下rec1这个长方形的长和宽
>>>10 8#是我们输入的值
rec2 = Rectangle(6,4)
#Rectangle已经创建过rec1实例了,所以不会再生成新的实例,此时的rec1和rec2指向是同一个对象:Rectangle(6,4)
print(rec1.length,rec1.width)#打印下rec1的长和宽
print(rec2.length,rec2.width)#打印下rec2的长和宽
>>>
6 4
6 4
print(id(rec1)==id(rec2))
>>>Ture #对象/实例只有一个,只是赋给了不同的变量名

单例模式在程序设计中比较典型的应用场景:多个用户同时调用某个模块时,会生成一些日志,我们希望这些日志存在同一个文件内,而不是多个文件。

在生成日志模块我们就可以采用单例模式进行设计。

反射

概念:简单来说就是可以利用字符串来映射模块中的相应方法然后可以操作模块中相应的方法

我们以一个饭店点菜的实际场景来理解Python的反射机制

class Food_list:#创建一个饭店菜单的类
    def yxrs(self):
        return '鱼香肉丝'
    def gbjd(self):
        return '宫保鸡丁'
    def hspg(self):
        return '红烧排骨'

hasatter

hasatter(对象,属性或方法名)

判断对象中是否有某个属性或某个方法,返回值是布尔型

guke1 = Food_list()#实例化一个顾客对象
while True:
    diancai = input('请点菜:  ')
    if hasattr(guke1,diancai):#判断顾客点的菜有没有
        print('好的,马上去做')
        break
    else:
        print('这个没有,换个别的吧')
>>>请点菜:  佛跳墙
>>>这个没有,换个别的吧
>>>请点菜:  yxrs
>>>好的,马上去做

getattr

getattr(对象,属性或方法名,缺省值) 判断对象中是否有某个属性或某个方法,如果有返回方法本身,没有则返回缺省值

guke1 = Food_list()#实例化一个顾客对象
def fun1():#定义一个方法,用作getattr的缺省值
    return '没有这道菜'
foodname = input('请输入要翻译的菜名:')
a = getattr(guke1,foodname,fun1)
#判断guke1中有没有foodname方法,有则返回这个方法,没有则返回fun1方法
print(a())#执行返回的方法
>>>请输入要翻译的菜名:yxrs#guke1中有yxrs方法,则返回这个方法
>>>鱼香肉丝
>>>请输入要翻译的菜名:dsfsfs#guke1中有没有dsfsfs方法,则返回fun1方法
>>>没有这道菜

setattr

setattr(对象,属性,新值)

将实例的属性改为新的值,如果属性不存在则新建

我们给实例guke1加个价格属性

guke1 = Food_list()#实例化一个顾客对象
setattr(guke1,'proce','23元')
print(guke1.proce)
>>>
23元

总结

本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注我们的更多内容!

(0)

相关推荐

  • python单例模式的应用场景实例讲解

    1.应用场景 需要频繁创建和销毁的对象: 创建花费太多时间或资源,但经常使用的对象: 工具类对象: 经常访问数据库或文件的对象. 2.实例 如果每个对象中封装了相同的数据,但是需要创建多个对象的时候,而且这两个实例所有的功能是一样的,所以我们就可以使用一个实例完成,在这里我们就可以使用单例模式,如下 class Person: def __init__(self): self.name = '123123' self.age = '20' def f1(self): pass def f2(se

  • Python实现单例模式的5种方法

    目录 基本介绍 优缺点 Python实现 方式1,元类实现: 方式2,继承实现: 方式3,装饰器实现: 方式4,模块实现: 方式5,@classmethod实现单例模式: 基本介绍 一个对象只允许被一次创建,一个类只能创建一个对象,并且提供一个全局访问点. 单例模式应该是应用最广泛,实现最简单的一种创建型模式. 特点:全局唯一,允许更改 优缺点 优点: 避免对资源的多重占用,如写入文件操作 节省内存 防止命名空间被污染 缺点: 没有接口,不能继承,与单一职责原则冲突,一个类应该只关心内部逻辑,而

  • Python类反射机制使用实例解析

    这篇文章主要介绍了Python类反射机制使用实例解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 反射就是通过字符串的形式,导入模块:通过字符串的形式,去模块寻找指定函数并执行. Python有四个内置函数: 函数 功能 getattr(object, attr[, default]) 获取指定字符串名称的对象属性或方法,如果对象有该属性则返回属性值,如果有该方法则返回该方法的内存地址,如果都没有就报错,如果指定了默认值找不到不会报错会取默认

  • Python反射机制实例讲解

    目录 1. 反射的四个函数 2. 类的反射操作 3. 当前模块的反射操作 4. 其他模块反射操作 5. 反射应用场景之一 6. 反射应用场景之二 7. 总结 通常,我们操作对象的属性或者方法时,是通过点"."操作符进行的.例如下面的代码: class Person: type = "mammal" def __init__(self, name): self.name = name def say_hi(self): print('Hello, my name is

  • 小结Python的反射机制

    前言: 前两天用Python实现了ftp服务器.在小项目中就用到了反射.因此写个笔记巩固下. 反射的定义:检测和修改它本身状态或行为的一种能力(自省). 而通过反射,Python可以通过字符串的映射或修改程序运行的状态和方法. 反射的四个方法.hasattr,getattr,setattr,delattr hasattr:判断一个方法是否存在与这个类中 class Person(object): def __init__(self,name): self.name = name def talk

  • python 6种方法实现单例模式

    单例模式是一个软件的设计模式,为了保证一个类,无论调用多少次产生的实例对象,都是指向同一个内存地址,仅仅只有一个实例(只有一个对象). 实现单例模式的手段有很多种,但总的原则是保证一个类只要实例化一个对象,下一次再实例的时候就直接返回这个对象,不再做实例化的操作.所以这里面的关键一点就是,如何判断这个类是否实例化过一个对象. 这里介绍两类方式: 一类是通过模块导入的方式: 一类是通过魔法方法判断的方式: # 基本原理: - 第一类通过模块导入的方式,借用了模块导入时的底层原理实现. - 当一个模

  • Python动态导入模块和反射机制详解

    一.前言 何谓动态导入模块,就是说模块的导入可以根据我们的需求动态的去导入,不是像一般的在代码文件开头固定的导入所需的模块. 何谓反射机制,利用字符串的形式在模块或对象中操作(查找/获取/删除/添加)成员. 下面进入具体实例介绍环节.先创建一个示例文件example.py,简单写入几个加减乘除函数,如下,方便下文讲解使用. flag = 1 # 此变量在介绍反射机制时会用到 def my_sum(a, b): return a + b def my_sub(a, b): return a - b

  • python 实现单例模式的5种方法

    一.classmethod装饰器 # 全局变量 ip = '192.168.13.98' port = '3306' class MySQL: __instance = None def __init__(self, ip, port): self.ip = ip self.port = port @classmethod def instance(cls, *args, **kwargs): if args or kwargs: cls.__instance = cls(*args, **kw

  • Python中的单例模式与反射机制详解

    目录 单例模式 反射 hasatter getattr setattr 总结 单例模式 一般情况下,类可以生成任意个实例,而单例模式只生成一个实例 我们先用单例模式设计一个Rectangle类 然后用__new__方法设计单例模式,代码如下 class Rectangle: def __init__(self,length,width): self.length=length self.width=width def __new__(cls, *args, **kwargs):#__new__至

  • Python中的自省(反射)详解

    首先通过一个例子来看一下本文中可能用到的对象和相关概念. 复制代码 代码如下: #coding:  UTF-8 import sys #  模块,sys指向这个模块对象 import inspect def foo(): pass #  函数,foo指向这个函数对象   class Cat(object): #  类,Cat指向这个类对象     def __init__(self,  name='kitty'):         self.name = name     def sayHi(s

  • Java中的反射机制详解

    Java中的反射机制详解 反射,当时经常听他们说,自己也看过一些资料,也可能在设计模式中使用过,但是感觉对它没有一个较深入的了解,这次重新学习了一下,感觉还行吧! 一,先看一下反射的概念: 主要是指程序可以访问,检测和修改它本身状态或行为的一种能力,并能根据自身行为的状态和结果,调整或修改应用所描述行为的状态和相关的语义. 反射是Java中一种强大的工具,能够使我们很方便的创建灵活的代码,这些代码可以再运行时装配,无需在组件之间进行源代码链接.但是反射使用不当会成本很高! 看概念很晕的,继续往下

  • java中类加载与双亲委派机制详解

    目录 类加载是什么 类加载器 双亲委派机制 BootStrapClassLoader ExtClassLoader AppClassLoader 为什么使用双亲委派机制 全盘负责委托机制 自定义类加载器 打破双亲委派机制 类加载是什么 把磁盘中的java文件加载到内存中的过程叫做类加载 当我们用java命令运行某个类的main函数启动程序时,首先需要通过类加载器把主类加载到JVM. 有如下 User 类 package dc.dccmmtop; public Class User { publi

  • Python中__init__.py文件的作用详解

    __init__.py 文件的作用是将文件夹变为一个Python模块,Python 中的每个模块的包中,都有__init__.py 文件. 通常__init__.py 文件为空,但是我们还可以为它增加其他的功能.我们在导入一个包时,实际上是导入了它的__init__.py文件.这样我们可以在__init__.py文件中批量导入我们所需要的模块,而不再需要一个一个的导入. # package # __init__.py import re import urllib import sys impo

  • Java 反射机制详解及实例

    Java 反射机制详解及实例 反射,当时经常听他们说,自己也看过一些资料,也可能在设计模式中使用过,但是感觉对它没有一个较深入的了解,这次重新学习了一下,感觉还行吧!            一,先看一下反射的概念: 主要是指程序可以访问,检测和修改它本身状态或行为的一种能力,并能根据自身行为的状态和结果,调整或修改应用所描述行为的状态和相关的语义. 反射是Java中一种强大的工具,能够使我们很方便的创建灵活的代码,这些代码可以再运行时装配,无需在组件之间进行源代码链接.但是反射使用不当会成本很高

  • Java动态代理和反射机制详解

    反射机制 Java语言提供的一种基础功能,通过反射,我们可以操作这个类或对象,比如获取这个类中的方法.属性和构造方法等. 动态代理:分为JDK动态代理.cglib动态代理(spring中的动态代理). 静态代理 预先(编译期间)确定了代理者与被代理者之间的关系,也就是说,若代理类在程序运行前就已经存在了,这种情况就叫静态代理 动态代理 代理类在程序运行时创建的代理方式.也就是说,代理类并不是在Java代码中定义的,而是在运行期间根据我们在Java代码中的"指示"动态生成的. 动态代理比

  • Python 中闭包与装饰器案例详解

    项目github地址:bitcarmanlee easy-algorithm-interview-and-practice 1.Python中一切皆对象 这恐怕是学习Python最有用的一句话.想必你已经知道Python中的list, tuple, dict等内置数据结构,当你执行: alist = [1, 2, 3] 时,你就创建了一个列表对象,并且用alist这个变量引用它: 当然你也可以自己定义一个类: class House(object): def __init__(self, are

  • Python中is与==的使用区别详解

    目录 一.== 是比较两个对象的内容是否相等 二.is 比较的是两个实例对象是不是完全相同 三.使用is注意python对于小整数使用对象池存储问题 四.使用is注意python关于字符串的intern机制存储 5.python中对于None值的比较:使用is 一.== 是比较两个对象的内容是否相等 即两个对象的"值""是否相等,不管两者在内存中的引用地址是否一样. //地址一样,值也一样.所以==成立. st1 ='aaaaa' st2 = 'bbbbb' st3 = 'b

随机推荐