Python——魔术方法
特殊属性
__name__ 类、函数、方法等的名字
__module__ 类定义所在的模块
__class__ 对象或类所属的类
__bases__ 类的基类的元组,顺序为他们在基类列表中出现的顺序
__doc__ 类、函数的文档字符串,如果没有定义则为None
__mro__ 类的面容,class.mro()返回的结果 保存在__mro__中
__dict__ 类或实例的属性,可写的字典
查看属性
__dir__ 返回类或者对象的所有成员名称列表。dir() 函数就是调用__dir__()。
使用实例调用时,如果提供__dir__(),则返回其返回值,要求是可迭代对象
如果没有提供__dir__(),则会从实例和类及祖先类中手机信息
如果dir([obj]) 参数obj包含方法 __dir__(),该方法将被调用。如果Obj 不包含 __dir__(),该方法将最大限度收集属性信息
dir(obj) 对于不同类型的对象obj具有不同的行为:
1.如果对象是模块对象,返回的列表包含模块的属性名和变量名
2.如果对象是类型或者类对象,返回的列表包含类的属性名,及它的基类的属性名
3.如果obj不写 即dir(),返回列表包含内容不同
- 在模块中,返回模块的属性和变量名
- 在函数中,返回本地作用域的变量名
- 在方法中,返回本地作用域的变量名
dir()测试如下:
class Person:
def show(self):
a = 100
t = int(a)
print(dir())
def test(a=50, b=100):
print(dir())
Person().show()
test()
-----------------------------------------------
['a', 'self', 't']
['a', 'b']
魔术方法***
分类:
1.创建、初始化与销毁
__new__ 、__init__ 与__del__
2.hash
3.bool
4.可视化
5.运算符重载
6.容器和大小
7.可调用对象
8.上下文管理
9.反射
10.描述器
11.其它杂项
实例化
__new__方法很少使用,即使创了该方法,也会使用 return super().__new__(cls)基类obj 的 __new__方法来创建实例并返回
class A:
def __new__(cls, *args, **kwargs):
print(cls)
print(*args)
print(**kwargs)
return super().__new__(cls) # 返回cls的实例
# return 1
# return None
def __init__(self, name):
self.name = name
a = A("tom")
print(a)
--------------------------------------------
<class '__main__.A'>
tom
<__main__.A object at 0x00000281924C46A0>
hash
hash(x) x都一样,求得hash应该不变的,幂等性:一般来说,x不一样,hash应该不一样.
不同的hash算法都有hash冲突的问题,即不同的x求得同样的hash值
hash值相同就会去重吗??
不是的
list 类为为什么不可hash ??
源码中有一句 __hash__ = None, 也就是调用 __hash__() 相当于 None(),一定报错
所有类都继承object,而在这个类是具有 __hash__() 方法的,如果一个类不能被hash,就把__hash__设置为 = None
class A:
def __init__(self, name, age=18):
self.name = name
def __hash__(self):
return 1
# def __eq__(self, other):
# return self.name == other.name
def __repr__(self):
return self.name
print(hash(A("tom")))
print((A('tom'), A('tom')))
print([A('tom'), A('tom')])
print('-------------------------')
s = {A('tom'), A('tom')} # set
print(s)
print({tuple('t'), tuple('t')})
print({('tom',), ('tom',)})
print({"tom", "tom"})
---------------------------------------------------------------
1
(tom, tom)
[tom, tom]
-------------------------
{tom, tom} # set 没有去重
{('t',)}
{('tom',)}
{'tom'}
__hash__方法只是返回一个hash值作为set 的key,但是去重, 还需要__eq__ 来判断两个对象是否相等。
hash值相等,只是hash冲突,不能说明两个对象是相等的。
因此,一般来说提供__hash__方法是为了作set或者dict的key,所以 去重,要同时提供__eq__方法
不可hash对象 isinstance(p1, collection.Hashable) 一定为False(import collections)
去重需要提供 __eq__方法
class A:
def __init__(self, name, age=18):
self.name = name
def __hash__(self):
return 1
def __eq__(self, other): # 提供了__eq__方法
return self.name == other.name
def __repr__(self):
return self.name
print(hash(A("tom")))
print((A('tom'), A('tom')))
print([A('tom'), A('tom')])
print('-------------------------')
s = {A('tom'), A('tom')}
print(s)
print({tuple('t'), tuple('t')})
print({('tom',), ('tom',)})
print({"tom", "tom"})
------------------------------------------------
1
(tom, tom)
[tom, tom]
-------------------------
{tom} #去重了
{('t',)}
{('tom',)}
{'tom'}
bool
可视化
注意:类型判断要使用type或isinstance, 不能通过判断print输出是否带引号来判断输出值的类型。
class A:
def __init__(self, name, age=18):
self.name = name
self.age = age
def __repr__(self):
return "repr:{},{}".format(self.name, self.age)
def __str__(self):
return "str:{},{}".format(self.name, self.age)
def __bytes__(self):
return "{} is {}".format(self.name, self.age).encode()
print(A('tom'))
print([A('tom')])
print([str(A('tom'))])
print(bytes(A('tom')))
print('str:a,1')
s = '1'
print(s)
s1 = 'a'
print(s1)
print([s1], (s,))
print({s, "a"})
-----------------------------------------------------------------
str:tom,18
[repr:tom,18]
['str:tom,18']
b'tom is 18'
str:a,1
1
a
['a'] ('1',)
{'1', 'a'}
运算符重载
operator模块提供以下的特殊方法,可以将类的实例使用下面的操作符来操作
实现A类的2 个实例相减
class A:
def __init__(self, name, age=18):
self.name = name
self.age = age
def __sub__(self, other):
return self.age - other.age
def __isub__(self, other):
# self.age -= other.age
# return self
# return self.__clas__(self.name, self - other)
return A(self.name, self - other)
tom = A('tom')
jerry = A('jerry',16)
print(tom - jerry)
print(jerry - tom, jerry.__sub__(tom))
print(id(tom))
tom -= jerry
print(tom.age, id(tom))
-------------------------------------------------------
2
-2 -2
1864834369800
2 1864834369800
__isub__方法定义,一般会in-place来修改自身
如果没有定义 __isub__方法,则会调用__sub__
运算符重载应用场景
往往是用面向对象实现的类,需要做大量的运算,而运算符时这种运算在数学上最常见的表达方式,例如 对 - 进行了运算符重载,实现了类 的二元操作,重新定义为 tom - jerry
@functools.total_ordering装饰器
__lt__, __le__,__eq__,__gt__,__ge__是比较大小必须实现的方法,但是全部写完太麻烦了,,使用@functools.total_ordering装饰器就可以大大简化代码
但是要求__eq__必须实现,其他方法__lt__,__le__,__gt__,__ge__ 实现其中一个
__eq__ 等于 可以推断 不等于
__gt__ 大于 可以推断 小于
__ge__ 大于等于 可以推断 小于 等于
也就是用3 个方法,就可以吧所有比较解决了,所以total_ordering可以不使用
class Point:
def __init__(self, x, y):
self.x = x
self.y = y
def __hash__(self):
return hash((self.x, self.y))
def __eq__(self, other):
return self is other or (self.x == other.x and self.y == other.y)
def __add__(self, other):
return Point(self.x + other.x, self.y + other.y)
def __iadd__(self, other):
self.x, self.y = self.x + other.x, self.y + other.y
return self
def __eq__(self, other):#判断 == , !=
return self.x == other.x
def __gt__(self, other):# 判断> , <
return self.x > other.x
def __ge__(self, other):# 判断>= , <=
return self.x >= other.x
a = Point(3,2)
b = Point(1,3)
c = Point(3,1)
print(a + b)
print(a > b)
print(a > c)
print(a < b)
print(a < c)
print(a >= c)
print(a <= c)
-----------------------------------------------------------
<__main__.Point object at 0x000001F51ACA49E8>
True
False
False
False
True
True
容器相关方法
为什么空字典,空字符串,空元祖,空列表,空集合可以等效为False
因为 __bool__未定义时,回调用 __len__ 结果为0 等下 False
class A(dict):
def __missing__(self, key):
print('Missingn key:', key)
return 0
a = A()
print(a['k'])
------------------------------
Missingn key: k
0
可调用对象
Python 中一切皆对象,函数也不例外
函数即对象,对象foo加上(), 就是调用此函数对象的__call__()方法
def foo():
print(foo.__module__, foo.__name__)
foo()
# 等价于
foo.__call__()
可调用对象: 定义一个类,并实例化的到其实例,将实例像函数一样调用
class Point:
def __init__(self, x, y):
self.x = x
self.y = y
def __call__(self, *args, **kwargs):
return "<Point {}:{}>".format(self.x, self.y)
p = Point(4, 5)
print(p)
print(p())
--------------------------------------------------------
<__main__.Point object at 0x000001F7E8E30A20>
<Point 4:5>
class Adder:
def __call__(self, *args):
ret = 0
for x in args:
ret += x
self.ret = ret
return ret
adder = Adder()
print(adder(4, 5, 6))
print(adder.ret) # 当adder(4,5,6)不存在时,__call__方法不会调用,Adder无adder.ert属性,会抛异常
-------------------------------------------------------
<Point 4:5>
15
15
声明:本站所有文章资源内容,如无特殊说明或标注,均为采集网络资源。如若本站内容侵犯了原著者的合法权益,可联系本站删除。