一.构造方法
1.初始化 _ _ init _ _()
在实例化后自动调用,以完成实例初始化,我们可以把要先初始化的属性放到这个函数里面。
class Student():
# 初始化方法
def __init__(self, name):
self.name = name
#对象实例化
stu = Student('小明')
print(stu.name)
2.析构函数
_ _ del_ _:当使用del 删除对象时,会调用他本身的析构函数,另外当对象在某个作用域中调用完毕,在跳出其作用域的同时析构函数也会被调用一次,这样可以用来释放内存空间。
提示开发者,对象被销毁,方便调试进行一些必要的清理工作
_ _ del_ _被调用, 还可以通过del关键字删除变量
二.魔法方法
1.主要的魔法方法
魔法方法 | 含义 |
---|---|
基本的魔法方法 | |
__ new__(cls[, …]) | 1. __ new__ 是在一个对象实例化的时候所调用的第一个方法 2. 它的第一个参数是这个类,其他的参数是用来直接传递给 __ init__ 方法 3. __ new__ 决定是否要使用该 __ init__ 方法,因为 __ new__ 可以调用其他类的构造方法或者直接返回别的实例对象来作为本类的实例,如果 __ new__ 没有返回实例对象,则 __ init__ 不会被调用 4. __ new__ 主要是用于继承一个不可变的类型比如一个 tuple 或者 string |
__ init__(self[, …]) | 构造器,当一个实例被创建的时候调用的初始化方法 |
__ del__(self) | 析构器,当一个实例被销毁的时候调用的方法 |
__ call__(self[, args…]) | 允许一个类的实例像函数一样被调用:x(a, b) 调用 x.__ call__(a, b) |
__ len__(self) | 定义当被 len() 调用时的行为 |
__ repr__(self) | 定义当被 repr() 调用或者直接执行对象时的行为 |
__ str__(self) | 定义当被 str() 调用或者打印对象时的行为 |
__ bytes__(self) | 定义当被 bytes() 调用时的行为 |
__ hash__(self) | 定义当被 hash() 调用时的行为 |
__ bool__(self) | 定义当被 bool() 调用时的行为,应该返回 True 或 False |
__ format__(self, format_spec) | 定义当被 format() 调用时的行为 |
有关属性 | |
__ getattr__(self, name) | 定义当用户试图获取一个不存在的属性时的行为 |
__ getattribute__(self, name) | 定义当该类的属性被访问时的行为 |
__ setattr__(self, name, value) | 定义当一个属性被设置时的行为 |
__ delattr__(self, name) | 定义当一个属性被删除时的行为 |
__ dir__(self) | 定义当 dir() 被调用时的行为 |
__ get__(self, instance, owner) | 定义当描述符的值被取得时的行为 |
__ set__(self, instance, value) | 定义当描述符的值被改变时的行为 |
__ delete__(self, instance) | 定义当描述符的值被删除时的行为 |
比较操作符 | |
__ lt__(self, other) | 定义小于号的行为:x < y 调用 x.__ lt__(y) |
__ le__(self, other) | 定义小于等于号的行为:x <= y 调用 x.__ le__(y) |
__ eq__(self, other) | 定义等于号的行为:x == y 调用 x.__ eq__(y) |
__ ne__(self, other) | 定义不等号的行为:x != y 调用 x.__ ne__(y) |
__ gt__(self, other) | 定义大于号的行为:x > y 调用 x.__ gt__(y) |
__ ge__(self, other) | 定义大于等于号的行为:x >= y 调用 x.__ ge__(y) |
算数运算符 | |
__ add__(self, other) | 定义加法的行为:+ |
__ sub__(self, other) | 定义减法的行为:- |
__ mul__(self, other) | 定义乘法的行为:* |
__ truediv__(self, other) | 定义真除法的行为:/ |
__ floordiv__(self, other) | 定义整数除法的行为:// |
__ mod__(self, other) | 定义取模算法的行为:% |
__ divmod__(self, other) | 定义当被 divmod() 调用时的行为 |
__ pow__(self, other[, modulo]) | 定义当被 power() 调用或 ** 运算时的行为 |
__ lshift__(self, other) | 定义按位左移位的行为:<< |
__ rshift__(self, other) | 定义按位右移位的行为:>> |
__ and__(self, other) | 定义按位与操作的行为:& |
__ xor__(self, other) | 定义按位异或操作的行为:^ |
__ or__(self, other) | 定义按位或操作的行为:| |
反运算 | |
__ radd__(self, other) | (与上方相同,当左操作数不支持相应的操作时被调用) |
__ rsub__(self, other) | (与上方相同,当左操作数不支持相应的操作时被调用) |
__ rmul__(self, other) | (与上方相同,当左操作数不支持相应的操作时被调用) |
__ rtruediv__(self, other) | (与上方相同,当左操作数不支持相应的操作时被调用) |
__ rfloordiv__(self, other) | (与上方相同,当左操作数不支持相应的操作时被调用) |
__ rmod__(self, other) | (与上方相同,当左操作数不支持相应的操作时被调用) |
__ rdivmod__(self, other) | (与上方相同,当左操作数不支持相应的操作时被调用) |
__ rpow__(self, other) | (与上方相同,当左操作数不支持相应的操作时被调用) |
__ rlshift__(self, other) | (与上方相同,当左操作数不支持相应的操作时被调用) |
__ rrshift__(self, other) | (与上方相同,当左操作数不支持相应的操作时被调用) |
__ rand__(self, other) | (与上方相同,当左操作数不支持相应的操作时被调用) |
__ rxor__(self, other) | (与上方相同,当左操作数不支持相应的操作时被调用) |
__ ror__(self, other) | (与上方相同,当左操作数不支持相应的操作时被调用) |
增量赋值运算 | |
__ iadd__(self, other) | 定义赋值加法的行为:+= |
__ isub__(self, other) | 定义赋值减法的行为:-= |
__ imul__(self, other) | 定义赋值乘法的行为:*= |
__ itruediv__(self, other) | 定义赋值真除法的行为:/= |
__ ifloordiv__(self, other) | 定义赋值整数除法的行为://= |
__ imod__(self, other) | 定义赋值取模算法的行为:%= |
__ ipow__(self, other[, modulo]) | 定义赋值幂运算的行为:**= |
__ ilshift__(self, other) | 定义赋值按位左移位的行为:<<= |
__ irshift__(self, other) | 定义赋值按位右移位的行为:>>= |
__ iand__(self, other) | 定义赋值按位与操作的行为:&= |
__ ixor__(self, other) | 定义赋值按位异或操作的行为:^= |
__ ior__(self, other) | 定义赋值按位或操作的行为:|= |
一元操作符 | |
__ pos__(self) | 定义正号的行为:+x |
__ neg__(self) | 定义负号的行为:-x |
__ abs__(self) | 定义当被 abs() 调用时的行为 |
__ invert__(self) | 定义按位求反的行为:~x |
类型转换 | |
__ complex__(self) | 定义当被 complex() 调用时的行为(需要返回恰当的值) |
__ int__(self) | 定义当被 int() 调用时的行为(需要返回恰当的值) |
__ float__(self) | 定义当被 float() 调用时的行为(需要返回恰当的值) |
__ round__(self[, n]) | 定义当被 round() 调用时的行为(需要返回恰当的值) |
__ index__(self) | 1. 当对象是被应用在切片表达式中时,实现整形强制转换 2. 如果你定义了一个可能在切片时用到的定制的数值型,你应该定义 __ index__ 3. 如果 __ index__ 被定义,则 __ int__ 也需要被定义,且返回相同的值 |
上下文管理(with 语句) | |
__ enter__(self) | 1. 定义当使用 with 语句时的初始化行为 2. __ enter__ 的返回值被 with 语句的目标或者 as 后的名字绑定 |
__ exit__(self, exc_type, exc_value, traceback) | 1. 定义当一个代码块被执行或者终止后上下文管理器应该做什么 2. 一般被用来处理异常,清除工作或者做一些代码块执行完毕之后的日常工作 |
容器类型 | |
__ len__(self) | 定义当被 len() 调用时的行为(返回容器中元素的个数) |
__ getitem__(self, key) | 定义获取容器中指定元素的行为,相当于 self[key] |
__ setitem__(self, key, value) | 定义设置容器中指定元素的行为,相当于 self[key] = value |
__ delitem__(self, key) | 定义删除容器中指定元素的行为,相当于 del self[key] |
__ iter__(self) | 定义当迭代容器中的元素的行为 |
__ reversed__(self) | 定义当被 reversed() 调用时的行为 |
__ contains__(self, item) | 定义当使用成员测试运算符(in 或 not in)时的行为 |
2.举例:_ _ add_ _
class Rectangle():
def __init__(self, length, width):
self.length = length
self.width = width
def __del__(self):
del self.width
del self.length
def __add__(self, other):
add_length = self.length + other.length
add_width = self.width + other.width
return add_length, add_width
a = Rectangle(1, 2)
b = Rectangle(3, 4)
print(a + b)
3. str与repr方法扩展
_ _ str_ _与 _ _ repr _ _是自定义的字符串描述,当我们打印或者查看某个对象是,最终看到的结果是这两个方法的返回值。
- _ _ str_ _
区别在于,查看对象a时,前者看到对象,后者看到__str __的返回值 - str与repr方法比较
-
两者同时存在,只会触发__str __(通过Print输出时)
-
交互模式下,直接输入对象c,调用的是__repr__方法
-
总结: 打印对象的时候,直接调用如果有__str__方法,则直接调用__str__方法,如果没有则调用__repr__方法,如果这两个方法都没有,则正常打印类的内存地址信息
4.call方法
- 正常情况下,实例后不能像函数一样调用,要想实例像函数一样被调用,需要定义__call__方法
5.开发及面试重点(_ _ new _ _ 单例模式)
-
__new__方法在类创建实例的时候自动调用
-
实例是通过类里面的__new__方法来创建实例,再调用__init__方法初始化实例
-
cls代表类本身
-
实例化时,new先被触发,创建实例;再出发init,初始化实例
-
如果没有返回实例对象,实例对象未被创建,就不会调用init初始化实例对象
-
这是为了实现实例之间的数据互通,使用同一个实例对象,同时也是是解决多继承问题
上图可以看出,对象a和b的id地址不同 -
hasattr(寻找对象,属性名):查找属性
-
instance: 属性名称,用于修改实例化对象
-
super() 函数用于调用父类的方法
-
super().new(cls) : 调用父类的new方法,同时也是创建新的实例:
在第一次一旦进入创建出来之后,那么下次新的实例出来时,不会再次进入new。因为已经实例,存在instance
class Rectangle(object):
def __init__(self, length, width):
self.length = length
self.width = width
def __call__(self, *args, **kwargs):
print(self.length, self.width)
def __new__(cls, *args, **kwargs):
# 如果没有属性instance,则创建属性
if not hasattr(cls, 'instance'):
# 通过父类(object),创建一个实例对象
cls.instance = super().__new__(cls)
return cls.instance
a = Rectangle(1, 2)
b = Rectangle(3, 4)
print(id(a))
print(id(b))
print(a.length, a.width, a.instance)
print(b.length, b.width, a.instance)
- 实例对象id都一样,但是名字不同
- 对实例a重新初始化,使用相同的属性,实现数据互通
三.扩展
- __ class __: 查看类名
实例.class - __ dict __: 查看全部属性,返回属性值(键值对形式)
实例.dict - __ doc __: 查看对象文档,即类中三引号部分,
类名.doc - __ bases __: 查看父类
类名.base - __ mro __: 查看多继承情况下,子类调用父类搜索顺序
子类名.mro/(实例.class.mro)