PYTHON-组合 封装 多态 property装饰器
# 组合
'''
软件重用的重要方式除了继承之外还有另外一种方式,即:组合
组合指的是,在一个类中以另外一个类的对象作为数据属性,称为类的组合 1. 什么是组合
一个对象的属性是来自于另外一个类的对象,称之为组合 2. 为何用组合
组合也是用来解决类与类代码冗余的问题 3. 如何用组合
# obj1.xxx=obj2
'''
'''
>>> class Equip: #武器装备类
... def fire(self):
... print('release Fire skill')
...
>>> class Riven: #英雄Riven的类,一个英雄需要有装备,因而需要组合Equip类
... camp='Noxus'
... def __init__(self,nickname):
... self.nickname=nickname
... self.equip=Equip() #用Equip类产生一个装备,赋值给实例的equip属性
...
>>> r1=Riven('锐雯雯')
>>> r1.equip.fire() #可以使用组合的类产生的对象所持有的方法
release Fire skill # class Foo:
# aaa=1111
# def __init__(self,x,y):
# self.x=x
# self.y=y
# def func1(self):
# print('foo中的功能')
#
# class Bar:
# bbb=2222
# def __init__(self, m, n):
# self.m = m
# self.n = n
# def func2(self):
# print('bar中的功能')
#
# obj1=Foo(10,20)
# obj2=Bar(30,40)
#
# obj1.xxx=obj2 #组合
# print(obj1.x,obj1.y,obj1.aaa,obj1.func1)
# print(obj1.xxx.m,obj1.xxx.n,obj1.xxx.bbb,obj1.xxx.func2) # class OldboyPeople:
# school = 'Oldboy'
# def __init__(self, name, age, gender):
# self.name = name
# self.age = age
# self.gender = gender
#
# class OldboyStudent(OldboyPeople):
# def choose_course(self):
# print('%s is choosing course' %self.name)
#
# class OldboyTeacher(OldboyPeople):
# def __init__(self, name, age, gender,level,salary):
# super(OldboyTeacher, self).__init__(name, age, gender)
# # OldboyPeople.__init__(self, name, age, gender)
# self.level=level
# self.salary=salary
# def score(self,stu,num):
# stu.num=num
# print('老师%s给学生%s打分%s' %(self.name,stu.name,num))
#
# class Course:
# def __init__(self,course_name,course_price,course_period):
# self.course_name=course_name
# self.course_price=course_price
# self.course_period=course_period
# def tell_course(self):
# print('课程名:<%s> 价钱:[%s] 周期:[%s]' % (self.course_name, self.course_price, self.course_period))
#
# python_obj=Course('python开发',3000,'5mons')
# linux_obj=Course('linux运维',5000,'3mons')
#
#
# stu1=OldboyStudent('egon练习',18,'male')
# stu1.choose_course()
#
# stu1.courses=[]
# stu1.courses.append(python_obj)
# stu1.courses.append(linux_obj)
# stu1.courses[0].tell_course()
#
#
# tea2=OldboyTeacher('kevin',38,'male',5,3000)
# tea2.score(stu1,60)
''' # 封装
# 引子
# 从封装本身的意思去理解,封装就好像是拿来一个麻袋,
# 把小猫,小狗,小王八,还有alex一起装进麻袋,然后把麻袋封上口子。
# 照这种逻辑看,封装=‘隐藏’,这种理解是相当片面的 '''
1. 什么是封装
装指的是把属性装进一个容器
封指的是将容器内的属性隐藏起来
封装本质上就是影藏一部分内容 使外界无法直接访问 只能通过提供的接口来访问
对于开发者 作用对数据的访问加以限制 提高安全性
对于使用者而言 简化了操作 2. 为何要封装(*****)
封装不是单纯意义的隐藏
封装数据属性的目的:让外部使用者间接地操作数据,在接口上附加额外的逻辑,从而严格
控制使用者对属性的操作
封装函数(功能)属性的目的:隔离复杂度(很多功能是给内部用,给外部封装起来隔离复杂度) 3. 如何封装(*****)
只需要在属性前加上__(两个下划线),该属性就会被隐藏起来,该隐藏具备的特点:
1. 只是一种语法意义上的变形,即__开头的属性会在检测语法时发生变形 _类名__属性名
2. 这种隐藏式对外不对内的(因为在类内部检测语法时所有的代码统一都发生的变形)
3. 这种变形只在检测语法时发生一次,在类定义之后新增的__开头的属性并不会发生变形
4. 如果父类不想让子类覆盖自己的属性,可以在属性前加__开头 本质是 语法级别上变形(在语法检测期间) _类名__属性名
这么一来 我么完全可以绕过语法限制
'''
'''
# class Foo:
# __x=111 #_Foo__x
# def __init__(self,m,n):
# self.__m=m # self._Foo__m=m
# self.n=n
#
# def __func(self): #_Foo__func
# print('Foo.func')
#
# def func1(self):
# print(self.__m) #self._Foo__m
# print(self.__x) #self._Foo__x # print(Foo.__dict__)
# Foo.__x
# Foo.__func
# print(Foo._Foo__x)
# print(Foo._Foo__func) obj=Foo(10,20)
# print(obj.__dict__)
# print(obj.n)
# print(obj.__m)
# print(obj._Foo__m) # obj.func1()
# obj._Foo__func() # obj.__yyy=3333 #特点三,
# print(obj.__dict__)
# print(obj.__yyy) # Foo.__zzz=444
# Foo.zzz=333
# print(Foo.__dict__)
# print(Foo.__zzz) # class Foo:
# def __f1(self): #_Foo__f1
# print('Foo.f1')
#
# def f2(self):
# print('Foo.f2')
# self.__f1() #self._Foo__f1
#
# class Bar(Foo):
# def __f1(self): #_Bar__f1
# print('Bar.f1')
#
# obj=Bar()
# obj.f2() # 封装数据属性的真实意图
# class People:
# def __init__(self,name,age):
# self.__name=name
# self.__age=age
# def tell_info(self):
# print('<name:%s age:%s>' %(self.__name,self.__age))
# def set_info(self,new_name,new_age):
# if type(new_name) is not str:
# print('name error')
# return
# self.__name=new_name
# if type(new_age) is not int:
# print('age error')
# return
# self.__age=new_age
# def clear_info(self):
# del self.__name
# del self.__age
#
# obj1=People('aaa',10)
# obj1.tell_info()
# obj1.set_info('aaa',111)
# print(obj1.__dict__)
# obj1.clear_info()
# print(obj1.__dict__) # 封装函数属性的真实意图
# class ATM:
# def __card(self):
# print('插卡')
# def __auth(self):
# print('用户认证')
# def __input(self):
# print('输入取款金额')
# def __print_bill(self):
# print('打印账单')
# def __take_money(self):
# print('取款')
#
# def withdraw(self):
# self.__card()
# self.__auth()
# self.__input()
# self.__print_bill()
# self.__take_money()
#
# a=ATM()
# a.withdraw()
''' # 特性(property)
# 什么是特性property
# property是一种特殊的属性,访问它时会执行一段功能(函数)然后返回值 # 为什么要用property
# 将一个类的函数定义成特性以后,对象再去使用的时候obj.name,
# 根本无法察觉自己的name是执行了一个函数然后计算出来的,
# 这种特性的使用方式遵循了统一访问的原则 # property装饰器:将类中定义的函数(功能)伪装成属性
# @property #查看属性的功能
# @xxx.setter #修改属性的功能
# @xxx.deleter #删除属性的功能
# name=property(xxx_name,yyy_name,zzz_name) #古老的做法非装饰下使用这种,(按照括号里的参数名顺序)
'''
例一:BMI指数(bmi是计算而来的,但很明显它听起来像是一个属性而非方法,如果我们将其做成一个属性,更便于理解) 成人的BMI数值:
过轻:低于18.5
正常:18.5-23.9
过重:24-27
肥胖:28-32
非常肥胖, 高于32
体质指数(BMI)=体重(kg)÷身高^2(m)
EX:70kg÷(1.75×1.75)=22.86 # class People:
# def __init__(self,name,weight,height):
# self.name=name
# self.weight=weight
# self.height=height
# @property
# def bmi(self):
# print(self.weight/(self.height**2))
#
# obj1=People('aaa',65,1.78)
# obj1.bmi
# obj1.height=1.70
# # obj1.bmi()
# obj1.bmi # 需要了解的property的用法 setter,deleter
# class People:
# def __init__(self,name):
# self.__name=name
# @property
# def name(self):
# return '<name:%s>' %self.__name
# @name.setter
# def name(self,new_name):
# if type(new_name) is not str:
# print('name error')
# return
# self.__name=new_name
# @name.deleter
# def name(self):
# del self.__name
#
# obj=People('egon练习')
# print(obj.name)
#
# obj.name=123
# print(obj.name)
#
# del obj.name
# print(obj.__dict__) # 常规工作常使用套路
# class People:
# def __init__(self,name):
# self.__name=name
# def xxx_name(self):
# return '<name:%s>' %self.__name
#
# def yyy_name(self,new_name):
# if type(new_name) is not str:
# print('名字必须是str类型')
# return
# self.__name=new_name
# def zzz_name(self):
# del self.__name
#
# name=property(xxx_name,yyy_name,zzz_name)
#
# obj=People('egon练习')
# print(obj.name)
#
# obj.name=123
# print(obj.name)
#
# del obj.name
# print(obj.__dict__)
''' # 多态
'''
1. 什么是多态
同一种事物的多种形态
在程序中用继承可以表现出多态 指的是可以在不考虑对象具体类型的前提下直接使用对象
(比如动物有多种形态:人,狗,猪)
2. 为何要用多态
多态性:可以在不用考虑对象具体类型(类)的前提下而直接使用对象下的方法
多态的精髓:统一标准 3. 如何用多态 多态性
一 什么是多态动态绑定(在继承的背景下使用时,有时也称为多态性)
多态性是指在不考虑实例类型的情况下使用实例 在面向对象方法中一般是这样表述多态性:向不同的对象发送同一条消息
(!!!obj.func():是调用了obj的方法func,又称为向obj发送了一条消息func),
不同的对象在接收时会产生不同的行为(即方法)。也就是说,
每个对象可以用自己的方式去响应共同的消息。
所谓消息,就是调用函数,不同的行为就是指不同的实现,即执行不同的函数。 比如:老师.下课铃响了(),学生.下课铃响了(),老师执行的是下班操作,
学生执行的是放学操作,虽然二者消息一样,但是执行的效果不同 多态性分为静态多态性和动态多态性
静态多态性:如任何类型都可以用运算符+进行运算
动态多态性:如下 peo=People()
dog=Dog()
pig=Pig() #peo、dog、pig都是动物,只要是动物肯定有talk方法
#于是我们可以不用考虑它们三者的具体是什么类型,而直接使用
peo.talk()
dog.talk()
pig.talk() #更进一步,我们可以定义一个统一的接口来使用
def func(obj):
obj.talk() 二 为什么要用多态性(多态性的好处)
其实大家从上面多态性的例子可以看出,我们并没有增加什么新的知识,
也就是说python本身就是支持多态性的,这么做的好处是什么呢? 1.增加了程序的灵活性
以不变应万变,不论对象千变万化,使用者都是同一种形式去调用,如func(animal)
2.增加了程序额可扩展性
通过继承animal类创建了一个新的类,使用者无需更改自己的代码,
还是用func(animal)去调用 >>> class Cat(Animal): #属于动物的另外一种形态:猫
... def talk(self):
... print('say miao')
...
>>> def func(animal): #对于使用者来说,自己的代码根本无需改动
... animal.talk()
...
>>> cat1=Cat() #实例出一只猫
>>> func(cat1) #甚至连调用方式也无需改变,就能调用猫的talk功能
say miao 这样我们新增了一个形态Cat,由Cat类产生的实例cat1,
使用者可以在完全不需要修改自己代码的情况下。
使用和人、狗、猪一样的方式调用cat1的talk方法,即func(cat1) ''' # 强制子类遵循父类的方法:(python并不推崇)
# class xxx(metaclass=abc.ABCMeta):
# @abc.abstractmethod
'''
鸭子类型 逗比时刻: Python崇尚鸭子类型,即‘如果看起来像、叫声像而且走起路来像鸭子,那么它就是鸭子’
python程序员通常根据这种行为来编写程序。
例如,如果想编写现有对象的自定义版本,可以继承该对象
也可以创建一个外观和行为像,但与它无任何关系的全新对象,
后者通常用于保存程序组件的松耦合度。 例1:利用标准库中定义的各种‘与文件类似’的对象,
尽管这些对象的工作方式像文件,但他们没有继承内置文件对象的方法 # 二者都像鸭子,二者看起来都像文件,因而就可以当文件一样去用
class TxtFile:
def read(self):
pass
def write(self):
pass class DiskFile:
def read(self):
pass
def write(self):
pass 例2:其实大家一直在享受着多态性带来的好处,
比如Python的序列类型有多种形态:字符串,列表,元组,多态性体现如下
# str,list,tuple都是序列类型
s = str('hello')
l = list([1, 2, 3])
t = tuple((4, 5, 6)) # 我们可以在不考虑三者类型的前提下使用s,l,t
s.__len__()
l.__len__()
t.__len__() len(s)
len(l)
len(t)
'''
PYTHON-组合 封装 多态 property装饰器的更多相关文章
- 面向对象之 组合 封装 多态 property 装饰器
1.组合 什么是组合? 一个对象的属性是来自另一个类的对象,称之为组合 为什么要用组合 组合也是用来解决类与类代码冗余的问题 3.如何用组合 # obj1.xxx=obj2''''''# class ...
- 面向对象(三)——组合、多态、封装、property装饰器
组合.多态.封装.property装饰器 一.组合 1.什么是组合 组合指的是某一个对象拥有一个属性,该属性的值是另外一个类的对象 class Foo(): pass class Bar(): pas ...
- python面向对象:组合、封装、property装饰器、多态
一.组合二.封装三.property装饰器四.多态 一.组合 ''' 1. 什么是组合 一个对象的属性是来自于另外一个类的对象,称之为组合 2. 为何用组合 组合也是用来解决类与类代码冗余的问题 3. ...
- python面向编程:类的组合、封装、property装饰器、多态
一.组合 二.封装 三.propert装饰器 四.多态 一.组合 ''' 1. 什么是组合 一个对象的属性是来自于另外一个类的对象,称之为组合 2. 为何用组合 组合也是用来解决类与类代码冗余的问题 ...
- 组合,多态与多态性,封装以及property装饰器介绍
一:组合: 什么是组合:组合指的是某一个对象拥有一个属性,该属性的值是另外一个类的对象. 为何要用组合:通过为某一个对象添加属性(属性的值是另外一个类的对象)的方式,可以间接地将两个类关联/整合/组合 ...
- 面向对象(三)--多态、封装、property装饰器
一.多态与多态性 1.什么是多态 多态指的是同一种类/事物的不同形态 class Animal: def speak(self): pass class People(Animal): def spe ...
- 面向对象之封装 及@property装饰器使用
目录 封装 1.封装的定义 2.封装的目的: 3.封装的三种方式 4.封装的优点 5.访问限制(封装) @property 装饰器 属性property底层实现 封装 1.封装的定义 将复杂的丑陋的, ...
- 初识面向对象-封装、property装饰器、staticmathod(静态的方法)、classmethod(类方法) (五)
封装 # class Room:# def __init__(self,name,length,width):# self.__name = name# self.__length = length# ...
- 【python】@property装饰器
Python内置的@property装饰器可以把类的方法伪装成属性调用的方式.也就是本来是Foo.func()的调用方法,变成Foo.func的方式.在很多场合下,这是一种非常有用的机制. class ...
随机推荐
- 百度地图BMapLib.InfoBox 手机兼容源码修改
InfoBox.prototype.initialize = function (map) { var me = this; var div = this._div = baidu.dom.creat ...
- oi程序提交注意:bool
比如我一个程序用了bool类型(#include<stdbool.h>) 在poj以c的方式提交不通过显示Compile Error,而用gcc的方式提交通过了, 也许其它的#includ ...
- NO.7:别让异常逃离析构函数
1.析构函数绝对不要吐出异常,如果一个析构函数可能抛出异常,析构函数应该捕获任何异常,然后要么吞下它们或者退出程序 2.如果用户需要对析构内的可能抛出异常的操作做出反应,则应该将操作放入除析构函数外的 ...
- Codeforces Round #523 (Div. 2) C. Multiplicity
C. Multiplicity 题目链接:https://codeforc.es/contest/1061/problem/C 题意: 给出一串数,问它的“好序列“有多少.好序列的定义是,首先是一个子 ...
- Kubernetes Pv & Pvc
Kubernetes PV & pvc 介绍 PersistentVolume(pv)和PersistentVolumeClaim(pvc)是k8s提供的两种API资源,用于抽象存储细节.管理 ...
- sed命令使用介绍(转载)
sed命令介绍 (转自:http://www.cnblogs.com/ggjucheng/archive/2013/01/13/2856901.html) 简介 sed 是一种在线编辑器,它一次处理一 ...
- X-UA-compatible浅谈
最近了解到svg,原来它出现之前好几年,微软已经推出了vml,但是那时候却被人吐槽无数,看来过早的创新也是失败的原因之一呢~ 为什么谈到这个话题呢?因为IE史上有一个特别奇怪的浏览器IE8,它及不兼容 ...
- 修改Linux Operating System的时间与时区
修改Linux Operating System的时间与时区 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 有的小伙伴可能会遇到安装了linux系统后,尽管时区选择正确,也会发现系 ...
- python自动化运维之路~DAY4
python自动化运维之路~DAY4 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.模块导入方式详解. 关于Python有几种导入模块的方式,请参考我的另外一篇博客(第20 ...
- 函数和常用模块【day04】:作用域、局部和全局变量(四)
本节内容 概述 课前前引 局部变量和全局变量 总结 一.概述 我们之前写代码,都需要声明变量,但是我们思考过变量的作用范围吗?今天我们就来讲讲变量的作用范围,这个作用范围又叫作用域.首先我们根据变量的 ...