这是我参与更文挑战的第 25 天,活动详情查看: 更文挑战
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. 重写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)
5. 运算符重载
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在类里面都会提供一些以双下划线开头的默认方法,帮助我们更好的创建实例对象,初始化实例对象属性,实例对象重载方法,查看类里面的属性,获取实例对象的属性值等等
在创建新类的时候,我们也可以根据需求对这些方法进行重写。
好啦,以上是本期内容,欢迎大佬们评论区指正,下次见~