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

您所在的位置:网站首页 python中继承和多态 python面向对象之封装、继承与多态

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

2023-03-23 01:35| 来源: 网络整理| 查看: 265

目录

面向对象程序设计

类和对象

属性查找

 数据属性

实例属性 

类的封装

封装数据

封装⽅法(隔离复杂度)

类的⽅法

类的继承 

继承概述

单继承

多继承

⽅法重写

继承原理

深度优先&⼴度优先

类的内置⽅法

类的多态

多态性

面向对象程序设计

所谓编程范式,就是程序员使⽤特定的语法+数据结构和算法编写代码,最终的⽬的是告诉计算机来解析并且来执⾏这些代码。在编程范式的领域,主要分为⾯向过程的编程⽅式和⾯向对象的编程⽅式,具体如下:         ⾯向过程,也就是说流程化的⼲⼀件事,严格的按照顺序来进⾏,很明显,这与⽬前的敏捷模式格格不⼊的。但是它也是具备优点的,它的优点是把复杂的问题流程化,进⼀步达到简单化的过程。所谓优点也是缺点,在这⾥体现的⾮常明显,正因为对进⾏了流程化,也就导致了⼀套流程只能解决⼀个问题,⽐如⽣产汽⻋的流⽔线等,它的可扩展性存在很⼤的缺陷,因此也级有了⾯向对象的编程⽅式。         ⾯向对象的编程,在上帝的视⻆下,来审视万事万物,⼀切皆是对象。与⾯向过程的机械式的编程⽅式⽽⾔,⾯向对象的编程⽅式更加看重的是对象,⽽⾮过程,它的优点具体为:解决了⾯向过程可扩展性的问题,在软件的⻆度⽽⾔,⾯向对象的程序设计只是来解决程序扩展性的问题

类和对象

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

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

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

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

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

class Person(object): '''基于⼈的定义''' def __init__(self,name,age): self.name=name self.age=age if __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=name self.age=age if __name__ == '__main__': person=Person(name='xcj',age=18)  数据属性

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

class Person(object): '''基于⼈的定义''' country='中国' city='⻄安' def __init__(self,name,age): self.name=name self.age=age if __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=name self.age=age if __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 print(p1.address) AttributeError: 'Person' object has no attribute 'address' 实例属性 

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

class Person(object): '''基于⼈的定义''' def __init__(self,name,age): self.name=name self.age=age def 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=name self.age=age def 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 p1=Person(name='xcj') TypeError: __init__() missing 1 required positional argument: 'age' 类的封装

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

封装数据

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

class Person(object): def __init__(self,name,age): self.name=name self.age=age def 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=name self.age=age if __name__ == '__main__': person=Person('xcj','18') person.setInfo('name','10') person.info()

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

raise TypeError('age不是整型的数据类型') TypeError: age不是整型的数据类型 封装⽅法(隔离复杂度)

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

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

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

类的⽅法

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

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

普通⽅法:属于对象也属于类特性⽅法:属于对象,编写特性⽅法的使⽤需要特别注意,该⽅法不能有形式参数,具备只读属性静态⽅法:属于类,使⽤类名直接调用

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

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

类的继承 

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

class Person(object): '''基于⼈的定义''' def __init__(self,name,age): self.name=name self.age=age def 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=salary def 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 = name self.age = age def 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 = salary def 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): pass class Student(Person): pass

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

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

继承多个类,具体如下:

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

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

单继承

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

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

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

多继承

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

#从左到右 class Father(object): def __init__(self,sex): self.sex=sex def show(self): print('我是爸爸') class Mother(object): def __init__(self,sex): self.sex=sex def show(self): print('我是妈妈') class Son(Father,Mother): def __init__(self,sex,school): super().__init__(sex) self.school=school if __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): pass if __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): pass if __name__ == '__main__': print(Son.mro())

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

[, , , ]

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

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

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

class A: def show(self): print('A') class B(A): pass class C(A): def show(self): print('C') class D(B,C): pass if __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=name self.age=age def __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)

 



【本文地址】


今日新闻


推荐新闻


CopyRight 2018-2019 办公设备维修网 版权所有 豫ICP备15022753号-3