1.概述
在 Python类中有些方法名、属性名的前后都添加类双下划,这种方法、属性通常都属于 Python的特殊方法和特殊属性,开发者可以通过重写这些方法或者直接调用这些方法来实现特殊的功能。
在我们前面已经有学过特殊方法的init()构造方法,开发者可以通过重写init方法来实现自己的初始化逻辑。
方法 | 说明 | 例子 |
---|---|---|
__init__ |
构造方法 | 对象创建 |
__del__ |
析构方法 | 销毁对象 |
__repr__ , __ str __
|
打印,转换 | print(a) |
__call__ |
函数调用 | a() |
__getattr__ |
点号运算 | a.XXX |
__setattr__ |
属性赋值 | a.xxx = value |
__getitem__ |
索引运算 | a[key] |
__setitem__ |
索引赋值 | a[key] = value |
__len__ |
长度 | len(a) |
每个运算符实际上都对应的方法。
运算符 | 特殊方法 | 说明 |
---|---|---|
运算符+ | __add__ |
加法 |
运算符- | __sub__ |
减法 |
<,<=,== |
__lt__ ,__le__ ,__eq__
|
比较运算符 |
>,>=,!= |
__gt__ , __ge__ , __ ne __
|
比较运算符 |
I,^,& |
__or__ , __ xor __ ,__ and __
|
或,异或,与 |
<<,>> |
__lshift__ , __rshift__
|
左移、右移 |
*,/,%,// |
__mul__ , __truediv__ ,__mod__ ,__floordiv__
|
乘,浮点除,横运算(取余),整数除 |
** | __pow__ |
指数运算 |
2. 重写repr
Python 默认情况下,__repr__()
会返回和实例对象有关的 “类名+object at+内存地址”信息
__reper__(self)
与__init__(self)
的性质一样,Python 中的每个类都包含__repr__()
方法,因为 object 类包含__reper__()
方法,而 Python 中所有的类都直接或间接继承自 object 类。执行 print(实例对象名) 等同于执行 print(实例对象名.
__repr__()
),程序的输出结果是一样的(输出的内存地址可能不同)如果我们要详细了解该实例化对象,可以通过重写类的
__repr__()
方法
class Animal:
def __init__(self,name,food): # Animal的属性特征
self.name = name
self.food = food
def __repr__(self): # 重写repr类
return "Animal[=name"+self.name+"likefood="+self.food+"]"
Cat = Animal("Tome","fish")
print(Cat)
3. 重写call
类实例方法__call__()
,该方法的功能类似于在类中重载 () 运算符,使得类实例对象可以像调用普通函数那样,以“对象名()”的形式使用
对于可调用对象,实际上“类实例对象名()”可以理解为是类实例对象名.__call__()
”的简写
class Animal:
def __call__(self,name,food):
print("调用__call__方法 {0} like to eat {1}".format(name,food))
Cat = Animal()
Cat.__call__("Tome","fish")
Cat("Tom","fish")
4. 重写dict
在Python内部创建一个类,类里面包含的类属性和类实例属性在堆内存内部都是以字典的形式进行存储的,其属性名作为键,值作为键对应的值。
-
类名.__dict__()
会输出包含类中所有的类属性的信息的字典 -
类实例对象.__dict__()
会输出包含实例对象的属性信息的字典 - 类实例对象可以通过字典的形式修改属性值
实例对象.__dict__[属性名]="新的属性值"
class Animal:
zone = "Aisa" # 类属性
time = "2021" # 类属性
def __init__(self,name,food): # 实例属性特征
self.name = name
self.food = food
#通过类名调用__dict__查看Animal所有的类属性
print("Animal类属性为:",Animal.__dict__)
Cat = Animal("Tome","fish")
#通过实例对象调用__dict__,查看实例对象的属性
print("实例对象Cat 属性为:",Cat.__dict__)
Cat.__dict__["name"]= "JueJing"
print("实例对象Cat通过属性名name来修改属性值:",Cat.name)
5. 重写hasattr/getattr/setattr
Python 中有提供判断/获取/设置类实例对象特定名称的属性和方法。
- hasattr() 来判断类实例对象是否包含指定的属性名称或者方法,返回的值是布尔类型True或者False
- getattr() 来获取实例对象中指定的属性的值
- setattr() 如果类实例对象指定的属性/方法不存在则会动态添加属性/方法,如果存在则会把指定属性覆盖以前的旧值
class Animal:
def __init__(self,name,food): # 实例属性特征
self.name = name
self.food = food
def get_age(self): #普通方法,可以调用私有属性和方法
print("{0} 的年龄是{1}岁".format(self.__name,self.age))
Cat = Animal("Tome","fish")
#使用hasattr函数
print(hasattr(Cat,"name"))
print(hasattr(Cat,"food"))
#使用getattr
print(getattr(Cat,"name"))
print(getattr(Cat,"food"))
print(getattr(Cat,"get_age"))
#使用setattr
setattr(Cat,"name","BOb")
print(Cat.name)
6. 运算符重载
Python 一切皆是对象,每钟序列都是python的一个类,例如list(),dict()等
Python内部使用一种重载运算符的技术来实现与运算符对应的处理的方式,当实例对象去进行运算符操作时,系统就可以调用类中相应的方法来处理
class Person:
def __init__(self, name):
self.name = name
def __add__(self, other): #对运输符+号进行重载
if isinstance(other, Person):
return "{0}---{1}".format(self.name,other.name)
else:
return "输入错误"
P1 = Person("Tom")
P2 = Person("Bob")
Sum = P1+P2 # 类对象可以进行运算+运算
print(Sum)
总结
Python在类里面都会提供一些以双下划线开头的默认方法,帮助我们更好的创建实例对象,初始化实例对象属性,实例对象重载方法,查看类里面的属性,获取实例对象的属性值等等
在创建新类的时候,我们也可以根据需求对这些方法进行重写。
好啦,以上是本期内容,欢迎大佬们评论区指正,下次见~