百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 热门文章 > 正文

Python关于 继承和多态 的知识 python继承和多态总结

bigegpt 2024-12-29 01:32 5 浏览

面向对象的三大特性:

1. 封装:根据职责将属性和方法封装到一个抽象的类中

2. 继承:子承父类,能实现代码的重用, 相同的代码不需要重复的编写

3. 多态:不同的子类对象调用相同的父类方法, 产生不同的执行结果


目录:继承,多继承,多态


继承的简单介绍:

继承就是为了解决两个有大量重复性代码的类。 # 抽象出一个更抽象的类的公共代码。

# 其意义是重用代码,方便代码的管理与修改

# 类的继承包括属性和方法;继承可以理解为是子承父业


继承的格式和__bases__:

class 类名1(类名2):

pass

#类名1继承了类名2的属性和方法,类名2是类名1的父类

#查询父类的魔法方法:__bases__;若没有规定继承的父类,默认继承的父类是object



继承到底继承了什么?

1.继承父类的属性、方法、魔法方法

#dir()可以查看类、实例对象的内置属性和方法


2.继承父类的私有属性,但是不能在子类的方法里直接调用父类的私有属性


3.继承中,属性或方法的寻找顺序:

依托于继承顺序 ---- 首先寻找实例对象本身 ---- 实例类 --- 父类 ---- 。。。。 --- object

#查看继承顺序:

【类名.__mro__】,

或者【实例.__class__.__mro__】;__class__用来查看类名


  • 1.继承父类的属性、方法、魔法方法

#dir()可以查看类、实例对象的内置属性和方法

class Father():
    sex = '男'                   #属性
    def __init__(self):        #魔法方法,注意要在实例化后,实例对象才有hair属性;而类是没有这个属性的
        self.hair = '黑色'
  
    def run(self):             #方法
        print('跑步')
        
class Son(Father):      #Son类继承Father类的sex属性,__init__魔法方法,run方法
    pass

 #查看Father,Son,xiaoming的内置属性和方法
print(dir(Father))    
print(dir(Son))
xiaoming = Son()
print(dir(xiaoming))



  • 2.继承父类的私有属性,但是不能在子类的方法里直接调用父类的私有属性
class Father():
    __sex = '男'

    def get(self):
        print(self.__sex)       #在Father类的方法里调用Father类的私有属性__sex

class Son(Father):              #Son继承Father类
    def get_from(self):
        print(self.__sex)         #在Son类的方法里调用Father类的私有属性__sex

a = Father()
a.get()        
b = Son()
b.get_from()         #在子类方法中不能直接调用父类的私有属性

#在当前类中可以在方法里调用自己类的私有属性

Traceback (most recent call last):

File "C:/Users/Administrator/Desktop/practice/func[副本].py", line 14, in <module>

b.get_from()

File "C:/Users/Administrator/Desktop/practice/func[副本].py", line 9, in get_from

print(self.__sex)

AttributeError: 'Son' object has no attribute '_Son__sex'

#在子类方法中不能直接调用父类的私有属性


  • 3.继承中,属性或方法的寻找顺序:

依托于继承顺序 ---- 首先寻找实例对象本身 ---- 实例类 --- 父类 ---- 。。。。 --- object

class Father():
    age = 55

class Son(Father):
    age = 24

xiaoming = Son()
xiaoming.age = 12
print(xiaoming.age)

#查看xiaoming实例对象的继承顺序:
print(xiaoming.__class__.__mro__)   #或者print(Son.__mro__)

12

#在Father类,Son类,xiaoming实例对象都有age属性,但是最终age属性取决于xiaoming这个实例对象继承的顺序


(<class '__main__.Son'>, <class '__main__.Father'>, <class 'object'>)

#属性或方法的继承首先寻找实例对象本身,然后才是按照以上顺序继承,即Son-Father-object


多继承

1.多继承的格式


2.若子类和父类属性同名,依照继承顺序继承,其属性或方法取决于继承顺序


3.属性及方法重构(重写)---通过在子类中定义和父类相同的属性名或方法名来实现,这样就会优先继承子类的属性或方法,覆盖父类的属性或方法


4.重构沿用,定向继承指定类的父类的属性或方法,用super()方法

#super():默认继承当前类的父类(就近原则);

也可以指定继承特定类的父类-----super(指定类, self)


5.super(指定类, self).__init__() ----- 会继承父类初始化里的全部类,而且可以指定父类__init__里的属性调用

  • 1.多继承格式的例子
class People():
    pass

class Father(People):
    pass

class Mother(People):
    pass

class Son(Father,Mother):   
    pass
#Son继承Father,Mother,People类,若Father和Mother类有属性或方法同名,
优先继承Father类,因为Father类在前,谁在前就优先继承谁

print(Son.__mro__)    #查询Son类的继承顺序

(<class '__main__.Son'>, <class '__main__.Father'>, <class '__main__.Mother'>, <class '__main__.People'>, <class 'object'>)


  • 2.若子类和父类属性同名,依照继承顺序继承,其属性或方法取决于继承顺序
class People():
    typ = '人类'

class Father(People):
    typ = '男人'

class Mother(People):
    typ = '女人'

class Son(Father,Mother):   
    pass
#Son继承Father,Mother,People类,若Father和Mother类有属性或方法同名,
优先继承Father类,因为Father类在前,谁在前就优先继承谁
    
print(Son.__mro__)
print(Son.typ)

(<class '__main__.Son'>, <class '__main__.Father'>, <class '__main__.Mother'>, <class '__main__.People'>, <class 'object'>)

男人 ----继承Father类的typ属性之后,就不会再往后继承


  • 3.属性及方法重构(重写)

---通过在子类中定义和父类相同的属性名或方法名来实现,这样就会优先继承子类的属性或方法,覆盖父类的属性或方法

class People():
    def __init__(self,typ):             #在People类里,使用初始化魔法方法__init__,需要传必备参数typ
        self.typ = typ

class Father(People):
    typ = '男人'

class Mother(People):
    typ = '女人'

class Son(Father,Mother):
    def __init__(self):
        self.age = 12              #在Son类里,使用初始化魔法方法__init__,不需要传参数

son = Son()    
#因为son实例对象优先继承Son类,所以son继承了Son的初始化,不需要传入参数;重构People的初始化方法
若只有People类里有初始化方法,则son在实例化时,必须传入参数,否则会报错

print(son.age)

12


  • 4.重构沿用,定向继承指定类的父类的属性或方法,用super(指定类, self)方法
class People():
    def __init__(self,typ):
        self.typ = typ

class Father(People):
    typ = '男人'
    def run(self):
        print('遗传爸爸跑步技能')

class Mother(People):
    name = '继承妈妈的姓:李'
    def run(self):
        print('遗传妈妈跑步技能')

class Son(Father,Mother):
    def run(self):

        print(super(Son, self).name)
#继承Son类的父类里的name属性,因为Father类没有name,所以再往上一级查找name属性,即继承Mother类的name属性
        super(Son, self).run()
#继承Son类的父类(即Father类) 里的run方法
        super(Father,self).run()
#继承Father类的父类(即Mother类) 里的run方法

        print('孩子跑步')

print(Son.__mro__) #查看继承顺序,方便查看super()方法到底继承哪个类的父类的方法
son = Son(18)
son.run()

(<class '__main__.Son'>, <class '__main__.Father'>, <class '__main__.Mother'>, <class '__main__.People'>, <class 'object'>)

继承妈妈的姓:李

遗传爸爸跑步技能

遗传妈妈跑步技能

孩子跑步


  • 5.super(指定类, self).__init__() ----- 会继承父类初始化里的全部类,而且可以指定父类__init__里的属性调用
class People():
    def __init__(self,typ):
        self.typ = typ
        self.skin_color = 'yellow'
        self.hair_color = 'black'
        return self.typ, self.hair_color    #返回指定的属性,调用时需用变量接收再打印,或者print输出


class Father(People):
    def run(self):
        print('遗传爸爸跑步技能')

class Mother(People):
    name = '继承妈妈的姓:李'
    def run(self):
        print('遗传妈妈跑步技能')

class Son(Father,Mother):
    def __init__(self):
        self.jicheng = super(Son, self).__init__('孩子')
#在Son类的初始化方法里,创建新的属性名来继承(接收)它父类中的初始化方法,即People类里的,
在People类的__init__必须传一个参数
#调用指定属性:不需要在super()后指定属性名调用所需属性,只需在实例化调用时,
使用和父类中初始化的属性名相同的属性名就可以调用,即【子类实例对象.父类中的属性名】

        self.age = 10
        #Son类自己的初始化属性

        
#实例化调用
son = Son()
print(dir(son))       #继承了Father,Mother,People里的属性,方法,初始化魔法方法

print(son.jicheng)      
#用新的属性名jicheng,接收了People__init__里的return的值,若return多个属性,则以元组的形式输出

print(son.typ)              #调用和父类中初始化属性名一样的typ
print(son.skin_color)   #调用和父类中初始化属性名一样的skin_color
print(son.hair_color)   #调用和父类中初始化属性名一样的hair_color
print(son.age)

['__class__', '__delattr__', '__dict__', '__dir__', '__doc__', '__eq__', '__format__', '__ge__', '__getattribute__', '__gt__', '__hash__', '__init__', '__init_subclass__', '__le__', '__lt__', '__module__', '__ne__', '__new__', '__reduce__', '__reduce_ex__', '__repr__', '__setattr__', '__sizeof__', '__str__', '__subclasshook__', '__weakref__', 'age', 'hair_color', 'jicheng', 'name', 'run', 'skin_color', 'typ']

('孩子', 'black') ----- son.jicheng

孩子 ----son.typ

yellow --- son.skin_color

black ---- son.hair_color

10 ---- son.age


多态

定义:不同的子类对象调用相同的父类方法, 产生不同的执行结果

多态的两个前提:
1.继承;
2.重写父类方法:子类继承父类方法,当子类里有和父类同名的方法,则子类里的方法会覆盖父类同名的方法

多态的好处:可以增加代码的灵活度


调用Dog类(父类)的game方法

class Dog(object):               #父类
    def __init__(self, name):
        self.name = name

    def game(self):
        print('%s在玩耍...' % self.name)

class Sq(Dog):                  #子类,继承Dog类的name属性和game方法
    def game(self):             #重写父类的game方法,并覆盖Dog类的game方法
        print('%s飞到天上去玩耍...' % self.name)

class Person(object):              
    def __init__(self, name):
        self.name = name

    def game_with_dog(self, dog):
        print('%s和%s快乐的玩耍....' % (self.name, dog.name))
        dog.game()     
#调用game方法,取Dog类还是Sq类的game方法,取决于Person实例对象xm传入的参数

jm = Dog('金毛')
xm = Person('小明')
xm.game_with_dog(jm)    
#xm实例对象传入的参数属于Dog类,所以调用的是Dog类里的game方法和Dog类里的name属性

小明和金毛快乐的玩耍....

金毛在玩耍...


调用Sq类(子类)的game方法

class Dog(object):               #父类
    def __init__(self, name):
        self.name = name

    def game(self):
        print('%s在玩耍...' % self.name)

class Sq(Dog):                  #子类,继承Dog类的name属性和game方法
    def game(self):             #重写父类的game方法,并覆盖Dog类的game方法
        print('%s飞到天上去玩耍...' % self.name)

class Person(object):              
    def __init__(self, name):
        self.name = name

    def game_with_dog(self, dog):
        print('%s和%s快乐的玩耍....' % (self.name, dog.name))
        dog.game()      
#调用game方法,取Dog类还是Sq类的game方法,取决于Person实例对象xm传入的参数

jm = Sq('神犬')
xm = Person('小明')
xm.game_with_dog(jm)    
#xm实例对象传入的参数属于Sq类,所以调用的是Sq类里的game方法和Sq类里的name属性

小明和神犬快乐的玩耍....

神犬飞到天上去玩耍...

相关推荐

为3D手游打造, Visual Studio Unity扩展下载

IT之家(www.ithome.com):为3D手游打造,VisualStudioUnity扩展下载7月30日消息,微软正式发布升级版VisualStudioToolsforUnity扩...

由ArcMap属性字段自增引出字段计算器使用Python的技巧

1.前言前些日子有人问我ArcMap中要让某个字段的值实现自增有什么方法?我首先想到像SQLServer中对于数值型字段可以设置自增。所以我打开ArcCatalog查看发现只提供默认值,没办法只能看...

微软首次回答 HoloLens 相关问题,终于爆料了

fengo2015/04/2115:11注:本文作者张静是NVIDIAGPU架构师,微信公众号“黑客与画家”(HackerAndPainter),知乎专栏地址。欢迎各位童鞋与他交流探讨。...

C#指针的应用(c#指针类型)

C#在有限的范围内支持指针。C#的指针只不过是一个持有另一类型内存地址的变量。但是在C#中,指针只能被声明为持有值类型和数组的内存地址。与引用类型不同,指针类型不被默认的垃圾收集机制所跟踪。出于同...

C# 堆栈(Stack)(c# 堆栈中定位调用messagebox 的地方)

C#集合在C#中,堆栈(Stack)是一种后进先出(LIFO,LastInFirstOut)的数据结构。堆栈(Stack)适用于存储和按顺序处理数据,其中最新添加的元素会最先被移除。堆...

欢迎回来:Fortran意外重回流行编程语言20强榜单

TIOBE指数是用来确定一种编程语言受欢迎程度的指标之一。它并不表明哪种编程语言是最好的,也不表明哪种编程语言写的代码行数最多,而是利用在谷歌、维基百科、必应、亚马逊、YouTube等各种引擎和网站上...

C#+NET MAUI实现跨平台/终端(linux,win,ios等)解决方案

简介.NETMulti-platformAppUI(.NETMAUI)是一个跨平台的框架,用于使用C#和XAML创建移动和桌面应用程序。使用.NETMAUI,您可以用一套代码库开发可以在A...

C#代码安全红线:SQL注入防护终极方案,让你的系统固若金汤

在数字化时代,应用系统的安全性至关重要。而SQL注入攻击,长期盘踞在OWASP(OpenWebApplicationSecurityProject)漏洞榜单的前列,成为众多基于数据库的应用系统...

C# (一)状态机模式(状态机代码实现)

最近空闲,炒炒隔夜饭,以前这些模式在自己项目种应用过不少,但一直没有像别人那样写一个系列,最近年纪大了,很多东西都忘记了,特别AI的兴起,更少写代码了,反正没什么事情,自己在重写一遍吧。创建型模式(5...

C# 中 Predicate 详解(c#中的replace)

Predicate泛型委托:表示定义一组条件并确定指定对象是否符合这些条件的方法。此委托由Array和List类的几种方法使用,用于在集合中搜索元素。Predicate<T>...

C#中$的用法?(c#中&&什么意思)

文章来自AI问答。在C#中,$符号用于字符串插值(StringInterpolation)。字符串插值是C#6.0引入的一种特性,它允许你在字符串中直接嵌入表达式,而不需要使用string.For...

C#并行编程:Parallel类(c# 并行处理)

在Parallel类中提供了三个静态方法作为结构化并行的基本形式:Parallel.Invoke方法:并行执行一组委托。Parallel.For方法:执行与C#for循环等价的并行方法。Parall...

颠覆认知!用Span重构foreach循环竟让数据处理快如闪电

在C#编程的世界里,数据处理效率始终是开发者们关注的焦点。随着项目规模的扩大和数据量的激增,哪怕是细微的性能提升,都可能对整个应用的响应速度和用户体验产生深远影响。近年来,C#引入的Span<T...

Unity3D手游开发实践《腾讯桌球》客户端开发经验总结

本次分享总结,起源于腾讯桌球项目,但是不仅仅限于项目本身。虽然基于Unity3D,很多东西同样适用于Cocos。本文从以下10大点进行阐述:1.架构设计2.原生插件/平台交互3.版本与补丁4.用脚本,...

.NET 7 AOT 的使用以及 .NET 与 Go 互相调用

目录背景C#部分环境要求创建一个控制台项目体验AOT编译C#调用库函数减少体积C#导出函数C#调用C#生成的AOTGolang部分安装GCCGolang导出函数.NETC#...