在此对常用的运算符重载做个小结。包括:__init__ , __str__(__repr__) , __call__ , __add__

这部分知识属于面向对象部分。

###关于运算符重载 以下是主要概念:

  • 类可以截获python运算符,可覆盖多数内置类型运算:如 加法,切片, 打印, 点号运算
  • 目的是让对象的用法和外观看起来更像内置类型
  • 运算符重载让对象与python内置对象模型更紧密结合(接口),提供了一致性,以及与预期接口的兼容性
  • 以双下划线命名(X),是特殊钩子,用来拦截运算,定义了固定不变的映射关系
  • 运算符覆盖方法没有默认值,如果没有定义,又执行相应运算,会触发异常

运算符重载的使用者主要是开发工具的人,而不是应用开发这者,不应该盲目使用,清晰优于隐晦

举个例子:

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
class 	Myclass:
	def __init__(self, value):
		self.data= value
	def __add__(self.other):
		return Myclass(self.date, other)
	def __mul__(self, other):
		self.data = self.data * other

	def display(self):
		print self.data

a = Myclass('abc')	
a.display() 	#'abc'

b = a + 'xyz'
b.display() 	#'abcxyz',注意__add__返回了新对象


a * 3
a.display() 	#'abcabcabc' ,注意__mul__原地修实例

#补充说明:__add__ , __mul__都是`二元运算`。以__add__(self.other)为例,当进行b = a + 'xyz'时,a 赋值给self,'xyz'赋值给other,一般而言实例放在左边,如果实例在右边需要使用__radd__

###更多细节

####init 每次从类产生实例时, python会在自动调用__init__方法(被称为构造器),这是最常用的运算符重载方法.
需要注意的地方是__init__常常用于调用超类的构造器,除非明确调用,否则超类构造器会被覆盖

1
2
3
4
5
6
7
8
class Super:
	def __init__(self,x):
		...default code...

class Sub(Super):
	def __init__(self,x,y):
		Super.__init__(self,x)
		...custom code

####repr__与__str__会返回对象的字符串表现形式 当类的实例打印或转换为字符串时,repr(及其近亲__str)就会自动调用,这些方法可替对象定义更好的显示格式。

1
2
3
4
5
6
7
8
9
class Myclass:
	def __init__(self,value=0):
		self.data = value

	def __repr__(self):
		return 'data is %s' self.data

x =  Myclass()
print x		#data is 0

那么__repr__与__str__的区别是什么,简单地说__repr__更底层,如果没有__str__则调用__repr__(也就是__str__优先),那么可以利用__repr__更底层的特性为所有环境提供统一的字符串显示。为终端用户提供__str__,开发期间则显示__repr__.

####call__拦截调用 当实例被调用时,使用__call,这样可以让实例表现得像函数

1
2
3
4
5
6
7
8
9
class Myclass:
	def __init__(self, value):
		self.value = value
	def __call__(self, other)
		return self.value * other

x = Myclass(2)
x(3)  #6
x(4)  #8

这可能是python中保留状态信息的最好方式,状态的记忆是明确地使用属性赋值运算实现的。 功能很像函数式编程里的闭包

###其他

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
#  __del__: 析构方法,与__init__相对
#  __getattr__: 点号运算(获取属性),如:X.name
#  __setattr__: 属性赋值运算, 如: X.name = 'wwj'
#  __getitem__: 索引运算 , X[key],没有__item__时的for循环迭代器。否则__item__优先
#  __setitem__: 索引赋值运算 , 如X[key] = value
#  __iter__: 迭代环境  用于循环,测试,列表,映射
#  __cmp__: 比较 , 如:X == Y , X > Y
#  __lt__: 小于 。 如: X < Y 
#  __eq__: 等于
#  __radd__: 左侧加法,如'abc' + X (对象在右侧)
#  __iadd__: 实地(增强的)加法 ,如:X += Y