博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Python中的魔法方法
阅读量:5026 次
发布时间:2019-06-12

本文共 4083 字,大约阅读时间需要 13 分钟。

1、什么是魔法方法?

     魔法方法就是可以给你的类增加魔力的特殊方法,如果你的对象实现(重载)了这些方法中的某一个,那么这个方法就会在特殊的情况下被 Python 所调用,你可以定义自己想要的行为,而这一切都是自动发生的。它们经常是两个下划线包围来命名的(比如 __init__,__lt__),Python的魔法方法是非常强大的,所以了解其使用方法也变得尤为重要!

2、__init__(self[, ...]),__new__(cls[, ...]),__del__(self)

  __init__ 构造器,当一个实例被创建的时候初始化的方法。但是它并不是实例化调用的第一个方法,__new__才是实例化对象调用的第一个方法,它只取下 cls参数,并把其他参数传给 __init__。 __new__很少使用,但是也有它适合的场景,尤其是当类继承自一个像元组或者字符串这样不经常改变的类型的时候。

  __new__ 使用时注意以下四点:

1. __new__ 是在一个对象实例化的时候所调用的第一个方法2. 它的第一个参数是这个类,其他的参数是用来直接传递给 __init__ 方法3. __new__ 决定是否要使用该 __init__ 方法,因为 __new__ 可以调用其他类的构造方法或者直接返回别的实例对象来作为本类的实例,如果 __new__ 没有返回实例对象,则 __init__ 不会被调用4. __new__ 主要是用于继承一个不可变的类型比如一个 tuple 或者 string5. __new__ return的是一个构建的实例

  

  __new__实现单例模式:

class Person(object):    def __init__(self, name, age):        self.name = name        self.age = age    def __new__(cls, *args, **kwargs):        if not hasattr(cls,'instance'):            cls.instance = super(Person,cls).__new__(cls)        return cls.instancea = Person('p1',20)b = Person('p2',21)print(a == b)#这里的打印结果是True,可见a,和b都是同一个实例#单例作用:#第一、控制资源的使用,通过线程同步来控制资源的并发访问;#第二、控制实例产生的数量,达到节约资源的目的。#第三、作为通信媒介使用,也就是数据共享,它可以在不建立直接关联的条件下,让多个不##相关的两个线程或者进程之间实现通信。#比如,数据库连接池的设计一般采用单例模式,数据库连接是一种数据库资源。

  __del__ 析构器,当实例被销毁时调用

3、__call__(self[,args ...]),__getitem__(self,key),__setitem__(self,key,value)

  __call__ 允许一个类的实例像函数一样被调用

class Person(object):    def __init__(self, name, age):        self.name = name        self.age = age        self.instance = add    def __call__(self,*args):        return self.instance(*args)def add(args):    return args[0] + args[1]a = Person('p1', 20)print(a([1,2]))#这里将打印 3 #可见当创建a这个对象之后,如果定义了__call__函数则对象是可以像函数一样调用的

  __getitem__ 定义获取容器中指定元素的行为,相当于self[key]

class Person(object):    def __init__(self, name, age):        self.name = name        self.age = age        self._registry = {            'name': name,            'age': age        }    def __call__(self, *args):        return self.instance(*args)    def __getitem__(self, key):        if key not in self._registry.keys():            raise Exception('Please registry the key:%s first !' % (key,))        return self._registry[key]a = Person('p1', 20)print(a['name'],a['age'])#这里打印的是 'p1' 20 #可见__getitem__使实例可以像字典一样访问

  __setitem__ 设置容器中指定元素的行为,相当于self[key] = value 

4、__getattr__(self,name),__getattribute__(self,name),__setattr__(self,name,value),__delattr__(self,name),__get__(self,instance,owner),__set__(self,instance,value),__delete__(self,instance)

  __getattr__ 定义当用户试图访问一个不存在属性的时候的行为

  __setattr__ 定义当一个属性被设置的时候的行为

  __getattribute__ 定义当一个属性被访问的时候的行为

  

class Person(object):    def __init__(self, name, age):        self.name = name        self.age = age        self._registry = {            'name': name,            'age': age        }    def __getattr__(self, item):        print("don't have the attribute ",item)        return False    def __setattr__(self, key, value):        self.__dict__[key] = value    def __getattribute__(self, item):        #注意此处不要用 self.__dict__[item]        #因为self.__dict__依然会被__getattribute__拦截 这样就会陷入循环        return object.__getattribute__(self,item)a = Person('p1', 20)print(a.hh)        #这里会打印 don't have the attribute hh 以及 Falsea.hh = 'fdf'        #这里设置该属性值为'fdf'print(a.hh)           #这里将打印出 fdf

  __delattr__ 定义当一个属性被删除的时候的行为

  __get__  定义当描述符的值被取得的时候的行为

  __set__ 定义当描述符的值被设置的时候的行为

  __delete__ 定义当描述符的值被删除的时候的行为

class Descriptor(object):    def __init__(self):        self.des = None    def __get__(self, instance, owner):        return instance.__dict__.get(self.des,None)    def __set__(self, instance, value):        instance.__dict__[self.des] = valueclass Person(object):    des = Descriptor() #这里的Descriptor就是一个描述符类    def __init__(self, name, age):        self.name = name        self.age = age        self._registry = {            'name': name,            'age': age        }a = Person('p1', 20)a.des = 10    #这里会调用Descriptor的__set__方法print(a.des)   #这里会调用Descriptor的__get__方法

  关于描述符有疑惑的朋友可以自己百度也可参考这篇文章 http://www.geekfan.net/7862/ ,网上也有很多相关的文章,这里的重点不是描述符所以这里不会有关于描述符过多的解释!

  

  

  

转载于:https://www.cnblogs.com/styier/p/6111370.html

你可能感兴趣的文章
Android Studio默认快捷键
查看>>
发布开源库到JCenter所遇到的一些问题记录
查看>>
第七周作业
查看>>
函数式编程与参数
查看>>
flush caches
查看>>
SSAS使用MDX生成脱机的多维数据集CUB文件
查看>>
ACM_hdu1102最小生成树练习
查看>>
MyBatis源码分析(一)--SqlSessionFactory的生成
查看>>
android中ListView点击和里边按钮或ImageView点击不能同时生效问题解决
查看>>
CTF常用工具之汇总
查看>>
java的面向对象 (2013-09-30-163写的日志迁移
查看>>
HDU 2191 【多重背包】
查看>>
51nod 1433 0和5【数论/九余定理】
查看>>
【AHOI2013复仇】从一道题来看DFS及其优化的一般步骤和数组分层问题【转】
查看>>
less 分页显示文件内容
查看>>
如何对数据按某列进行分层处理
查看>>
[Qt] this application failed to start because it could not find or load the Qt platform plugin
查看>>
Git Submodule管理项目子模块
查看>>
学会和同事相处的30原则
查看>>
NOJ——1568走走走走走啊走(超级入门DP)
查看>>