python魔术方法总结

时间:2020-12-09 18:31:32   收藏:0   阅读:21

获取属性

实例对象属性寻找的顺序如下:

① 首先访问 getattribute() 魔法方法(隐含默认调用,无论何种情况,均会调用此方法)

② 去实例对象t中查找是否具备该属性: t.dict 中查找,每个类和实例对象都有一个 dict 的属性

③ 若在 t.dict 中找不到对应的属性, 则去该实例的类中寻找,即 t.class.dict

④ 若在实例的类中也招不到该属性,则去父类中寻找,即 t.class.bases.__dict__中寻找

⑤ 若以上均无法找到,则会调用 getattr 方法,执行内部的命令(若未重载 getattr 方法,则直接报错:AttributeError)

以上几个流程,即完成了属性的寻找。
注意: 第④步,一个类一旦重载了 getattribute() 方法,如果找不到属性,则必须要手动加入第④步,否则无法进入到 第⑤步 (getattr)的。可以用super解决


class Test:
    def __getattr__(self, name):
        print(‘__getattr__‘)

    def __getattribute__(self, name):
        print(‘__getattribute__‘)
        super().__getattribute__(name)

    def __setattr__(self, name, value):
        print(‘__setattr__‘)

    def __delattr__(self, name):
        print(‘__delattr__‘)


>>> t=Test()
>>> t.x
__getattribute__
__getattr__

某个类,只要是内部定义了方法 get, set, delete 中的一个或多个,就可以称为描述符


class Desc(object):

    def __get__(self, instance, owner):
        print("__get__...")
        print("self : \t\t", self)
        print("instance : \t", instance)
        print("owner : \t", owner)
        print(‘=‘*40, "\n")

    def __set__(self, instance, value):
        print(‘__set__...‘)
        print("self : \t\t", self)
        print("instance : \t", instance)
        print("value : \t", value)
        print(‘=‘*40, "\n")


class TestDesc(object):
    x = Desc()

#以下为测试代码
t = TestDesc()
t.x

#以下为输出信息:

__get__...
self :          <__main__.Desc object at 0x0000000002B0B828>
instance :      <__main__.TestDesc object at 0x0000000002B0BA20>
owner :      <class ‘__main__.TestDesc‘>

注意: 查找顺序问题:当Python解释器发现实例对象的字典中,有与描述符同名的属性时,描述符优先,会覆盖掉实例属性。

属性查询优先级小总结

    ① getattribute(), 无条件调用

    ② 数据描述符:由 ① 触发调用 (若人为的重载了该 getattribute() 方法,可能会导致无法调用描述符)

    ③ 实例对象的字典(若与描述符对象同名,会被覆盖哦)

    ④ 类的字典

    ⑤ 非数据描述符

    ⑥ 父类的字典

    ⑦ getattr() 方法

基本魔术方法

切片和索引

class Student(object):
    def __init__(self, name, scores):
        self.name = name
        self.scores = scores

    # 索引的是学生的成绩
    def __getitem__(self, index):  # 1).索引值的获取
        print(index, type(index))
        return self.scores[index]


    def __setitem__(self, key, value): # 2). 索引值的重新赋值
        self.scores[key] = value


    def __delitem__(self, key):  # 3). 删除索引值
        del  self.scores[key]


    def __mul__(self, other):  # 4). 实现*的效果, 具体返回什么取决于代码的业务需求
        """对于学生的每一门成绩乘3"""
        return  [i*other for i in self.scores]


    def __add__(self, other):  # 5). 连接的时候必须是同一种数据;类型
        # 将两个学生的成绩拼接起来
        return  self.scores + other.scores

    def __contains__(self, item): # 6). 判断某个元素是否存在于这个对象中?
        return  item in self.scores


    def __iter__(self):   # 7). 迭代, 使得该对象可以实现for循环
        # 将列表转换为迭代的类型, 可以for循环, 一定要返回iter类型的数据;
        return  iter(self.scores)

    def __lt__(self, other):  # 8). 比较两个对象的大小;
        return  (sum(self.scores)/3) < (sum(other.scores)/3)

liming = Student(‘liming‘, [100, 89, 100])

# # 1).索引值的获取
# print(liming[0])
# print(liming[1])
# print(liming[2])
#
# # 2).索引值的重新赋值
# liming[0] = 90
# print(liming[0])
#
# # 3). 删除索引值
# del liming[0]
# # print(liming[:2])
#


# # 1).切片值的获取
# print(liming[:2])
# print(liming[-2:])
#
#
# # 2).切片值的重新赋值
# liming[:2] = [10, 10]
# print(liming.scores)
#
# # 3). 删除切片值
# del liming[:2]
# print(liming.scores)

#
# # 4). 判断是否可以重复?
# print(liming * 3)
#
# # 5). 连接?
xiaohong = Student(‘小红‘, [100, 90, 90])
# print(xiaohong + liming)
#
#
#
# # 6). 成员操作符? 判断是否在对象里面存在?
# print(100 in xiaohong)
# print(101 in xiaohong)
# print(101 not in xiaohong)
#
#


# # 7). 实现for循环?
# for item in liming:
#     print(item)


# 8). 比较对象的大小?
# print(liming >  xiaohong)
print(liming <  xiaohong)

比较操作符

算数运算符

反运算

增量赋值运算

一元操作符

类型转换

上下文管理(with 语句)

容器类型

原文:https://www.cnblogs.com/lyalong/p/14109061.html

评论(0
© 2014 bubuko.com 版权所有 - 联系我们:wmxa8@hotmail.com
打开技术之扣,分享程序人生!