一、多态与多态性

1、什么是多态

多态指的是同一种类/事物的不同形态

  1. class Animal:
  2. def speak(self):
  3. pass
  4.  
  5. class People(Animal):
  6. def speak(self):
  7. print('say hello')
  8.  
  9. class Dog(Animal):
  10. def speak(self):
  11. print('汪汪汪')
  12.  
  13. class Pig(Animal):
  14. def speak(self):
  15. print('哼哼哼')

多态

2、抽象类

只是用来建立规范的,不能用来实例化的,更无需实现内部的方法

  1. import abc
  2.  
  3. class Animal(metaclass=abc.ABCMeta):
  4. @abc.abstractmethod
  5. def speak(self):
  6. pass
  7.  
  8. @abc.abstractmethod
  9. def run(self):
  10. pass
  11.  
  12. class People(Animal):
  13. def speak(self):
  14. print('say hello')
  15.  
  16. def run(self):
  17. pass
  18.  
  19. class Dog(Animal):
  20. def speak(self):
  21. print('汪汪汪')
  22.  
  23. def run(self):
  24. pass
  25.  
  26. class Pig(Animal):
  27. def speak(self):
  28. print('哼哼哼')
  29.  
  30. def run(self):
  31. pass
  32.  
  33. obj1=People()
  34. obj2=Dog()
  35. obj3=Pig()

抽象类

  • 鸭子类型

python不推崇强制的继承关系,而崇尚的是鸭子类型,叫声像鸭子,走路也像鸭子,那它就是鸭子。

只要约定俗成的建立统一的一套标准/规范就行,就像Linux中有一种观点就是一切皆文件!

例如,如果想编写现有对象的自定义版本,可以继承该对象

也可以创建一个外观和行为像,但与它无任何关系的全新对象,后者通常用于保存程序组件的松耦合度。

例1:利用标准库中定义的各种‘与文件类似’的对象,尽管这些对象的工作方式像文件,但他们没有继承内置文件对象的方法

  1. #二者都像鸭子,二者看起来都像文件,因而就可以当文件一样去用
  2. class TxtFile:
  3. def read(self):
  4. pass
  5.  
  6. def write(self):
  7. pass
  8.  
  9. class DiskFile:
  10. def read(self):
  11. pass
  12. def write(self):
  13. pass

例2:其实我们一直在享受着多态性带来的好处,比如Python的序列类型有多种形态:字符串,列表,元组,多态性体现如下

  1. #str,list,tuple都是序列类型
  2. s=str('hello')
  3. l=list([1,2,3])
  4. t=tuple((4,5,6))
  5.  
  6. #我们可以在不考虑三者类型的前提下使用s,l,t
  7. s.__len__()
  8. l.__len__()
  9. t.__len__()
  10.  
  11. len(s)
  12. len(l)
  13. len(t)

二、封装

1、什么是封装

装:往容器/名称空间里存入名字
封:代表将存放于名称空间中的名字给藏起来,这种隐藏对外不对内

2、怎么封装

封装的方法是在属性前加上__(双下划线)

(1)在类内定义的属性前加__开头(没有__结果)

(2)__开头的属性实现的隐藏仅仅只是一种语法意义上的变形,并不会真的限制类外部的访问,外部可以调用类内的接口来操作属性

(3)该变形操作只在类定义阶段检测语法时发生一次,类定义阶段之后新增的__开头的属性并不会变形

  1. 1 class Foo:
  2. 2 __x=111 # _Foo__x 在定义的时候已经将__x改为_Foo__x
  3. 3 __y=222 # _Foo__y 在定义的时候就将__y改成_Foo__y
  4. 4
  5. 5 def __init__(self,name,age):
  6. 6 self.__name=name # 函数再定义的时候添加属性,可以被封装
  7. 7 self.__age=age
  8. 8
  9. 9 def __func(self): # 在定义的时候已经将__func改为_Foo__func
  10. 10 print('func')
  11. 11
  12. 12 def get_info(self):
  13. 13 print(self.__name,self.__age,self.__x) # 内部可以访问封装的属性
  14. 14
  15. 15 print(self._Foo__name,self._Foo__age,self._Foo__x)
  16. 16
  17. 17 print(Foo.__x) # 报错 无法直接访问封装的属性
  18. 18
  19. 19 print(Foo.__dict__)
  20. 20
  21. 21 print(Foo._Foo__x) # 111 封装只是在属性前面加了“_类名”,可以访问封装的属性
  22. 22
  23. 23 Foo.__z=333
  24. 24
  25. 25 print(Foo.__z) # 333 后期添加的属性不会被封装,封装只发生在类定义的阶段

3、封装数据属性

(1)将数据属性隐藏起来,类外就无法直接操作属性

(2)需要类内开辟一个接口来外部的使用可以间接地操作属性,可以在接口内定义任意的控制逻辑,从而严格控制使用对属性的操作

  1. class People:
  2. def __init__(self,name,age):
  3. self.__name=name
  4. self.__age=age
  5.  
  6. def tell_info(self): #接口用来访问封装的数据属性
  7. print('<name:%s age:%s>' %(self.__name,self.__age))
  8.  
  9. def set_info(self,name,age): # 接口用来修改封装的数据属性的值
  10. if type(name) is not str:
  11. print('名字必须是str类型')
  12. return
  13. if type(age) is not int:
  14. print('年龄必须是int类型')
  15. return
  16. self.__name=name
  17. self.__age=age
  18.  
  19. obj=People('egon',18)
  20. # obj.tell_info()
  21.  
  22. # obj.set_info('EGON',19)
  23. # obj.set_info(123,19)
  24. obj.set_info('EGON','')
  25. obj.tell_info()

封装的数据属性

4、封装函数属性

  1. class ATM:
  2. def __card(self):
  3. print('插卡')
  4. def __auth(self):
  5. print('用户认证')
  6. def __input(self):
  7. print('输入取款金额')
  8. def __print_bill(self):
  9. print('打印账单')
  10. def __take_money(self):
  11. print('取款')
  12.  
  13. def withdraw(self):
  14. self.__card()
  15. self.__auth()
  16. self.__input()
  17. self.__print_bill()
  18. self.__take_money()
  19.  
  20. a=ATM()
  21. a.withdraw()

封装函数属性

三、property装饰器

property装饰器是用来将类内的函数属性伪装成数据属性

  1. class People:
  2. def __init__(self, weight, height):
  3. self.weight = weight
  4. self.height = height
  5.  
  6. @property # property装饰器
  7. def bmi(self):
  8. return self.weight / (self.height ** 2)
  9.  
  10. obj = People(70, 1.75)
  11.  
  12. print(obj.bmi) # bmi被property伪装成数据属性

property装饰器

  1. class People:
  2. def __init__(self,name):
  3. self.__name=name
  4.  
  5. @property
  6. def name(self):
  7. return '<名字:%s>' %self.__name
  8.  
  9. @name.setter
  10. def name(self,obj):
  11. if type(obj) is not str:
  12. print('name必须为str类型')
  13. return
  14. self.__name=obj
  15.  
  16. @name.deleter
  17. def name(self):
  18. # print('不让删')
  19. del self.__name
  20.  
  21. obj=People('egon')
  22.  
  23. print(obj.name) # egon
  24. obj.name='EGON'
  25. #obj.name=123
  26. print(obj.name) # EGON

property拓展

面向对象(三)--多态、封装、property装饰器的更多相关文章

  1. 面向对象之组合、封装、多态、property装饰器

    概要: 组合 封装 property装饰器 多态 Python推崇鸭子类型:解耦合,统一标准(不用继承) 1. 组合 继承:会传递给子类强制属性 组合:解耦合,减少占用内存.如:正常继承,如果一个班级 ...

  2. Python10/24--组合/封装/property装饰器/多态

    组合的应用: 1. 什么是组合 一个对象的属性是来自于另外一个类的对象,称之为组合 2. 为何用组合 组合也是用来解决类与类代码冗余的问题 3. 如何用组合 '''class Foo: aaa=111 ...

  3. 面向对象(三)——组合、多态、封装、property装饰器

    组合.多态.封装.property装饰器 一.组合 1.什么是组合 组合指的是某一个对象拥有一个属性,该属性的值是另外一个类的对象 class Foo(): pass class Bar(): pas ...

  4. 面向对象之 组合 封装 多态 property 装饰器

    1.组合 什么是组合? 一个对象的属性是来自另一个类的对象,称之为组合 为什么要用组合 组合也是用来解决类与类代码冗余的问题 3.如何用组合 # obj1.xxx=obj2''''''# class ...

  5. python面向对象:组合、封装、property装饰器、多态

    一.组合二.封装三.property装饰器四.多态 一.组合 ''' 1. 什么是组合 一个对象的属性是来自于另外一个类的对象,称之为组合 2. 为何用组合 组合也是用来解决类与类代码冗余的问题 3. ...

  6. PYTHON-组合 封装 多态 property装饰器

    # 组合'''软件重用的重要方式除了继承之外还有另外一种方式,即:组合组合指的是,在一个类中以另外一个类的对象作为数据属性,称为类的组合 1. 什么是组合 一个对象的属性是来自于另外一个类的对象,称之 ...

  7. 组合,多态与多态性,封装以及property装饰器介绍

    一:组合: 什么是组合:组合指的是某一个对象拥有一个属性,该属性的值是另外一个类的对象. 为何要用组合:通过为某一个对象添加属性(属性的值是另外一个类的对象)的方式,可以间接地将两个类关联/整合/组合 ...

  8. python面向编程:类的组合、封装、property装饰器、多态

    一.组合 二.封装 三.propert装饰器 四.多态 一.组合 ''' 1. 什么是组合 一个对象的属性是来自于另外一个类的对象,称之为组合 2. 为何用组合 组合也是用来解决类与类代码冗余的问题 ...

  9. 面向对象之封装 及@property装饰器使用

    目录 封装 1.封装的定义 2.封装的目的: 3.封装的三种方式 4.封装的优点 5.访问限制(封装) @property 装饰器 属性property底层实现 封装 1.封装的定义 将复杂的丑陋的, ...

随机推荐

  1. Leetcode148-Sort_List

    Sort_List 在LeetCode 里面,因为只有归并排序的时间复杂度为O(1),所以快速排序用不了,前面两个都没用直接看最后一个归并排序. 冒泡排序(超时了) public ListNode s ...

  2. 怎么写Java项目?

    我们通常说的Java项目也都是JavaWeb,J2ee项目;现在说的是JavaWeb. 最简单的办法 多看别人项目源码 在别人基础上修改,而现在要一点一点掰开看看. 1.立项: 要做什么东西,最后的要 ...

  3. 【可视化】Vue基础

    作者 | Jeskson 来源 | 达达前端小酒馆 Vue简介 Vue框架,框架的作者,尤雨溪,组件化,快速开发的特点. 生命周期 beforeCreate:组件刚刚被创建 created:组件创建完 ...

  4. [LeetCode] 142. Linked List Cycle II 单链表中的环之二

    Given a linked list, return the node where the cycle begins. If there is no cycle, return null. To r ...

  5. scss 入门基础

    在一个项目中,样式是必不可少的一部分,而对于一个完整的项目来说是有个基准色调的.在项目需求变化不大的情况下,可以直接在css中写这些颜色值之类的东西.但是如果遇到一个朝令夕改的领导或者甲方,那会变得相 ...

  6. 推荐分享AB测试服务商

    推荐分享AB测试服务商 1.吆喝科技 2.testin 3.云眼

  7. RabbitMQ心跳检测与掉线重连

    1.RabbitMQListener,自定义消息监听器 using RabbitMQ.Client; using RabbitMQ.Client.Events; using System; using ...

  8. golang web 方案

    概要 开发 web 框架 数据库 认证 日志 配置 静态文件服务 上传/下载 发布 docker 打包 部署中遇到的问题 时区问题 概要 轻量的基于 golang 的 web 开发实践. golang ...

  9. [Powershell]使用Msbuild构建基于.NET Framework的WebAPI项目

    查找最高版本的MsBuildTools. 清理缓存垃圾. 还原NuGet包. 构建解决方案. 按项目发布程序到本地. 按项目ZIP打包. <# .NOTES ================== ...

  10. talk about string,char

    [1].关于sprintf和snprintf的正确使用 考虑以下有缺陷的例子:void f(const char *p){ char buf[11]={0}; sprintf(buf,"%1 ...