Python3 学习第五弹:类与面向对象

对于面向对象总是要提到,万物皆对象。好似博大精深的感觉。
接下来一起看看python的面向对象的例子

创建一个对象

	class Person:
		type = 'person'
		def __init__(self, name = 'Noname'):
			self.name = name
		def hello(self):
			print("hello, I'm " + self.name)
	>>> Mike = Person('Mike')
	>>> Mike.hello()
	hello, I'm Mike
	对于一个对象,均会有默认构造方法,当然我们可以通过重写__init__来改变构造方法。当一个对象创建时会自动调用构造方法。

类的属性

	对于上一例的类来说
        其类属性:type以及默认类属性
	实例属性:name
	很容易理解,实例属性只有当创建一个类对象时才会产生,而类属性是类固有的属性
	类属性访问可以通过dir(Person)或Person.__dict__查看
	而实例属性可以通过Mike.__dict__查看

	基本类属性:
	__doc__	函式的文档. 字符串, 如果没有 的话就为 None	可写
	__name__	函式名	可写
	__module__	定义函式的模块名, 或者如果没有 对应模块名, 就为 None	
	__bases__ 返回该类的所有父类构成的元组
	__class__ 实例所对应的类
	__dict__ 类的属性构成的字典

	>>> class Person:
		'''Michael Scofield's file'''
		name = 'Michael'
		def __init__(self, age):
			self.age = age
	>>> Person.__doc__
	"Michael Scofield's file"
	>>> Person.__name__
	'Person'
	>>> Person.__bases__
	(<class 'object'>,)
	>>> Person.__dict__
	mappingproxy({'__dict__': <attribute '__dict__' of 'Person' objects>, '__doc__': "Michael Scofield's file", '__weakref__': <attribute '__weakref__' of 'Person' objects>, 'name': 'Michael', '__module__': '__main__', '__init__': <function Person.__init__ at 0x00000000033DBBF8>})
	>>> a = Person(30)
	>>> a.__class__
	<class '__main__.Person'>

继承的应用

	子类可以通过继承来获得父类的属性,此外可以有自己的类属性
		>>> class Bird:
			def __init__(self):
				self.hungry = True
			def eat(self):
				if self.hungry:
					print('eat something')
					self.hungry = False
				else:
					print('Too fat')
		>>> class SongBird(Bird):
			def __init__(self, song):
				self.song = song
			def sing(self):
				print(self.song)
		>>> a = SongBird('hello')
		>>> a.sing()
		hello
	我们常常会遇到子类有父类的相似函数,但是又不太一样。就可以通过重写来建立自己的函数,在运行该函数的时候会先在子类中找该函数,若无再从父类中寻找,例如上例中的__init__方法重写
	但是又会出现一个问题,
		>>> a.eat()
		AttributeError: 'SongBird' object has no attribute 'hungry'
	由于没有调用到父类的init使得hungry的属性消失,这时候利用super可以消除这种影响。
		>>> class SongBird(Bird):
				def __init__(self, song):
					super(SongBird, self).__init__()
					self.song = song
		>>> a.eat()
		eat something
		>>> a.eat()
		Too fat
	这里的super不是一个函数,而是一个类,表示SongBird的父类并传入自身对象self
	当然在出现super之前有另一个解决办法,就是直接调用父类的函数
		>>> class SongBird(Bird):
				def __init__(self, song):
					Bird.__init__(self)
					self.song = song
	但是由于一旦改动其父类,需要每次改变该类中的父类如上例中的Bird,因此最好使用super

通过property函数对类进行封装

    >>> class rec:
        def __init__(self):
            self.width = 0
            self.height = 0
        def setSize(self, sz):
            self.width = self.height = sz
        def getSize(self):
            return self.width, self.height
        size = property(getSize, setSize)
    >>> r = rec()
    >>> r.size
    (0, 0)
    >>> r.size = 1
    >>> r.size
    (1, 1)
    >>> r.width = 2
    >>> r.size
    (2, 1)
    
    先看看这样的描述更符合一种封装的思想,不必去在意其中实现的方法,只看重结果。
    那么是如何实现的呢?
    property(fget, fset, fdel, doc)可以通过4种参数来调用相应的过程,其中所所涉及的方法就是前面__get__,__set__,__delete__三种方法,实现了其中任何一种方法的对象就叫做描述符(descriptor)。
    就像上面的例子getSize实现了__get__方法,也就是在每次调用size时,对size取值都会调用到__get__方法,运行getSize函数
    举个例子,如果r是rec的一个类对象,那么r.size会调用__get__方法(getter),r.size = value会调用__set__方法(setter),del r.size会调用__delete__方法(deleter)。

    另外property封装的办法就是通过@进行一种类似装饰器描述

    class C:
        def __init__(self):
            self._x = None
    
        @property #定义属性x,生成__get__方法
        def x(self):
            """I'm the 'x' property."""
            return self._x
    
        @x.setter #生成__set__方法
        def x(self, value):
            self._x = value
    
        @x.deleter #生成__delete__方法
        def x(self):
            del self._x

    需要说明的是,属性x名称可以任意改变,而且用法与上一例的size相同。但要保证属性x名称相同,而且定义函数名均用属性名表示

类方法(类似于操作符重载)

	像上面例子中的__init__就是构造方法,当创建一个新类对象时自动调用,此外,还有许多可供使用的类方法,几乎覆盖了python所有操作符。
	接下来我们来看一看常见的类方法。

	>>基本方法:

	1> __init__(self[, arg1, ...]) 构造函数,在函数对象创建后调用,无返回值

	2> __new__(self[, arg1, ...]) 构造函数,创建函数对象的函数,返回值即该对象
	一般来说__new__不会被用到,除非子类需要不可变类型时才会用到。

	3> __del__(self) 在对象删除时调用,一般不改变,因为对象删除的时间由python自身决定

	4> __str__(self) 返回要打印的字符串(对用户较为友好),内建有str()及print

	5> __repr__(self) 返回对编译器友好的字符串,对细节处理上略有不同

	repr 与 str 区别:
		The str() function is meant to return representations of values which are fairly human-readable, while repr() is meant to generate representations which can be read by the interpreter (or will force a SyntaxError if there is not equivalent syntax). For objects which don't have a particular representation for human consumption, str() will return the same value as repr(). Many values, such as numbers or structures like lists and dictionaries, have the same representation using either function. Strings and floating point numbers, in particular, have two distinct representations.

	6> __format__(self, format_spec) 调用format()运行该函数

	7> __call__(self, *args) 类可以当作函数来调用,每次调用这个类对象时会运行,相当于重载了括号

	8> __len__(self) 在调用len()时会运行会运行该函数

	>>比较方法:

	1> __lt__(self, obj) 重载 <

	2> __le__(self, obj) 重载 <=

	3> __eq__(self, obj) 重载 ==

	4> __ne__(self, obj) 重载 !=

	5> __gt__(self, obj) 重载 >

	6> __ge__(self, obj) 重载 >=

	其中若不重载>, >默认会与<的对象交换,反之相同

	>>> class Person:
		def __init__(self, age):
			self.age = age
		def __lt__(self, b):
			return self.age - b.age
		def __gt__(self, b):
			return self.age > b.age
	>>> a = Person(1)
	>>> b = Person(2)
	>>> a > b
	False
	>>> a < b
	-1
	>>> b > a
	True
	>>> b < a
	1

	>>属性操作方法
	(注意以下的attr属性都是用字符串表示)

	1> __getattr__(self, attr) 获取属性不成功时调用,使用getattr()时调用(很奇怪在用python3测试时,即使获取属性成功也调用了)

	2> __setattr__(self, attr, value) 设置属性或新建属性时调用,使用setattr()时调用

	>>> class Person:
		def __init__(self, age):
			self.age = age
		def __getattr__(self, attr):
			print('__getattr__ used')
		def __setattr__(self, attr, val):
			print('__setattr__ used')
	>>> a = Person(1)
	__setattr__ used
	>>> a.age
	__getattr__ used
	>>> a.age=1
	__setattr__ used
	
	3> __delattr__(self, attr) 删除属性,使用delattr()调用

	4> __getattribute__(self, attr) 获取属性时无论访问成功与否都调用

	5> __get__(self, attr) 描述器,获取属性
	6> __set__(self, attr, value) 描述器,设置属性
	7> __delete__(self, attr) 描述器 删除属性
		以上三个暂时不明白其用法

	>> 数值运算符号重载

	1> __add__(self, other) 重载+
	2> __sub__(self, other) 重载-
	3> __mul__(self, other) 重载*
	4> __truediv__(self, other) 重载/
	5> __floordiv__(self, other) 重载//
	6> __mod__(self, other) 重载%
	7> __divmod__(self, other) 重载divmod(a, b) #返回元组(a//b, a%b)
	8> __pow__(self, other[, modulo]) 重载pow(a, b) **
	9> __lshift__(self, other) 重载<<
	10> __rshift__(self, other) 重载>>
	11> __and__(self, other) 重载&
	12> __xor__(self, other) 重载^
	13> __or__(self, other) 重载|

	ADD1:
		若在上述重载计算操作符函数前加入r,即__radd__(self, other),则表示其参数位置对调。
		例如, 计算表达式 x - y, y 是一个定义了方法 __rsub__() 的类实例, 那么在 x.__sub__(y) 返回 NotImplemented 时才会调用 y.__rsub__(x).
	ADD2:
		若在上述重载计算操作符函数前加入i,即__iadd__(self, other),则该重载符为 += 。以此类推

	14> __neg__(self) 重载负号-
	15> __pos__(self) 重载正号+
	16> __abs__(self) 重载abs()
	17> __invert__(self) 重载^

	强制转换类型
	18> __complex__(self) 重载complex() #表示复数类型
	19> __int__(self) 重载int()
	20> __float__(self) 重载float()
	21> __round__(self) 重载round()

	>>序列操作符重载

	1> __len__(self) 序列中项的个数

	2> __getitem__(self, index) 获取序列时调用

	3> __setitem__(self, index, value) 设置序列项或者新建项时调用

	4> __delitem__(self, index) 删除单个序列元素时调用

	>>> class Class:
		def __init__(self):
			self.student = ['LWT', 'CYL', 'Bash']
		def __len__(self):
			print('len called')
			return len(self.student)
		def __getitem__(self, index):
			print('getitem called')
			return self.student[index]
		def __setitem__(self, index, value):
			print('setitem called')
			self.student[index] = value
		def __delitem__(self, index):
			print('delitem called')
			del self.student[index]
	>>> a609 = Class()
	>>> len(a609)
	len called
	3
	>>> a609[1]
	getitem called
	'CYL'
	>>> a609[2] = 'Bug'
	setitem called
	>>> a609
	<__main__.Class object at 0x0000000003C3FD30>
	>>> del a609[2]
	delitem called

	其实就相当于对列表进行重载其操作。

	以上,大概包括了大部分的重载类型。

  

posted @ 2014-10-06 18:59  Estimator  阅读(398)  评论(0编辑  收藏  举报