python中的魔法方法

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.instance

a = 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 以及 False
a.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] = value

class 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/ ,网上也有很多相关的文章,这里的重点不是描述符所以这里不会有关于描述符过多的解释!

  

  

  

时间: 2024-10-12 08:35:57

python中的魔法方法的相关文章

Python中的sort()方法的使用

本文和大家分享的主要是python中的sort()方法相关知识,一起来看看吧,希望对大家学习python有所帮助. 一.基本形式 sorted(iterable[, cmp[, key[, reverse]]]) iterable.sort(cmp[, key[, reverse]]) 参数解释: (1)iterable指定要排序的list或者iterable,不用多说: (2)cmp为函数,指定排序时进行比较的函数,可以指定一个函数或者lambda函数,如: students为类对象的list

Python中的__new__()方法与实例化

@Python中的__new__()方法与实例化 __new__()是在新式类中新出现的方法,它作用在构造方法建造实例之前,可以这么理解,在Python 中 存在于类里面的构造方法__init__()负责将类的实例化,而在__init__()启动之前,__new__()决定是否 要使用该__init__()方法,因为__new__()可以调用其他类的构造方法或者直接返回别的对象来作为本类 的实例. 如果将类比喻为工厂,那么__init__()方法则是该工厂的生产工人,__init__()方法接受

关于python中的特殊方法

研究了几个小时,大概对python中的特殊方法一知半解,现在写写自己的理解,以及记录一些找到的资源.待自己有比较深入理解的时候,再来更新 https://docs.python.org/3/reference/datamodel.html#special-method-names 有一篇文章是要必须推荐给读者阅读的:Python Attributes and Methods(http://www.cafepy.com/article/python_attributes_and_methods/p

python中http请求方法有哪些

在python开发(http://www.maiziedu.com/course/python-px/)中,我们一般通过get.post方式发送http请求,可是最近在使用python做接口测试,发现python中http请求方法有许多种,作者整理了一番,把python中http请求方法都总结了出来,分享给大家,具体内容如下所示:一.python自带库----urllib2python自带库urllib2使用的比较多,简单使用如下:import urllib2response = urllib2

Python中格式化format()方法详解

Python中格式化format()方法详解 Python中格式化输出字符串使用format()函数, 字符串即类, 可以使用方法; Python是完全面向对象的语言, 任何东西都是对象; 字符串的参数使用{NUM}进行表示,0, 表示第一个参数,1, 表示第二个参数, 以后顺次递加; 使用":", 指定代表元素需要的操作, 如":.3"小数点三位, ":8"占8个字符空间等; 还可以添加特定的字母, 如: 'b' - 二进制. 将数字以2为基数

Python中字典get方法的使用

Python中字典get方法的使用 说起来,这个功能是否需要学习还真是有待于讨论.其实,知道了字典这种数据结构以后,通过最基本的Python功能也能够实现一个类似的功能.不过,既然内置了这个功能那么我们就没有必要重复造车轮. 在字典内置的方法中,想说的方法为get.这个方法是通过键来获取相应的值,但是如果相应的键不存在则返回None.其实,None只是一个默认的返回值但是并不是一个不能修改的返回值.其实,如果查询失败,我们可以指定一个返回值. 上面提到的所有功能汇总在一个示范代码,具体如下: #

Python中的__new__()方法的使用

__new__() 函数只能用于从object继承的新式类. 先看下object类中对__new__()方法的定义: class object:   @staticmethod # known case of __new__   def __new__(cls, *more): # known special case of object.__new__     """ T.__new__(S, ...) -> a new object with type S, a s

python中去重的方法

在python中,有两种方法进行去重: 1)对于list数据,去重方式如下: a)原列表list_origin的内容如下: list_origin = ["aaa", "ccc", "bbb", "aaa", "ddd", "bbb"] b)进行去重,获得新的列表new_list: new_list = list(set(list_origin)) 这里的new_list=['aaa',

python中属性和方法的动态绑定

# 定义一个类 class Student(object):     pass # 实例化一个对象     s = Student() # 给这个对象绑定一个属性name s.name = 'John' print(s.name) John # 定义一个方法 def set_age(self, age):     self.age = age # 导入模块     from types import MethodType #给s这个对象绑定一个set_age的方法 s.set_age = Met