Python面向对象之多态、封装
一、多态
超过一个子类继承父类,出现了多种的形态。
例如,动物种类出现了多种形态,比如猫、狗、猪
- class Animal:pass
- class Cat(Animal):pass
- class Dog(Animal):pass
- class Pig(Animal):pass
Python是处处有多态,因为新式类都是继承object
多态的定义:一个类表现出的多种形态,通过继承来实现,一个父类有多个子类。再简化的说法就是“有多种形式”,就算不知道变量(参数)所引用的对象类型,也一样能进行操作,来者不拒。
count(),repr() 这类就是多态的代表
在java中的表示:在一个函数中需要给参数指定数据类型,如果这个地方可以接收两个以上类型的变量,那么这些类型应该有一个父类,这个父类是所有子类对象的类型
在python中:函数的参数不需要指定数据类型,所以不需要通过继承的形式来统一一组数据类型,换句话说 所有的对象其实都是object类型,所有在python中处处是多态。
经典类和新式类都有多态
- from abc import ABCMeta,abstractmethod
- class Animal(metaclass=ABCMeta):
- @abstractmethod
- def talk(self):pass
- class Cat(Animal):
- def talk(self):
- print('喵喵')
- class Dog(Animal):
- def talk(self):
- print('汪汪')
- c = Cat()
- c.talk()
- d = Dog()
- d.talk()
- '''
- 喵喵
- 汪汪
- '''
代码中有 Cat 和 Dog 两个类,都继承了类 Animal,它们都有 talk() 方法,输入不同的动物名称,会得出相应的结果。
鸭子类型:
在程序设计中,鸭子类型(英语:duck typing)是动态类型的一种风格。在这种风格中,一个对象有效的语义,不是由继承自特定的类或实现特定的接口,而是由当前方法和属性的集合决定。这个概念的名字来源于由 James Whitcomb Riley 提出的鸭子测试,“鸭子测试”可以这样表述:“当看到一只鸟走起来像鸭子、游泳起来像鸭子、叫起来也像鸭子,那么这只鸟就可以被称为鸭子。”
定义:不是明确的通过继承实现的多态,而是通过一个模糊的概念来判断这个函数能不能接收这个类型的参数
python中通过鸭子类型来体现多态:比如len(),print()
二、封装
广义的封装:
定义一个类,把所有的属性和方法都放在类里面
为了只有这个类的对象才能使用定义在类中的方法,其他都用不了
狭义上上的封装:
把一个变量藏在类里,在类的外部不能引用私有变量,只有类的内部能使用
- class A:
- __x = 0 # 私有的静态变量
定义一个类,执行的时候把类的内容读完才开辟类的空间,所有类的内容加载的时候就完成了
- class A:
- __x = 0 # 私有的静态变量
- print('AAA')
- '''
- AAA
- '''
定义类的内存机制:
① 类中的静态变量和方法名在程序加载的过程中就已经执行完毕了,不需要等待调用
② 在类加载完成之前,类的名字还没出现在内存空间中
③ 私有的静态属性可以在类的内部使用,用来隐藏某个变量的值
当私有变量在存储时会做了变形:_类名__私有变量
类名.变形的私有变量可以取值,但是从编程规范的角度上出发,不能在类的外部使用私有的变量
- class Course:
- __name = 'python' # 变形:_Course__name
- print(Course.__dict__) # 查看类里所有的属性和变量
- print(Course._Course__name) # python,不建议使用这方法取值
- '''
- {'__module__': '__main__', '_Course__name': 'python', '__dict__': <attribute '__dict__' of 'Course' objects>, '__weakref__': <attribute '__weakref__' of 'Course' objects>, '__doc__': None}
- python
- '''
定义私有变量的原因:
① 不想让使用者看到这个值
② 不想让使用者修改这个值
③ 让使用者修改这个值的时候做一些限制:保证了数据的安全
④ 有些方法或者属性不希望被子类继承
不想让使用者修改这个值
- class University:
- def __init__(self,name,site):
- self.__name = name # 定义了私有属性
- self.__site = site # 定义了私有属性
- def name(self): # 定义方法返回值
- return self.__name
- def site(self): # 定义方法返回值
- return self.__site
- uni = University('哈佛大学','马萨诸塞州剑桥市')
- print(uni.name()) # 调用类提供的方法取值
- print(uni.site()) # 调用类提供的方法取值
- print(uni.name) # 类对象的方法的内存地址
- print(uni) # 类对象的内存地址
- uni.name = '剑桥大学' # 在类对象里添加了name的值
- print(uni.__dict__)
- '''
- 哈佛大学
- 马萨诸塞州剑桥市
- <bound method University.name of <__main__.University object at 0x00000209EBDE7E48>>
- <__main__.University object at 0x00000209EBDE7E48>
- {'_University__name': '哈佛大学', '_University__site': '马萨诸塞州剑桥市', 'name': '剑桥大学'}
- '''
让使用者修改这个值的时候做一些限制
- class Supermarket:
- __discount_coupon = 1000
- def __init__(self,name,price):
- self.name = name
- self.__price = price # 定义私有变量
- def price(self): # 获取变量值的方法
- if self.__price > 5999:
- return self.__price - Supermarket.__discount_coupon
- else:
- return self.__price
- def change_price(self,new_price): # 改变变量值的方法
- if isinstance(new_price,int):
- self.__price = new_price
- else:
- print('价格修改失败')
- ap = Supermarket('IPhone',6000)
- print(ap.price()) # 通过方法获得私有变量的值
- ap.change_price(3000) # 通过方法改变私有变量的值
- print(ap.price())
- '''
- 5000
- 3000
- '''
类中的私有成员:
① 私有的静态属性
② 私有的对象属性
③ 私有的方法
私有变量在类的外部不能被定义,因为在类的内部带__ 定义私有变量都会变形,在外部被定义不会变形
- class Language:
- __China = 'chinese' # 发生变形 '_Language__China': 'chinese'
- print(Language.__dict__)
- Language.__Korea = 'Korean' # 在类的外部定义不会发生变形 __Korea': 'Korean'
- print(Language.__dict__)
- '''
- {'__module__': '__main__', '_Language__China': 'chinese', '__dict__': <attribute '__dict__' of 'Language' objects>, '__weakref__': <attribute '__weakref__' of 'Language' objects>, '__doc__': None}
- {'__module__': '__main__', '_Language__China': 'chinese', '__dict__': <attribute '__dict__' of 'Language' objects>, '__weakref__': <attribute '__weakref__' of 'Language' objects>, '__doc__': None, '__Korea': 'Korean'}
- '''
私有变量不能被继承
- # 封装前
- class A:
- def __init__(self,name):
- self.name = name
- class B(A):
- def get_name(self):
- print(self.name)
- b = B('haha')
- b.get_name()
- '''
- haha
- '''
- # 封装后
- class A:
- def __init__(self,name):
- self.__name = name
- class B(A):
- def get_name(self):
- print(self.__name)
- b = B('haha')
- b.get_name()
- AttributeError: 'B' object has no attribute '_B__name'
私有成员的特点:
① 不能在类的外部被调用,只能在类的内部被调用(在类的内部变形了,_类名__变量名)
② 不能被子类继承
三、property 内置函数
装饰器的使用:所有的装饰器函数、方法、类的上一行直接@装饰器的名字
装饰器的分类:
① 装饰函数
② 装饰方法
③ 装饰类
property 是一个装饰器函数
@property 是装饰方法的,将一个方法伪装成一个属性,相当于调用方法是不需要加括号
- class A:
- def __init__(self,name):
- self.__name = name
- @property
- def name(self):
- return self.__name
- a = A('小明')
- print(a.name)
- '''
- 小明
- '''
Python面向对象之多态、封装的更多相关文章
- Python面向对象三要素-封装(Encapsulation)
Python面向对象三要素-封装(Encapsulation) 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.封装概述 将数据和操作组织到类中,即属性和方法 将数据隐藏起来,给 ...
- Python面向对象之多态
多态 面向对象三大特性 封装 根据职责将属性和方法封装到一个抽象的类中:--定义类的准则 继承 实现代码的重用,相同的代码不需要重复的编写:--设计类的技巧:子类针对自己的需求,编写特定的代码: 多态 ...
- python 面向对象之多态与绑定方法
多态与多态性 一,多态 1,多态指的是一类事物有多种形态(python里面原生多态) 1.1动物有多种形态:人,狗,猪 import abc class Animal(metaclass=abc.AB ...
- Python 面向对象5 多态
一.多态 多态性(polymorphisn)是允许你将父对象设置成为和一个或更多的他的子对象相等的技术,赋值之后,父对象就可以根据当前赋值给它的子对象的特性以不同的方式运作,简单的说就是一句话:允许将 ...
- python 面向对象三大特性(封装 多态 继承)
今天我们来学习一种新的编程方式:面向对象编程(Object Oriented Programming,OOP,面向对象程序设计)注:Java和C#来说只支持面向对象编程,而python比较灵活即支持面 ...
- Python 面向对象 组合-多态与多态性-封装-property
面向对象-组合 1.什么是组合 组合指的是某一个对象拥有一个属性,该属性的值是另外一个类的对象 class Foo: xxx = 111 class Bar: yyy = 222 obj = Foo( ...
- python面向对象中的封装、继承、多态
封装 可以简单的理解为隐藏一切可以隐藏的实现细节,只向外界提供简单的编程接口.我们在类中定义的方法其实就是把数据和数据的操作封装起来了,在我们创建了对象之后,只需要给对象发送一个消息(调用方法)就可以 ...
- Python面向对象三大特性(封装、继承、多态)
封装 类中把某些属性和方法隐藏起来,或者定义为私有,只在类的内部使用,在类的外部无法访问,或者留下少量的接口(函数)供外部访问:从上一篇文章中的私有属性与私有方法中的代码体现了该特性. class m ...
- Python之路【第十篇】:Python面向对象之多态、多态性
阅读目录 一 多态 多态指的是一类事物有多种形态 动物有多种形态:人,狗,猪 import abc class Animal(metaclass=abc.ABCMeta): #同一类事物:动物 @ab ...
随机推荐
- MongoDB学习笔记一:MongoDB基础
目录 MongoDB是什么? 学了有什么用? MongoDB入门 安装 修改数据库位置 MongoDB的启动方式 MongoDB的图形化工具 MongoDB基本命令 增 查询 更新 删除 排序 投影 ...
- 单机prometheus vs 集群kube-prometheus+prometheus-operator
prometheus 组件: node-exporter:9100端口 https://segmentfault.com/a/1190000017959127
- 【serviceaccount 和 clusterrolebinding】
kubectl get clusterrolebinding kubectl create clusterrolebinding suosheng-rebinding --clusterrole=cl ...
- VSCode 代码格式化快捷键
转载自:http://geek-docs.com/vscode/vscode-tutorials/vscode-code-formatting.html VSCode 代码格式化快捷键,我们平常在做自 ...
- eNSP上VLAN的基础的配置及access接口
本实验模拟公司内部,为不同的部门划分不同的VLAN ,形成的不同广播域,来保护信息的安全,拓扑图如下所示
- JAVA 读取xml格式的数据
<?xml version="1.0" encoding="UTF-8"?> <column-enums> <type name= ...
- feign学习笔记
使用:
- [Visual Studio] - Unable to launch the IIS Express Web server 问题之解决
背景 Visual Studio 2015 在 Debug 模式下调试失败. 错误 解决 删除解决方案下 .vs/config 文件夹,重新运行解决方案可进行调试. 参考资料 https://stac ...
- Tp5.1 管理后台开发纪要
1. tp5.1 对网页是有缓存机制的 E:\phpStudy\PHPTutorial\WWW\NewAdmin\thinkphp\library\think\Template.php 下displa ...
- 【转】MySQL中EXISTS的用法
原文链接:https://www.cnblogs.com/qlqwjy/p/8598091.html 比如在Northwind数据库中有一个查询为 SELECT c.CustomerId,Compan ...