封装

1.封装的定义

	将复杂的丑陋的,隐私的细节隐藏到内部,对外提供简单的使用接口, 对外隐藏内部实现细节,并提供访问的接口;

2.封装的目的:

	 封装的目的: 面向对象的核心是对象二字,精髓在于整合,封装的目的其实就是把一堆数据属性和方法属性整合到对象中,我们可以把对象比喻成一个容器,其实就是为了把数据存入一个容器中。存的目的就是为了取的,那封装到对象中的好处就是可以通过 "对象.属性" 的方式把属性取出来。

3.封装的三种方式

	1.publish: 对所有人**公开权限**,在外部哪里都可以访问。

	2.protected: 这种只有在当前类,或者是继承它的 “**子类**” (派生类) 可以访问。

	3.private: 这种封装相当于**属性的私有化**,让任何人都不可以访问。但是python没有这种强	制限制的机制,只能通过访问限制机制去实现表面上的属性私有化

4.封装的优点

1.良好的封装能够减少耦合。

2.类内部的结构可以自由修改。

3.可以对成员变量进行更精确的控制。

4.隐藏信息,实现细节。
# 不封装
def login(name, pwd, root):
if name == 'tank' and pwd == '123' and root == 'vip':
print('登录成功!')
else:
print('登录失败!') login('tank', '123', 'vip') # 封装
class User:
def __init__(self, name, pwd, root):
self.__name = name
self.__pwd = pwd
self.__root = root def change_pwd(self, new_pwd):
self.__pwd = new_pwd @property
def get_name(self):
return self.__name @property
def get_pwd(self):
return self.__pwd @property
def get_root(self):
return self.__root user_obj = User('tank', '123', 'vip') def login(obj):
if obj.get_name == 'nim' and obj.get_pwd == '123' and obj.get_root == 'vip':
print('登录成功!') else:
print('登录失败') user_obj.change_pwd('321')
login(user_obj)

5.访问限制(封装)

	隐藏对象的属性和实现细节,仅对外提供公共访问方式。
目的:
1.为了保证 关键数据的安全性
2.对外部隐藏实现细节,隔离复杂度

1)私有变量

#其实这仅仅这是一种变形操作
#类中所有双下划线开头的名称如__x都会自动变形成:_类名__x的形式: class A:
__N=0 #类的数据属性就应该是共享的,但是语法上是可以把类的数据属性设置成私有的如__N,会变形为_A__N
def __init__(self):
self.__X=10 #变形为self._A__X
def __foo(self): #变形为_A__foo
print('from A')
def bar(self):
self.__foo() #只有在类内部才可以通过__foo的形式访问到. #A._A__N是可以访问到的,即这种操作并不是严格意义上的限制外部访问,仅仅只是一种语法意义上的变形

这种变形的特点:

	1.类中定义的__x只能在内部使用,如 self.__x,引用的就是形变的结果;

	2.这种变形其实正是针对外部的变形,在外部是无法通过__x这个名字访问到的;

	3.在子类定义的__x不会覆盖在父类定义的 __x,因为子类中变形成了:_子类名__x,而父类中变形成了:_父类名__x,即双下划线开头的属性在继承给子类时,子类时无法覆盖的。

这种变形需要注意的问题是:

1.这种机制也并没有真正意义上限制我们从外部直接访问属性,知道了类名和属性名就可以拼出名字:_类名__属性,然后就可以访问了,如a._A__N

2.变形的过程只在类的内部生效,在定义后的赋值操作,不会变形

>>> a=A()
>>> a.__dict__
{'_A__X':10}
>>> a.__Y = 1
>>> a.__dict__
{'_A__X':10,'__Y':1}

2)私有方法

	在继承中,父类如果不想让子类覆盖自己的方法,可以将方法定义成私有的。
# 正常情况
class A:
def fa(self):
print('from A')
def test(self):
print(self) # <__main__.B object at 0x00000000023A6FD0>
self.fa() class B(A):
def fa(self):
print('from B') b=B()
b.test() # 把fa定义成私有的,即__fa
class A:
def __fa(self): #在定义时就变形为_A__fa
print('from A')
def test(self):
print(self) # <__main__.B object at 0x00000000023A6FD0>
self.__fa() #只会与自己所在的类为准,即调用_A__fa class B(A):
def __fa(self):
print('from B') b=B()
b.test()

示例1:

#222  我们通过对__id_number进行修改,来对比外部是否能对__开头的属性进行修改.# p.show_id()  #12312  内部值并没有发生变化,所以通过__开头进行封装,外部通过普通方法并不能对内部的值进行修改.

示例2:

  # 通过__方法名()的方式,将内部内部具体实现细节封装起来,只给外部暴露一个简单的开机(open函数)接口
class PC:
def __init__(self,price,color,kind):
self.color=color
self.price=price
self.kind=kind
def open(self):
print('接通电源')
self.__check_device()
print('载入内核')
print('初始化内核')
self.__start_service()
print('启动GUI')
self.__login() def __check_device(self):
print('检测硬件1')
print('检测硬件2')
print('检测硬件3')
print('检测硬件4')
def __start_service(self):
print('启动服务1')
print('启动服务2')
print('启动服务3')
def __login(self):
print('login..') pc=PC(6000,'white','lenvo')
pc.open()
'''
接通电源
检测硬件1
检测硬件2
检测硬件3
检测硬件4
载入内核
初始化内核
启动服务1
启动服务2
启动服务3
启动GUI
login.. '''

示例3:

# e.g.1
class Downloader:
def __init__(self,filename,url,buffer_size):
self.filename=filename
self.url=url
self.__buffer_size=buffer_size def start_download(self):
if self.__buffer_size<=1024*1024:
print('开始下载...')
else:
print('内存不足..') d=Downloader('葫芦娃','http://www.baidu.com',1024*1024) d.buffer_size=1024*1024*10 #类中__开头的属性无法直接访问,所以无法修改。
d.start_download()
'''
开始下载...
''' # e.g.2
class Downloader:
def __init__(self,filename,url,buffer_size):
self.filename=filename
self.url=url
self.__buffer_size=buffer_size def start_download(self):
if self.__buffer_size<=1024*1024:
print('开始下载...')
print('当前缓冲区大小为%s'%self.__buffer_size)
else:
print('内存不足..') def set_buffer_size(self,size):
self.__buffer_size=size
def get_buffer_size(self):
return self.__buffer_size d=Downloader('葫芦娃','http://www.baidu.com',1024*1024) #通过函数去修改内部封装的属性
d.set_buffer_size(1024*1024/2) #通过函数访问内部封装的属性
print(d.get_buffer_size())
d.start_download() '''
524288.0
开始下载...
当前缓冲区大小为524288.0
'''

示例4

class Foo:
__name = 'rose' # 变形成 _Foo__name
def __init__(self,sex):
self.__sex=sex @property
def __sing(self): # 变形成_Foo__sing,外部可以通过变形后的名字从外部直接访问。
print('我是隐藏属性,不懂其变形情况在外部是无法访问的!') f = Foo('male')
print(f._Foo__name)
f._Foo__sing '''
rose
我是隐藏属性,不懂其变形情况在外部是无法访问的! '''

@property 装饰器

	@property 用来装饰类内部的函数:@property使对象调用某个方法时,将对象.方法()变成对	象.方法得形式,使对象点函数(方法)看起来像是点普通的数据属性一样,使得调用方式一	致。切记:装饰过后的函数名后就不要再加括号啦!!!

@key.setter 该装饰器用在修改属性的方法上
@key.deleter 该装饰器用在删除属性的方法上 注意:key是被property装饰的方法的名称,也就是属性的名称 ;内部会创建一个对象 变量名称就是函数名称,所以在使用setter和deleter时 必须保证使用对象去调用方法,所以是 key.setter和key.deletter。

属性property底层实现

一个静态属性property本质就是实现了get,set,delete三种方法

class Foo:
def get_AAA(self):
print('get的时候运行我啊') def set_AAA(self,value):
print('set的时候运行我啊') def delete_AAA(self):
print('delete的时候运行我啊')
AAA=property(get_AAA,set_AAA,delete_AAA) #内置property三个参数与get,set,delete一一对应 f1=Foo()
f1.AAA
f1.AAA='aaa'
del f1.AAA

示例1

class Foo:
def __init__(self,val):
self.__NAME=val #将所有的数据属性都隐藏起来 @property
def name(self):
return self.__NAME #obj.name访问的是self.__NAME(这也是真实值的存放位置) @name.setter
def name(self,value):
if not isinstance(value,str): #在设定值之前进行类型检查
raise TypeError('%s must be str' %value)
self.__NAME=value #通过类型检查后,将值value存放到真实的位置self.__NAME @name.deleter
def name(self):
raise TypeError('Can not delete') f=Foo('tank')
# print(f.name)
# f.name = 10 # 抛出异常'TypeError: 10 must be str'
f.name = 'zhang'
print(f.__dict__) # {'_Foo__NAME': 'zhang'}
# del f.name #抛出异常'TypeError: Can not delete'

示例2

class People:
def __init__(self,name,weight,height):
self.name=name
self.weight=weight
self.height=height
@property
def bmi(self):
return self.weight / (self.height**2) p1=People('egon',75,1.85)
print(p1.bmi) #直接实例对象 点 方法名称就可以了

示例3

class Goods:

    def __init__(self):
# 原价
self.original_price = 100
# 折扣
self.discount = 0.8 @property
def price(self):
# 实际价格 = 原价 * 折扣
new_price = self.original_price * self.discount
return new_price @price.setter
def price(self, value):
self.original_price = value @price.deleter
def price(self):
del self.original_price obj = Goods()
obj.price # 获取商品价格
obj.price = 200 # 修改商品原价
print(obj.price)
del obj.price # 删除商品原价

面向对象之封装 及@property装饰器使用的更多相关文章

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

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

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

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

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

    一.多态与多态性 1.什么是多态 多态指的是同一种类/事物的不同形态 class Animal: def speak(self): pass class People(Animal): def spe ...

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

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

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

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

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

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

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

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

  8. 初识面向对象-封装、property装饰器、staticmathod(静态的方法)、classmethod(类方法) (五)

    封装 # class Room:# def __init__(self,name,length,width):# self.__name = name# self.__length = length# ...

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

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

随机推荐

  1. tcl的第二个脚本

    set val(chan) Channel/WirelessChannel ;#Channel Typeset val(prop) Propagation/TwoRayGround ;# radio- ...

  2. python3.6调用c语言动态编译文件 c语言编译可执行文件和动态编译等

    1.c的代码 dfunc.c #include<stdio.h> int dgfunc(int n) { ){ ; }else{ )+dgfunc(n-); } } 2.动态编译 cmd ...

  3. java文件操作解析

    转载:http://blog.csdn.net/cynhafa/article/details/6882061 字节流与和字符流的使用非常相似,两者除了操作代码上的不同之外,是否还有其他的不同呢? 实 ...

  4. jmeter—获取当前时间(年、月、日),往前/往后n天

    import java.util.Calendar; Calendar cal = Calendar.getInstance(); int day = cal.get(Calendar.DATE); ...

  5. Python全栈开发之2、数据类型-数值、字符串、列表、字典、元组和文件处理

    一.Python 运算符 1.算术运算: 2.比较运算: 3.赋值运算: 4.逻辑运算: 5.成员运算: 二.基本数据类型 1.数字整型 int(整型) 在32位机器上,整数的位数为32位,取值范围为 ...

  6. kafka服务器批量copy文件脚本

    #!/bin/bashif [[ $# -lt 1 ]] ; then echo no params ; exit ; fiparams=$@for (( i=1 ; i <= 3 ; i = ...

  7. [Python3] 039 语法调试

    目录 语法调试 1. 调试技术 2. pdb 调试 插一个 gdb 3.Pycharm 调试 4. 单元测试 语法调试 1. 调试技术 调试流程 单元测试 → 集成测试 → 交测试部 分类: 静态调试 ...

  8. 【转帖】2018全球公有云IaaS榜单出炉:阿里、腾讯、中国电信、金山云列前十

    2018全球公有云IaaS榜单出炉:阿里.腾讯.中国电信.金山云列前十 https://news.cnblogs.com/n/628391/ 中国电信貌似就是用的华为的技术 阿里 腾讯 华为 金山 百 ...

  9. 小记---------sparkRDD的Transformation 和 Action 及案例 原理解释

    RDD :弹性分布式数据集:是一个容错的.并行的数据结构,可以让用户显式地将数据存储到磁盘或内存中,并控制数据的分区   RDD是Spark的核心数据结构,通过RDD的依赖关系形成Spark的调度顺序 ...

  10. # IDEA相关知识

    目录 IDEA相关知识 安装目录下: 配置目录下: 工程目录下: 名词解释 IDEA相关知识 安装目录下: bin:启动文件,配置信息,IDEA的一些属性信息 jre64:IDEA自带的运行环境 li ...