python面向对象编程之组合
前面讲了面向类与对象的继承,知道了继承是一种什么“是”什么的关系。然而类与类之间还有另一种关系,这就是组合。
先来看两个例子:
先定义两个类,一个老师类,老师类有名字,年龄,出生的年,月和日,所教的课程等特征以及走路,教书的技能。
class Teacher:def __init__(self,name,age,year,mon,day):self.name=nameself.age=ageself.year=yearself.mon=monself.day=daydef walk(self):print("%s is walking slowly"%self.name)def teach(self):print("%s is teaching"%self.name)
再定义一个学生类,学生类有名字,年龄,出生的年,月和日,学习的组名等特征以及走路,学习的技能。
class Student:def __init__(self,name,age,year,mon,day):self.name=nameself.age=ageself.year=yearself.mon=monself.day=daydef walk(self):print("%s is walking slowly"%self.name)def study(self):print("%s is studying"%self.name)
根据类的继承这个特性,可以把代码缩减一下。
定义一个人类,然后再让老师类和学生类继承人类的特征和技能:
class People:def __init__(self,name,age,year,mon,day):self.name=nameself.age=ageself.year=yearself.mon=monself.day=daydef walk(self):print("%s is walking"%self.name) class Teacher(People):def __init__(self,name,age,year,mon,day,course):People.__init__(self,name,age,year,mon,day)self.course=coursedef teach(self):print("%s is teaching"%self.name) class Student(People):def __init__(self,name,age,year,mon,day,group):People.__init__(self,name,age,year,mon,day)self.group=groupdef study(self):print("%s is studying"%self.name)
再对老师和学生进行实例化,得到一个老师和一个学生。
t1=Teacher("alex",28,1989,9,2,"python") s1=Student("jack",22,1995,2,8,"group2")
现在想知道t1和s1的名字,年龄,出生的年,月,日都很容易,但是想一次性打印出t1或s1的生日就不那么容易了,这时就需要用字符串进行拼接了,有没有什么更好的办法呢??
那就是组合。
继承是一个子类是一个父类的关系,而组合则是一个类有另一个类的关系。可以说每个人都有生日,而不能说人是生日,这样就要使用组合的功能 。可以把出生的年月和日另外再定义一个日期的类,然后用老师或者是学生与这个日期的类组合起来,就可以很容易得出老师t1或者学生s1的生日了,再也不用字符串拼接那么麻烦了。
来看下面的代码:
class Date:def __init__(self,year,mon,day):self.year=yearself.mon=monself.day=daydef birth_info(self):print("The birth is %s-%s-%s"%(self.year,self.mon,self.day)) class People:def __init__(self,name,age,year,mon,day):self.name=nameself.age=ageself.birth=Date(year,mon,day)def walk(self):print("%s is walking"%self.name) class Teacher(People):def __init__(self,name,age,year,mon,day,course):People.__init__(self,name,age,year,mon,day)self.course=coursedef teach(self):print("%s is teaching"%self.name) class Student(People):def __init__(self,name,age,year,mon,day,group):People.__init__(self,name,age,year,mon,day)self.group=groupdef study(self):print("%s is studying"%self.name) t1=Teacher("alex",28,1989,9,2,"python") s1=Student("jack",22,1995,2,8,"group2")
这样一来,可以使用跟前面一样的方法来调用老师t1或学生s1的姓名,年龄等特征以及走路,教书或者学习的技能。
print(t1.name) t1.walk() t1.teach()
输出为:
alex alex is walking alex is teaching
那要怎么能够知道他们的生日呢:
print(t1.birth)
输出为:
<__main__.Date object at 0x0000000002969550>
这个birth是子类Teacher从父类People继承过来的,而父类People的birth又是与Date这个类组合在一起的,所以,这个birth是一个对象。而在Date类下面有一个birth_info的技能,这样就可以通过调用Date下面的birth_info这个函数属性来知道老师t1的生日了。
t1.birth.birth_info()
得到的结果为:
The birth is 1989-9-2
同样的,想知道实例学生s1的生日也用同样的方法:
s1.birth.birth_info()
得到的结果为:
The birth is 1995-2-8
组合就是一个类中使用到另一个类,从而把几个类拼到一起。组合的功能也是为了减少重复代码。