python常用的魔法方法(双下划线)_python

来源:脚本之家  责任编辑:小易  
目录
前言
魔法方法
__init__方法
__new__方法
__call__方法
__str___方法
__del___方法
__enter__ & __exit__方法
item系列方法
attr系列方法
单例模式
模块导入的方式
通过__new__方法
自定义元类的方式
结语 

前言

本文介绍一下python中常用的魔法方法以及面向对象中非常重要的单例模式。

魔法方法

python中一切皆对象,因为python是面向对象的编程语言。python给类和对象提供了大量的内置方法,这些内置方法也称魔法方法。这些魔法方法总是在某种条件下自动触发执行,就像魔法一样。

__init__方法

该方法是用来接收定义类时类中__new__方法返回的空对象后为空对象进行初始化的操作,没有返回值。

class Test():
    def __init__(self, name):
        self.name = name
        
    def test(self):
        print(self.name)
   
t = Test('xu')
t1 = Test('python')

__new__方法

该方法是当类被调用实例化对象时首先被触发的方法,用来实例化一个空对象并返回。

class Test():
    def __new__(cls,*args, **kwargs):
        return object.__new__(cls, *args, **kwargs) 
    
    def __init__(self, name):
        self.name = name

__call__方法

如果想让一个对象变成一个可调用对象(加括号可以调用),需要在该对象的类中定义__call__方法,调用可调用对象的返回值就是__call__方法的返回值。

class Test():
    
    def __init__(self):
        self.name = 'python'
    
    def __call__(self, *args, **kwargs):  # self是Test类的对象
        print(self)  # <__main__.Test object at 0x000001C78CE78FD0>
        print(self.name)
        
t = Test()
t()  # python

__str___方法

当对象被访问打印时触发执行,该方法必须有一个字符串类型的返回值。

class Test():
    def __init__(self, name):
        self.name = name
 
    def __str__(self):
        return self.name
   
t = Test('xu')
print(t1)  # xu

__del___方法

__del__方法是在对象被删除时自动触发,由于python的垃圾回收机制会自动清理程序中没用的资源,因此如果一个对象只是占用应用程序的资源,没有必要定义__del__方法,但是如果设计到占用系统资源的话比如打开的文件对象,由于关系到操作系统的资源,python的垃圾回收机制派不上用场的时候,就需要为对象创建__del__方法,用于对象被删除后自动触发回收操作系统资源。

class Test:
    def __init__(self):
        self.x = open('a.txt',mode='w')
        # self.x = 占用的是操作系统资源

    def __del__(self):
        print('run')
        # 发起系统调用,告诉操作系统回收相关的系统资源
        self.x.close()

obj = T()
del obj # obj.__del__() 

__enter__ & __exit__方法

使用with上下文管理时,会触发对象中的__enter__方法,并将__enter__方法的返回值赋值给as声明的变量。
with语句正常结束的时候会触发__exit__方法,该方法的三个参数分别代表异常类型、异常值和溯源信息,如果with语句代码块出现异常,则with语句后的代码都不会被执行,但是如果该方法返回值为True,异常会被清空,with代码块后的代码还会被正常执行。代码如下:

class Open:
    def __init__(self):
        self.name = 'open'

    def __enter__(self):
        print('with语句执行时会首先执行的方法,返回值会赋值给as声明的变量')
        return self.name

    def __exit__(self, exc_type, exc_val, exc_tb):
        print('with中的代码块执行完毕时执行exit')
        print(exc_type, '如果出现异常表示异常类型')
        print(exc_val, '表示异常的值')
        print(exc_tb, '表示异常的溯源信息')
        return 123  # 非零 非空 非None为真

    
with Open() as test:
    print(test)
    raise TypeError('看一下错误信息')
print('我会不会被执行呢')  # 当__exit__方法返回值为真时,会被执行,否则不会被执行

item系列方法

item系列方法包括__setitem__、__getitem__、delitem__方法,这三种方法分别会在中括号赋值/修改值、中括号取值、中括号删除值时触发,比如可以自定义一个字典类,并自定义中括号赋值、取值、删除值的方法:

class MyDict(dict):

    def __setitem__(self, key, value):
        print('执行setitem', key, value)  # 执行setitem, x, 1
        self.__dict__[key] = value

    def __getitem__(self, item):
        print('执行getitem', item)  # 执行getitem x
        print(self.__dict__[item])  # 1

    def __delitem__(self, key):
        print('执行delitem', key)  # 执行delitem x
        self.__dict__.pop(key)


d = MyDict()
d['x'] = 1
print(d['x'])
del d['x']

attr系列方法

attr系列方法包括__setattr__,__getattr__,__delattr__,__setattr__在添加/修改属性时会触发,___delattr__删除属性的时候触发,__getattr__在使用.调用属性并且属性不存在时触发。如下代码所示

class Test:
    def __init__(self):
        self.name = 'python'

    def __setattr__(self, key, value):
        print('添加/修改属性setattr')
        self.__dict__[key] = value
        # self.key = value  # 会出现无线递归,因为对象.属性会调用__setattr__方法

    def __delattr__(self, item):
        print('删除属性delattr')
        self.__dict__.pop(item)

    def __getattr__(self, item):
        print('属性不存在时调用getattr')
t = Test()
t.x = 'x'
print(t.y)
del t.x

单例模式

单例模式是一种软件设计模式,为了保证一个类无论调用多少次产生的对象都指向同一个内存地址,即仅仅只有一个对象。
实现单例模式的方式有很多,总的原则就是保证一个类只要实例化一个对象,因此关键点就是如何判断这个类是否实例化过一个对象。

这里介绍几种实现方式,供大家参考:

模块导入的方式

这种方式的原理是模块导入后只运行一次,后面再次使用该模块中的类是直接从内存中查找。

# cls_singleton.py
class Foo(object):
    pass

instance = Foo()

# test.py
import cls_singleton

obj1 = cls_singleton.instance
obj2 = cls_singleton.instance
print(obj1 is obj2)  # True

通过__new__方法

原理就是判断类是否有实力,有就直接返回,没有就保存到_instance中

class Test:

    _instance = None

    def __init__(self, name, age):
        self.name = name
        self.age = age

    def __new__(cls, *args, **kwargs):
        # if cls._instance:
        #     return cls._instance                 # 有实例则直接返回
        # else:
        #     cls._instance = super().__new__(cls) # 没有实例则new一个并保存
        #     return cls._instance                 # 这个返回是给是给init,再实例化一次,也没有关系

        if not cls._instance:                         # 这是简化的写法,上面注释的写法更容易提现判断思路
            cls._instance = super().__new__(cls)
        return cls._instance


t1 = Test('python', 18)
t2 = Test('python1', 18)
print(t1 is t2)  # True

自定义元类的方式

这种方式的原理是类调用的过程,类定义时会调用元类下的__init__,类调用(实例化对象)时会触发元类下的__call__方法。

class Mymeta(type):

    def __init__(cls, name, bases, dic):
        super().__init__(name, bases, dic)
        cls._instance = None                    # 将记录类的实例对象的数据属性放在元类中自动定义了

    def __call__(cls, *args, **kwargs):                   # 此call会在类被调用(即实例化时触发)
        if cls._instance:      # 判断类有没有实例化对象
            return cls._instance
        else:        # 没有实例化对象时,控制类造空对象并初始化
            obj = cls.__new__(cls, *args, **kwargs)
            obj.__init__(*args, **kwargs)
            cls._instance = obj             # 保存对象,下一次再实例化可以直接返回而不用再造对象
            return obj


class Test(metaclass=Mymeta):
    def __init__(self, name, age):
        self.name = name
        self.age = age


t1 = Test('python', 18)
t2 = Test('python1', 18)
print(t1 is t2)  # True

结语 

到此这篇关于python常用的魔法方法(双下划线)的文章就介绍到这了,更多相关python 魔法方法内容请搜索真格学网以前的文章或继续浏览下面的相关文章希望大家以后多多支持真格学网!

您可能感兴趣的文章:浅谈Python魔法方法详解Python常用的魔法方法总结Python常用的魔法方法Python魔法方法 容器部方法详解Python魔法方法功能与用法简介Python魔法方法详解python魔法方法-属性访问控制详解python魔法方法-属性转换和类的表示详解python魔法方法-自定义序列详解

  • 本文相关:
  • python实现列表的排序方法分享
  • python设计模式之策略模式实例详解
  • 基于sklearn实现bagging算法(python)
  • python正则表达式实现简易计算器功能示例
  • python实现剪刀石头布小游戏(与电脑对战)
  • python开启线程,在函数中开线程的实例
  • python多进程同步lock、semaphore、event实例
  • python中生成一个指定长度的随机字符串实现示例
  • pip和pygal的安装实例教程
  • python验证企业工商注册码
  • python中魔法方法加减怎么用
  • Python中魔法方法中的反运算问题
  • python 3.5中 为什么无法重写魔法方法
  • 求大神讲解下PYTHON魔法的到底有什么意义?
  • psp 龙士传说无限加强版 中文版的 所有武器入手地点 和武器介...
  • 求百变小樱魔术卡中文版全集集名和收复库洛牌名!!
  • 英雄无敌3中文版地图全开秘籍
  • 最终幻想7中文版魔石合成攻略谁有?
  • 暗影格斗2中文版所有副本角色,比如魔法师
  • gba模拟人生2上流社会中文版魔法卡片俱乐部密码
  • 求女生爱看的,有中文版的魔法类动画片
  • 网站首页网页制作脚本下载服务器操作系统网站运营平面设计媒体动画电脑基础硬件教程网络安全vbsdos/bathtahtcpythonperl游戏相关vba远程脚本coldfusionruby专题autoitseraphzonepowershelllinux shellluagolangerlang其它首页python浅谈python魔法方法详解python常用的魔法方法总结python常用的魔法方法python魔法方法 容器部方法详解python魔法方法功能与用法简介python魔法方法详解python魔法方法-属性访问控制详解python魔法方法-属性转换和类的表示详解python魔法方法-自定义序列详解python实现列表的排序方法分享python设计模式之策略模式实例详解基于sklearn实现bagging算法(python)python正则表达式实现简易计算器功能示例python实现剪刀石头布小游戏(与电脑对战)python开启线程,在函数中开线程的实例python多进程同步lock、semaphore、event实例python中生成一个指定长度的随机字符串实现示例pip和pygal的安装实例教程python验证企业工商注册码python入门教程 超详细1小时学会pycharm 2020最新永久激活码(附python 列表(list)操作方法详解python 元组(tuple)操作详解python 字典(dictionary)操作详解pycharm 2020年最新激活码(亲测python strip()函数 介绍pycharm 使用心得(一)安装和首python中使用xlrd、xlwt操作excepython 中文乱码问题深入分析对python sklearn one-hot编码详解python如何保证输入键入数字的方法python使用qrcode二维码库生成二维码方法python中return函数返回值实例用法python pow函数的底层实现原理介绍python爬虫逆向分析某云音乐加密参数的实对python 窗体(tkinter)文本编辑器(text)python中实现变量赋值传递时的引用和拷贝django修改models重建数据库的操作python爬虫和反爬技术过程详解
    免责声明 - 关于我们 - 联系我们 - 广告联系 - 友情链接 - 帮助中心 - 频道导航
    Copyright © 2017 www.zgxue.com All Rights Reserved