组合与重用性,抽象类

组合与继承

1.继承的方式

通过继承建立了派生类与基类之间的关系,它是一种'是'的关系,比如白马是马,人是动物。

当类之间有很多相同的功能,提取这些共同的功能做成基类,用继承比较好,比如老师是人,学生是人

2.组合的方式

用组合的方式建立了类与组合的类之间的关系,它是一种‘有’的关系,比如教授有生日,教授教python和linux课程,教授有学生s1、s2、s3...

示例:继承与组合

class People:
    def __init__(self,name,age,sex):
        self.name=name
        self.age=age
        self.sex=sex

class Course:
    def __init__(self,name,period,price):
        self.name=name
        self.period=period
        self.price=price
    def tell_info(self):
        print('<%s %s %s>' %(self.name,self.period,self.price))

class Teacher(People):
    def __init__(self,name,age,sex,job_title):
        super().__init__(name,age,sex)
        self.job_title=job_title
        self.course = []
        self.students =[]

    def tell_info(self):
        for c in self.students:
            print('%s 教学生 %s'%(self.name,c.name))
        for s in self.course:
            print('%s 教课程 %s'%(self.name,s.name))


class Student(People):
    def __init__(self,name,age,sex):
        super().__init__(name,age,sex)
        self.course=[]


egon=Teacher('egon',18,'male','金牌讲师')
s1=Student('s1',18,'female')

python=Course('python','3mons',3000.0)
linux=Course('linux','4mons',5000.0)

#为老师egon和学生s1添加课程
egon.course.append(python)
egon.course.append(linux)
s1.course.append(python)

#为老师egon添加学生s1
egon.students.append(s1)

egon.tell_info()

#使用
for item in egon.course:
   item.tell_info()

# egon 教学生 s1
# egon 教课程 python
# egon 教课程 linux
# <python 3mons 3000.0>
# <linux 4mons 5000.0>

总结:

当类之间有显著不同,并且较小的类是较大的类所需要的组件时,用组合比较好

抽象类

什么是抽象类

与java一样,python也有抽象类的概念但是同样需要借助模块实现,抽象类是一个特殊的类,它的特殊之处在于只能被继承,不能被实例化

为什么要有抽象类

如果说类是从一堆对象中抽取相同的内容而来的,那么抽象类就是从一堆类中抽取相同的内容而来的,内容包括数据属性和函数属性。

  比如我们有香蕉的类,有苹果的类,有桃子的类,从这些类抽取相同的内容就是水果这个抽象的类,你吃水果时,要么是吃一个具体的香蕉,要么是吃一个具体的桃子。。。。。。你永远无法吃到一个叫做水果的东西。

从设计角度去看,如果类是从现实对象抽象而来的,那么抽象类就是基于类抽象而来的。

  从实现角度来看,抽象类与普通类的不同之处在于:抽象类中只能有抽象方法(没有实现功能),该类不能被实例化,只能被继承,且子类必须实现抽象方法。这一点与接口有点类似,但其实是不同的,即将揭晓答案

在python中实现抽象类

import abc
class Animal(metaclass=abc.ABCMeta):#只能被继承,不能被实例化

    @abc.abstractmethod
    def run(self):
        pass

    @abc.abstractmethod
    def eat(self):
        pass




class Pig(Animal): #子类继承抽象类,但是必须定义run和eat方法
    
    def run(self):
        print('pig is walking')

    def eat(self):
        print('pig is eating')


class Dog(Animal):#子类继承抽象类,但是必须定义run和eat方法
    def run(self):
        print('dog is walking')

    def eat(self):
        print('dog is eating')

dog = Dog()
dog.run()
dog.eat()

pig = Pig()
pig.run()
pig.eat()

# dog is walking
# dog is eating
# pig is walking
# pig is eating

抽象类与接口

抽象类的本质还是类,指的是一组类的相似性,包括数据属性(如all_type)和函数属性(如eat、run),而接口只强调函数属性的相似性。

抽象类是一个介于类和接口直接的一个概念,同时具备类和接口的部分特性,可以用来实现归一化设计

原文地址:https://www.cnblogs.com/xiao-apple36/p/9127821.html