继承

之前我们说到了类的公有属性和类的私有属性,其实就是类的封装,现在准备随笔的 是继承,是面向对象的第二大特性。

  面向对象编程 (OOP) 语言的一个主要功能就是“继承”。继承是指这样一种能力:它可以使用现有类的所有功能,并在无需重新编写原来的类的情况下对这些功能进行扩展。

  通过继承创建的新类称为“子类”或“派生类”,被继承的类称为“基类”、“父类”或“超类”,继承的过程,就是从一般到特殊的过程。在某些 OOP 语言中,一个子类可以继承多个基类。但是一般情况下,一个子类只能有一个基类,要实现多重继承,可以通过多级继承来实现。

  继承概念的实现方式主要有2类:实现继承、接口继承。

  1. 实现继承是指使用基类的属性和方法而无需额外编码的能力。
  2. 接口继承是指仅使用属性和方法的名称、但是子类必须提供实现的能力(子类重构爹类方法)。

  在考虑使用继承时,有一点需要注意,那就是两个类之间的关系应该是“属于”关系。例如,Employee 是一个人,Manager 也是一个人,因此这两个类都可以继承 Person 类。但是 Leg 类却不能继承 Person 类,因为腿并不是一个人。

  抽象类仅定义将由子类创建的一般属性和方法,OO开发范式大致为:划分对象→抽象类→将类组织成为层次化结构(继承和合成) →用类与实例进行设计和实现几个阶段

二、类的继承

1、继承的定义

在类名的括号中写入需要继承的类名即可

class Person(object):

    def talk(self):
print("person is talking...") class BlackPerson(Person): #继承Person这个类 def walk(self): #定义子类自身的walk方法
print("BlackPerson is walking.....") b = BlackPerson()
b.talk() #由于继承了父类的talk()方法,所以可以被调用
b.walk() #调用子类自身的walk方法 #输出
person is talking...
BlackPerson is walking.....

2、构造方法的继承

 说明:因为子类有自己的属性,但是又想继承父类的属性,所以需要先继承,再重构

继承类的构造方法2种写法:

  1. 经典类写法:父类.__init(self,name,age)
  2. 新式类写法:super(子类,self).__init__(name,age)

更多经典类和新式类知识补充:点击

注:建议使用新式类的写法,因为使用经典类的写法,在多继承的情况下,会出现重复调用参数的可能

class Person(object):
def __init__(self,name,age):
self.name = name
self.age = age
self.sex = "noraml" def talk(self):
print("person is talking...") class BlackPerson(Person):
def __init__(self,name,age,strength): #定义时需要传入父类的属性名
Person.__init__(self,name,age) #继承父类的构造方法,也可以写成:super(BlackPerson,self).__init__(name,age)
self.strength = strength #定义子类本身的属性
print(self.name,self.age,self.sex) def walk(self):
print("BlackPerson is walking.....") b = BlackPerson("xiaogao",18,"strong") #输出
xiaogao 18 noraml

这边不禁的有一个疑问?我不能重新写一遍,我为啥要继承父类中的方法呢?因为你重新写一遍的话,只能继承self.name和self.age,那self.sex怎么办,它也需要重写吗?所以啊,只有把父类中的构造函数全部继承过来,只能用上面这种办法?那它是怎么实现的呢?我们来画一个图:

3、子类对父类方法的重写

说明:如果我对父类中的方法不满意,我可以重写父类中的方法,当然还可以继承父类中的方法

class Person(object):
def __init__(self,name,age):
self.name = name
self.age = age
self.sex = "noraml" def talk(self):
print("person is talking...") class BlackPerson(Person): def talk(self): #重写父类的方法
Person.talk(self) #调用父类的方法
print("BlackPerson is talking ...") def walk(self):
print("BlackPerson is walking.....") b = BlackPerson("xiaogao",18) #子类不写,则继承父类的构造方法
b.talk()
b.walk() #输出
person is talking...
BlackPerson is talking ...
BlackPerson is walking.....

其实重写不是上面说的那么简单,只需要子类的方法名和父类的中的方法名一样就算重写了,其实不然,那怎么样才算重写呢?

重写的条件:

  1. 重写方法的方法名必须和父类中被重写的方法名一模一样
  2. 重写方法的传入的参数名和参数的个数必须和父类中被重写的方法一样

我们再来看看下面的例子:

class Person(object):

    def talk(self,food):
print("person is talking...{0}".format(food)) class BlackPerson(Person): def talk(self): #方法名和父类的方法名一样,但是少了一个food参数
print("BlackPerson is talking ...") b = BlackPerson("xiaogao",18,"strong")
b.talk() #输出
BlackPerson is talking ...

看着像重写了呀!其实不是重写,根据重写的条件明显两个方法的传入参数名和参数的个数都不一样,其实上面这种只是子类自己写了一个talk方法,只是名字一样,但是传入的参数和参数的个数不一样,并不是重写了父类中的方法。下面这个才是真正的重写:

class Person(object):

    def talk(self,food):
print("person is talking...{0}".format(food)) class BlackPerson(Person): def talk(self,food): #重写父类的方法(方法名和传入的参数名以及参数的个数与父类的方法一样)
print("BlackPerson is talking ...{0}".format(food)) b = BlackPerson("xiaogao",18,"strong")
b.talk("hotdog") #输出
BlackPerson is talking ...hotdog

类继承练习的小例子:

class SchoolMember(object):
'''学校成员基类''' member = 0 #设置类属性
def __init__(self,name,age,sex):
self.name = name
self.age = age
self.sex =sex
self.enroll() #每次生成一个对象注册一次
def enroll(self):
"注册"
print("just enroll a new school member [{0}]".format(self.name))
SchoolMember.member += 1 def tell(self):
print("------info:{0}-----".format(self.name))
for k,v in self.__dict__.items(): #__dict__()函数是获取对象的属性,以字典的形式返回
print("\t",k,v)
print("------end--------") def __del__(self):
print("开除了[{0}]...".format(self.name))
SchoolMember.member -= 1 class Teacher(SchoolMember):
"讲师类" def __init__(self,name,age,sex,salary,course):
SchoolMember.__init__(self,name,age,sex)
self.salary = salary
self.course = course def teaching(self):
"讲课方法"
print("Teacher [{0}] is teaching [{1}]".format(self.name,self.course)) class Student(SchoolMember):
"学生类"
def __init__(self,name,age,sex,couser,tuition):
SchoolMember.__init__(self,name,age,sex)
self.couser = couser
self.tuition = tuition
self.amount = 0 def pay_tuition(self,amount):
print("student [{0}] has just paied [{1}]".format(self.name,amount))
self.amount += amount t1 = Teacher("xiaogao",18,"F*M",3000,"Python")
s1 = Student("shuaigao",19,"M","PYS15",300000)
s2 = Student("gaogao",12,"M","PYS15",11000) print(SchoolMember.member)
del s1 #删除一个变量
t1.tell()
s2.tell()
print(SchoolMember.member) #会执行__del__函数 #输出
just enroll a new school member [xiaogao]
just enroll a new school member [shuaigao]
just enroll a new school member [gaogao]
3
开除了[shuaigao]...
------info:xiaogao-----
salary 3000
sex F*M
course Python
name xiaogao
age 18
------end--------
------info:gaogao-----
couser PYS15
sex M
name gaogao
amount 0
tuition 11000
age 12
------end--------
2
开除了[gaogao]...
开除了[xiaogao]...

多态

多态性(polymorphisn)是允许你将父对象设置成为和一个或更多的他的子对象相等的技术,赋值之后,父对象就可以根据当前赋值给它的子对象的特性以不同的方式运作。简单的说,就是一句话:允许将子类类型的指针赋值给父类类型的指针。
  那么,多态的作用是什么呢?我们知道,封装可以隐藏实现细节,使得代码模块化;继承可以扩展已存在的代码模块(类);它们的目的都是为了——代码重用。而多态则是为了实现另一个目的——接口重用!多态的作用,就是为了类在继承和派生的时候,保证使用“家谱”中任一类的实例的某一属性时的正确调用。
  很遗憾,在python不能直接的支持多态,所以我们只能间接的去执行
class Animal:

    def __init__(self,name):
self.name = name def talk(self):
print("animal {0} is talk".format(self.name)) class Cat(Animal):
def talk(self):
return "Meow!" class Dog(Animal):
def talk(self):
return "Woof!Woof!" def animal_talk(obj): #定义一个调用类的talk方法
print(obj.talk()) all_animal = [Dog("d1"),Cat("c1")] #把对象存入列表中 for animal in all_animal: #循环传入对象
animal_talk(animal) # 实现类似于Animal.talk(Dog("d1"))这种效果 #输出
Woof!Woof!
Meow!

小结:

  1. 多态在python中是不支持的
  2. 想要在python实现类似于Animal.talk(Dog("d1")),可以定义一个可以类方法的一个方法,调用时需要往这个方法传入类的实例

【python】-- 类的继承(新式类/经典类)、多态的更多相关文章

  1. c#中判断类是否继承于泛型基类

    在c#中,有时候我们会编写类似这样的代码: public class a<T> { //具体类的实现 } public class b : a<string>{} 如果b继承a ...

  2. 洗礼灵魂,修炼python(43)--巩固篇—经典类/新式类

    经典类 1.什么是经典类 就是在使用class关键词时,括号内不添加object类的就叫经典类,前面的博文里是绝对解析过的,所以你应该知道,经典类现在已经仅存在于python2了,因为python3不 ...

  3. Python 中的经典类新式类

    Python 中的经典类新式类 要知道经典类和新式类的区别,首先要掌握类的继承 类的继承的一个优点就是减少代码冗余 广度优先和深度优先,这主要是在多类继承的时候会使用到 经典类和新式类的主要区别就是类 ...

  4. day30 python类的继承,抽象类等

    Python之路,Day17 = Python基础17-面向对象入门 继承 class Student(People): pass print(Student.__bases__) # 查看 Stud ...

  5. python基础之继承派生、组合、接口和抽象类

    类的继承与派生 经典类和新式类 在python3中,所有类默认继承object,但凡是继承了object类的子类,以及该子类的子类,都称为新式类(在python3中所有的类都是新式类) 没有继承obj ...

  6. python之继承、抽象类、新式类和经典类

    一.上节补充1.静态属性静态属性 : 类的属性,所有的对象共享这个变量 如果用对象名去修改类的静态属性:在对象的空间中又创建了一个属性,而不能修改类中属性的值 操作静态属性应该用类名来操作 例1:请你 ...

  7. Python类总结-继承-子类和父类,新式类和经典类

    子类和父类 class Father(object): #子类在使用super调用父类时,Father后面要加object --新式类 def __init__(self): self.Fname = ...

  8. 【python】-- 类的多继承、经典类、新式类

    继承知识点补充 在python还支持多继承,但是一般我们很少用,有些语言干脆就不支持多继承,有多继承,就会带来两个概念,经典类和新式类. 一.多继承 之前我们都是讲的单继承,那么什么是多继承呢?说白了 ...

  9. Python新式类与经典类的区别

    1.新式类与经典类 在Python 2及以前的版本中,由任意内置类型派生出的类(只要一个内置类型位于类树的某个位置),都属于“新式类”,都会获得所有“新式类”的特性:反之,即不由任意内置类型派生出的类 ...

  10. python中新式类和经典类

    python中的类分为新式类和经典类,具体有什么区别呢?简单的说, 1.新式类都从object继承,经典类不需要. Python 2.x中默认都是经典类,只有显式继承了object才是新式类 Pyth ...

随机推荐

  1. ssh免密码登录的注意事项

    centos配置完免密码登录(注意修改配置文件,/etc/ssh/sshd_config),合并完公钥后,有的时候还得需要输入密码.这时候应该检查一下authorized_keys的权限问题.本机的正 ...

  2. [转载]linux 清除系统cached

    FROM: http://cqfish.blog.51cto.com/622299/197230 linux 清除系统cached top查看系统内存使用情况   Mem:    16432180k ...

  3. C#.NET为List加入扩展方法:获取唯一值

    public static class ListTools { /// <summary> /// 获取唯一值列表 /// </summary> /// <param n ...

  4. Cocos2d-x 3.1.1 学习日志13--物理引擎登峰造极之路

    cocos2dx在设计之初就集成了两套物理引擎,它们是box2d和chipmunk.我眼下使用的是最新版的cocos2dx 3.1.1.引擎中默认使用的是chipmunk.假设想要改使用box2d的话 ...

  5. HTML5 Canvas 六角光阑动态效果

    光阑是光具组件中光学元件的边缘.框架或特别设置的带孔屏障,本人实现了结构比较简单的六角光阑,效果有点像宇航员在徐徐张开的飞船舷窗中看到逐渐完整的地球,下面四张图可以感受一下. 当然看动态效果才能真正体 ...

  6. C++ 设置文件最近修改时间

    利用VS开发C++项目,经常发现修改系统时间后,每次编译过程会变得很慢,其原因就是当你把系统时间调到未来的一个时间点,然后有意或者无意编辑过一些代码文件,那么这些文件的时间戳就停留在未来. 当你把系统 ...

  7. vue-router $route

    1.$route 除了 $route.params 外,$route 对象还提供了其它有用的信息,例如,$route.query (如果 URL 中有查询参数).$route.hash 等等

  8. EFFECTIVE JAVA 类和接口

    第十六条:复合优先于继承 //这是一个不好的类---执行的结果 addCount = 4(addAll的实现依赖于HashSet的add方法,InstrumentHashSet方法重写了add方法有执 ...

  9. ios 使用gcd 显示倒计时

    __block ;//倒计时时间 dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, ...

  10. c#创建对象并动态添加属性

    //动态类,可以作为基类被继承 dynamic backObj = new ExpandoObject(); //ExpandoObject 为密封类 backObj.image0 = IMGNAME ...