失眠网,内容丰富有趣,生活中的好帮手!
失眠网 > python面向对象之封装 继承与多态

python面向对象之封装 继承与多态

时间:2024-03-30 15:08:00

相关推荐

python面向对象之封装 继承与多态

目录

面向对象程序设计

类和对象

属性查找

数据属性

实例属性

类的封装

封装数据

封装⽅法(隔离复杂度)

类的⽅法

类的继承

继承概述

单继承

多继承

⽅法重写

继承原理

深度优先&⼴度优先

类的内置⽅法

类的多态

多态性

面向对象程序设计

所谓编程范式,就是程序员使⽤特定的语法+数据结构和算法编写代码,最终的⽬的是告诉计算机来解析并且来执⾏这些代码。在编程范式的领域,主要分为⾯向过程的编程⽅式和⾯向对象的编程⽅式,具体如下:

⾯向过程,也就是说流程化的⼲⼀件事,严格的按照顺序来进⾏,很明显,这与⽬前的敏捷模式格格不⼊的。但是它也是具备优点的,它的优点是把复杂的问题流程化,进⼀步达到简单化的过程。所谓优点也是缺点,在这⾥体现的⾮常明显,正因为对进⾏了流程化,也就导致了⼀套流程只能解决⼀个问题,⽐如⽣产汽⻋的流⽔线等,它的可扩展性存在很⼤的缺陷,因此也级有了⾯向对象的编程⽅式。

⾯向对象的编程,在上帝的视⻆下,来审视万事万物,⼀切皆是对象。与⾯向过程的机械式的编程⽅式⽽⾔,⾯向对象的编程⽅式更加看重的是对象,⽽⾮过程,它的优点具体为:解决了⾯向过程可扩展性的问题,在软件的⻆度⽽⾔,⾯向对象的程序设计只是来解决程序扩展性的问题

类和对象

所谓类就是类别,类简单的理解就是⼀系列对象相似的特征与技能的结合体。⽐如我们定义⼀个⼈的类,那么⾥⾯的⽅法就是⼈的特征。在程序⾥⾯,需要特别强调的是先定义类,再使⽤类(对类进⾏实例化)。在Python中,定 义类的关键字是class,所有类的基类是object。如下定义⼀个⼈的类,具体如下:

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

定义类的时候得切记,类的⾸字⺟是⼤写的,我们要使⽤⼀个类,我们就需要对它进⾏实例化,具体如下:

class Person(object):def __init__(self,name,age):self.name=nameself.age=ageif __name__ == '__main__': person=Person(name='⽆涯',age=18)

在如上的案例中,person就是类Person()的实例化后的对象,我们使⽤对象就可以查看对象的内存空间的信息,如:

class Person(object): '''基于⼈的定义'''def __init__(self,name,age):self.name=nameself.age=ageif __name__ == '__main__': person=Person(name='xcj',age=18) print(person.__dict__)输出:{'name': '⽆涯', 'age': 18}

属性查找

在类中,关于属性主要分为数据属性实例属性,下⾯结合具体的案例来说明下这部分。具体案例代码如下:

class Person(object):'''基于⼈的定义'''country='中国'city='⻄安'def __init__(self,name,age):self.name=nameself.age=ageif __name__ == '__main__':person=Person(name='xcj',age=18)

数据属性

类的数据属性是针对所有对象共享的,也就是说针对⼀个类进⾏实例化后的对象,调⽤数据属性它的内存地址都是⼀样的,如下:

class Person(object): '''基于⼈的定义''' country='中国' city='⻄安'def __init__(self,name,age):self.name=nameself.age=ageif __name__ == '__main__': p1=Person(name='xcj',age=18) p2=Person(name='xyz',age=19) #内存地址一致print(id(p1.country)) print(id(p2.country))#内存地址不一致print(id(p1.name))print(id(p2.name))

执⾏后,输出的内存地址是⼀致的。类的数据属性是绑定对象的。在如上中,p1.country中,它寻找的顺序是⾸先在p1的命名空间⾥⾯去找country, 如果找不到就去类⾥⾯找,类⾥⾯如再不到,就到基类⾥⾯去找,最后找不到,就会抛出对应的错误信息。具体如下:

class Person(object):'''基于⼈的定义'''country='中国'city='⻄安'def __init__(self,name,age):self.name=nameself.age=ageif __name__ == '__main__':p1=Person(name='xcj',age=18)print(p1.address)

Traceback (most recent call last):File "/Applications/code/Yun/zeroTestDev/demo.py", line 17, in <module>print(p1.address)AttributeError: 'Person' object has no attribute 'address'

实例属性

下⾯来看实例属性,具体如下涉及到的案例代码为:

class Person(object):'''基于⼈的定义'''def __init__(self,name,age):self.name=nameself.age=agedef show(self):print('my name is {0},and age is {1}'.format(self.name,self.age))if __name__ == '__main__':p1=Person(name='xcj',age=18)p1.show()

在类⾥⾯定义的属性name,age我们成为实例属性,但是定义的时候必须遵守函数的形式参数的规则,有⼏个参数就需要传⼏个,如下⾯案例少传⼀个,具体案例源码为:

class Person(object):'''基于⼈的定义'''def __init__(self,name,age):self.name=nameself.age=agedef show(self):print('my name is {0},and age is {1}'.format(self.name,self.age))if __name__ == '__main__':p1=Person(name='xcj')p1.show()执⾏后就会显示如下的错误信息:Traceback (most recent call last):File "/Applications/code/Yun/zeroTestDev/demo.py", line 15, in <module>p1=Person(name='xcj')TypeError: __init__() missing 1 required positional argument: 'age'

类的封装

在Python中,封装的特性是通过数据属性与实例属性来进⾏体现的。封装的核⼼本质思想是针对类内部的属性进⾏封装(隐藏)。

封装数据

在类的内部把数据隐藏起来,对外提供⼀个接⼝来操作具体的数据,然后在基础上增加对数据操作的权限和判断,具体案例代码如下:

class Person(object):def __init__(self,name,age):self.name=nameself.age=agedef info(self):print('my name is {name},and my age is {age}'.format(name=self.name,age=self.age))def setInfo(self,name,age):if not isinstance(name,str):raise TypeError('name不是字符串的数据类型')if not isinstance(age,int):raise TypeError('age不是整型的数据类型')self.name=nameself.age=ageif __name__ == '__main__':person=Person('xcj','18')person.setInfo('name','10')person.info()

程序执⾏后,就会报如下的错误信息,具体错误信息如下:

raise TypeError('age不是整型的数据类型')TypeError: age不是整型的数据类型

封装⽅法(隔离复杂度)

在实际的⼯作中,业务操作的复杂度是⾮常高的,这样我们可以根据业务的形态针对⽅法进⾏封装,这样对外调⽤起来会更加的简单,具体如下:

class HomePage():def login(self):passdef order(self):passdef getOrder(self):self.login()self.order()

在如上的业务形态中,最终的⽬的是查看订单的信息,但是查看订单的前提是⾸先需要先登录,所以我们可以针对业务的具体诉求进⾏封装

类的⽅法

在Python的类⾥⾯,在类⾥⾯编写的特性函数称为⽅法,这些⽅法主要分为普通⽅法特性⽅法静态⽅法类⽅法,具体如下:

class Person(object):'''基于⼈的定义'''def __init__(self,name,age):self.name=nameself.age=agedef show(self):print('my name is {0},and age is {1}'.format(self.name,self.age))@propertydef info(self):print('我是特性⽅法')@staticmethoddef eat():print('⼈每天都需要吃饭')@classmethoddef driver(cls): print('我是类⽅法')if __name__ == '__main__':person=Person(name='xcj',age=18)person.show()person.infoPerson(name='xcj',age=18).eat()person.driver()

普通⽅法:属于对象也属于类

特性⽅法:属于对象,编写特性⽅法的使⽤需要特别注意,该⽅法不能有形式参数,具备只读属性

静态⽅法:属于类,使⽤类名直接调用

@staticmethod 静态⽅法只是名义上归属类管理,但是不能使⽤类变量和实例变量,是类的⼯具包放在函数前(该函数不传⼊self或者cls),所以不能访问类属性和实例属性

类⽅法:属于类,使⽤类名直接可以调⽤

类的继承

在Python⾥⾯,⼀个类是可以继承多个类的。我们可以把类的继承理解为类与类之间的关系,在Python3⾥⾯,类的继承使⽤的算法是MRO的算法。在继承中,继承的类叫⼦类或者派⽣类,被继承的类称为基类或者是⽗类。具体如下:

class Person(object):'''基于⼈的定义'''def __init__(self,name,age):self.name=nameself.age=agedef show(self):print('my name is {0},and age is {1}'.format(self.name,self.age))class Worker(Person):def __init__(self,name,age,salary):super().__init__(name,age)self.salary=salarydef info(self):print('姓名{0},年龄:{1},薪资:{2}'.format(self.name,self.age,self.salary))if __name__ == '__main__':worker=Worker(name='xcj',age=18,salary=1000)worker.info()worker.show()

⾸先需要明确的是,⼀个⼦类继承⽗类,是继承⽗类⾥⾯所有的实例属性和⽅法,以及类属性的,具体结合如下案例来查看源码:

class Person(object):'''基于⼈的定义'''city = '陕⻄省⻄安市'def __init__(self, name, age):self.name = nameself.age = agedef show(self):print('my name is {0},and age is {1}'.format(self.name, self.age))class Worker(Person):def __init__(self, name, age, salary):super().__init__(name, age)self.salary = salarydef info(self):print('姓名{0},年龄:{1},薪资:{2},城市:{3}'.format(self.name, self.age, self.salary, self.city))if __name__ == '__main__':worker = Worker(name='xcj', age=18, salary=1000)worker.info()worker.show()

继承概述

⾯向对象三⼤特性分别是封装,继承,多态。所谓继承其实就是类与类之间的关系,⽽继承的⽬的就是为了解决代码重⽤问题,所以继承在某些⽅⾯来说,它是⼀种创建新类的⽅式,在Python中,⼀个类可以继承⼀个类,也可以继承多个类。被继承的类被称为:基类,继承别⼈的类,被称为:派⽣类。如下案例,具体如下:

class Person(object):passclass Student(Person):pass

查看继承的⽅式是使⽤bases,也就是查看⼦类继承的所有的⽗类,具体如下:

class Person(object):passclass Student(Person):passif __name__ == '__main__':print(Student.__bases__)执⾏如上的代码后,输出的结果信息为:(<class '__main__.Person'>,)

继承多个类,具体如下:

class Person(object):passclass Animal(object):passclass Student(Person, Animal):passif __name__ == '__main__':print(Student.__bases__)执⾏代码后,输出的结果信息:(<class '__main__.Person'>, <class '__main__.Animal'>)

特别注意:⼦类可以继承⽗类所有的⽅法以及属性。

单继承

在Python中,针对类的继承,可以使⽤单个类的继承,也可以使⽤多个类的继承,下⾯我们先来看单个类的继承。

class Person(object):country='中国'def __init__(self,sex):self.sex=sexdef show(self):print('来⾃那个{0},性别是:{1}'.format(self.country,self.sex))class ManStudent(Person):def __init__(self,sex,score):Person.__init__(self,sex)self.score=scoreif __name__ == '__main__':student=ManStudent('男',90.9)student.show()

执⾏后,就可以看到⼦类继承了⽗类的所有属性。

多继承

多继承更多指的是在Python⾥⾯,⼀个类可以继承多个类,那么就会涉及到⼀个调⽤的优先级的顺序问题。这个顺序问题具体可以有两种⽅式,第⼀种是从左到右的原则,第⼆种是从下到上的原则。

#从左到右class Father(object):def __init__(self,sex):self.sex=sexdef show(self):print('我是爸爸')class Mother(object):def __init__(self,sex):self.sex=sexdef show(self):print('我是妈妈')class Son(Father,Mother):def __init__(self,sex,school):super().__init__(sex)self.school=schoolif __name__ == '__main__':son=Son(sex='男孩⼦',school='⻄安第⼀中学')son.show()

在⼀个⼦类⾥⾯,⼦类继承了多个⽗类,这个时候⼦类调⽤⽗类⾥⾯共同有的⽅法,那么调⽤的顺序是从左到右的顺序。

#从上到下顺序class Person(object):def show(self):print('我是⼈类')class Father(Person):def show(self):print('我是爸爸')class Son(Father):passif __name__ == '__main__':son=Son()son.show()

⽅法重写

⼦类继承⽗类后,可以调⽤⽗类的⽅法,但是由于⽗类的⽅法⽆法满⾜⼦类的要求后,⼦类可以重写⽗类的⽅法,叫⽅法的重写,如下案例:

class Father(object):def show(self):print('脾⽓⽕爆')class Son(Father):def show(self):print('脾⽓温柔')if __name__ == '__main__':son=Son()son.show()

继承原理

在Python的类继承中,其实存在⼀个⽅法解析MRO的列表,它其实就是所有基类的线性顺序列表,如下所示:

class Person(object):def show(self):print('我是⼈类')class Father(Person):def show(self):print('我是爸爸') class Son(Father):passif __name__ == '__main__':print(Son.mro())

执⾏后,就会输出类的执⾏顺序了,具体输出的信息如下:

[<class '__main__.Son'>, <class '__main__.Father'>, <class '__main__.Person'>, <class 'object'>]

所以在Python中,基于MRO的解析顺序规则,就会从左到右开始查找基类,如果找到第⼀个匹配的属性类,就会停⽌查找,如果没有,那就继续查找,直到查找到符合要求的为⽌。MRO其实就是通过⼀个C3线性化算法来实现的,它的核⼼思想为:

⼦类会优先于⽗类检查多个⽗类会根据它们在列表中的顺序被依次检查如果对下⼀个类存在两个合法的选择,只能选择第⼀个

深度优先&⼴度优先

Python由于历史的原因,把Python2的类叫经典类,Python3的类叫新式类,其实在Python2中,多个类的继承算法叫深度优先,⽽Python3叫⼴度优先。如下⾯代码:

class A:def show(self):print('A')class B(A):passclass C(A):def show(self):print('C')class D(B,C):passif __name__ == '__main__':obj=D()obj.show()

针对如上的案例代码,调⽤后,调⽤的顺序是不同的,具体如下:

Python3顺序:D--->B--->A--->C

Python2顺序:D--->B--->A

类的内置⽅法

在⼀个类⾥⾯,会存在很多的内置⽅法,今天主要讲解常⽤的内置⽅法,具体如下:

'''类的内置⽅法:__init__:类的构造⽅法__del__:析构⽅法__str__:返回对象的字符串__doc__:返回document的信息__call__:类实例化后的对象(),触发执⾏该内置⽅法执⾏'''class Person(object):'''把世界上所有的⼈归为⼀类'''def __init__(self,name,age):self.name=nameself.age=agedef __del__(self):print('执⾏介绍,资源得到释放')def __str__(self):return 'my name is {0},and my age is {1}'.format(self.name,self.age)def info(self):print('欢迎参加⾼级测试开发学习训练营')def __call__(self, *args, **kwargs):return self.info()if __name__ == '__main__': obj=Person(name='xcj',age=18) print('返回document:',obj.__doc__) print('返回字符串的对象信息:',obj) print('执⾏__call__的⽅法:',obj())

执行后的信息:

返回document: 把世界上所有的⼈归为⼀类返回字符串的对象信息: my name is xcj,and my age is 18 欢迎参加⾼级测试开发学习训练营执⾏__call__的⽅法: None执⾏介绍,资源得到释放

类的多态

多态性

多态的优势具体可以总结为如下⼏点,具体为:

增加了持续的灵活性增加了持续的额外扩展的功能

class Animal(object):def talk(self):print('动物都是会叫的')class Cat(Animal):def talk(self):print('猫也是动物,所以也是会叫的')def func(animal):return animal.talk()if __name__ == '__main__':cat=Cat()func(animal=cat)

所以多态性我们也是可以理解为对Cat⽽⾔只是产⽣了⼀个对象,使⽤者完全可以在不需要修改⾃⼰的情况下,就能够调⽤Cat对象的talk的⽅法。针对如上的代码进再次进⾏完善,具体为:

class Animal(object):def talk(self):print('动物都是会叫的')class Cat(Animal):def talk(self):print('猫也是动物,所以也是会叫的')class Person(Animal):def talk(self):print('⼈也是动物,所以⼈也是会叫的')def func(animal):return animal.talk()if __name__ == '__main__':person=Person()func(animal=person)

如果觉得《python面向对象之封装 继承与多态》对你有帮助,请点赞、收藏,并留下你的观点哦!

本内容不代表本网观点和政治立场,如有侵犯你的权益请联系我们处理。
网友评论
网友评论仅供其表达个人看法,并不表明网站立场。