封装和@property

一、复习

1、接口类和抽象类

  python中没有接口类,有抽象类,abc模块中的metaclass=ABCMeta,@abstructmethod,本质是做代码规范用的,希望在子类中实现和父类方法名完全一样的方法

  在Java的角度上是有区别的:

    Java本来支持单继承,所以就有了抽象类

    Java没有多继承,所以为了接口隔离原则,设计了接口这个概念,支持多继承了

  python既支持单继承也支持多继承,所以对于接口类和抽象类的区别就不那么明显了

甚至在python中没有内置接口类

2、多态和鸭子类型

  多态----python天生支持多态

  鸭子类型----不依赖父类的情况下实现两个相似的类中的同名方法

3、封装----私有的

  在python中只要__名字,就把这个名字私有化了,私有化了之后,就不能从类的外部直接调用了

  静态属性,方法,对象属性都可以私有化;这种私有化只是代码级别做了变形,并没有真的约束

  变形机制:_类名__名字,在类用这个调用,在类的内部直接__名字调用

二、封装和@property

1、其他语言,比如C语言里面的属性的编写习惯统

习惯将类里面的属性设置成私有属性,为了防止外部随便调用和修改,在内部的私有属性都会有一个get和set的方法,这也是这类语言编程习惯,一般C语言的私有属性的使用方法如下代码:

class Room:def __init__(self,name,length,width):self.__name = name  # 私有属性self.__length = length  # 私有属性self.__width = width  # 私有属性def get_name(self):  # get方法返回获取名字return self.__namedef set_name(self,newName):  # set方法判断是否需要改名if type(newName) is str and newName.isdigit()== False:self.__name = newNameelse:print('不合法的姓名')def area(self):return self.__length * self.__widthW = Room('wm',3,4)  # 实例化
print(W.area())
W.set_name('6')  # 传入新的名字
print(W.get_name())  # 打印最后的名字

运行结果:

12
不合法的姓名
wm

2、父类的私有属性能被子类调用吗?

假设可以被调用,如下代码:

class Foo:__key = '123'
class Son(Foo):print(Foo.__key)

运行结果:

Traceback (most recent call last):File "<encoding error>", line 26, in <module>File "<encoding error>", line 27, in Son
AttributeError: type object 'Foo' has no attribute '_Son__key'

从结果中可以看出假设是不成立的,为什么会这样呢,要想到私有化的那个变形机制,在父类中当key私有化之后,它本质上是以_Foo__key存储下来的,而当子类继承父类之后,实质上是以_Son__key存储的,所以当直接继承打印Foo.__key才会报错,并且不存

3、会用到私有的这个概念的场景

(1)隐藏起一个属性。不想要类的外部调用

(2)我想保护这个属性,不想让属性随意被改变

(3)我想保护这个属性不被子类继承

4、property----内置装饰器函数 只在面向对象中使用,使用类下的公有函数方法修改删除类里面的私有属性

from math import pi
class Circle:def __init__(self,r):self.r = r@propertydef perimeter(self):  # @property后面的self都不能传参数return 2*pi*self.r@propertydef area(self):  # @property后面的self都不能传参数return self.r**2*pi
c1 = Circle(5)
print(c1.area)  # 圆的面积 area后面没有括号了,直接拿对象.方法来获取
print(c1.perimeter)  # 圆的周长

运行结果:

78.53981633974483
31.41592653589793

@property是将函数或者方法伪装成属性,可以直接在外部 类名.函数名/方法发 进行调用,不用在函数或方法名后面加上括号,并且在加上之后,不能再通过 类名.函数名/方法名=‘XXX’ 赋值的方法进行修改了,所以一般用到@property的装饰器都是一些固定的,不经常变动的方法函数

5、利用@property和@xxx.setter修改私有属性  

可以通过下面的例子知道如何在加上@property装饰器后,还能对通过类名.函数方法=‘xxx’的方法进行修改相关的私有属性

class Person:def __init__(self,name):self.__name = name@property  # 将name函数伪装成属性,后面直接调用函数名字不用加上括号,并且限制了类名.函数方法='xxx'的改变def name(self):return self.__name + '是sb'@name.setter  # 改变机制,可以解除前面 类名.函数方法='xxx'的改变的限制def name(self,new_name):self.__name = new_name
cc = Person('ww')
print(cc.name)
cc.name = 'dd'  # 加上@name.setter以及后面的操作后,这里就可以改变名字属性了
print(cc.name)

运行结果:

ww是sb
dd是sb

要特别注意的是:@property里面的函数名字name和 @name.setter里面的name以及里面的函数方法名字这三个是同一个名字,只要是其中一个不一样,都不能进行修改再来一个购物打折的例子:
class Goods:discount = 0.8  # 折扣数,当不想要这个折扣的时候,就可以在这里直接改变就可以了,其他都不需要要def __init__(self,name,price):self.name = nameself.__price = price@property  # 将price伪装成为属性,并且不用传参数def price(self):return self.__price * Goods.discount  # 苹果的价格等于实际价格剩余打折数
apple = Goods('苹果',5)
print(apple.price)  

# 直接就可以调用函数price,但是也可以说是私有属性price,加上@property后可以等同,但是本质还是name函数

运行结果:

4.0

6、利用@property和xxx.deleter删除私有属性

没有执行del self.__name

class Person:def __init__(self,name):self.__name = name@property  # 将函数伪装成属性,实例化后可以直接调用私有属性,本质上还是函数def name(self):return self.__name@name.deleter  # 启动删除机制,没有实际执行删除操作,而是通过@name里面的函数方法执行相应的删除操作def name(self):print('执行了这个方法')# del self.__name  # 实际的删除操作
brother2 = Person('二哥')
print(brother2.name)
del brother2.name  # 一出现del就要回到@xxx.deleter的地方并且在其下面函数方法中执行对应的删除操作
print(brother2.name)

运行结果:

二哥
执行了这个方法
二哥

执行del self.__name

class Person:def __init__(self,name):self.__name = name@property  # 将函数伪装成属性,实例化后可以直接调用私有属性,本质上还是函数def name(self):return self.__name@name.deleter  # 启动删除机制,没有实际执行删除操作,而是通过@name里面的函数方法执行相应的删除操作def name(self):  # 不能传参数print('执行了这个方法')del self.__name  # 实际的删除操作
brother2 = Person('二哥')
print(brother2.name)
del brother2.name  # 一出现del触发删除机制,就要回到@xxx.deleter的地方并且在其下面函数方法中执行对应的删除操作
print(brother2.name)
print(brother2.name)

运行结果:

二哥
Traceback (most recent call last):
执行了这个方法File "<encoding error>", line 89, in <module>File "<encoding error>", line 81, in name
AttributeError: 'Person' object has no attribute '_Person__name'

从上面这两个程序以及运行结果可以验证:

(1)当 del brother2.name  的时候就会马上触发删除机制 @name.deleter,而触发删除机制后,只有做出对应的删除操作 del self.__name ,最后才会删除成功

(2)在存在@property的前提下,(1)中的三个name和@name deleter下面的函数名name所在的地方的名字以及@property下面的函数方法名字必须是相同的,否则就不能做到删除私有属性的操作


转载于:https://www.cnblogs.com/wxm422562/p/11061721.html

封装和@property相关推荐

  1. python全栈脱产第25天------组合、多态、封装、property装饰器

    一.组合 定义:组合指的是一个对象属性的值是另一个类的对象 用途:通过某一对象添加属性(属性的值时另一个类的对象)的方式,可以间接地将两个类关联/整合/组合到一起,从而减少代码冗余 用法: class ...

  2. 面向对象----封装之proprty

    封装之 property: property是一种特殊的属性,访问它时会执行一段功能(函数),然后返回值. 把类体的方法,通过property函数来装饰,装饰过的方法对象可以通过调用属性的方式来调用, ...

  3. Mars3D开发基础学习:Property属性机制

    平台是数据驱动和 time-dynamic visualization,这些可都是仰仗Property属性机制来实现的. Property最大的特点是和时间相互关联,在不同的时间可以动态地返回不同的属 ...

  4. python 类的封装

    一.什么是封装 在程序设计中,封装(Encapsulation)是对具体对象的一种抽象,即将某些部分隐藏起来,在程序外部看不到,其 含义是其他程序无法调用. 想要了解封装,就离不开"私有化& ...

  5. 封装与多态的基本使用

    封装与多态的基本使用 1 > 派生的实际应用 2 > 面向对象之封装 3 > property 4 > 面向对象之多态 4.1 > 鸭子类型 5 > 面向对象之反射 ...

  6. Mybatis复习笔记3:映射文件详解

    映射文件详解 参数处理(#和$的区别) #{}:可以获取map中的值或者实体对象属性的值: ${}:可以获取map中的值或者实体对象属性的值: select * from person where i ...

  7. MyBatis映射文件(二)

    目录 一.select元素 1.参数介绍 2.resultType (1)返回实体对象 (2)返回List集合 (3)返回Map集合,object> (4)返回Map<主键类型,实体类&g ...

  8. MyBatis Mapper Demo

    <?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE mapper PUBLIC "-/ ...

  9. mybatis关系映射(1对1,1对多,多对多)

    1.映射(多)对一.(一)对一的关联关系          1).使用列的别名             ①.若不关联数据表,则可以得到关联对象的id属性             ②.若还希望得到关联对 ...

最新文章

  1. 微信小程序云数据库触底分页加载,下拉无限加载,第一次请求数据随机,随机获取数据库的数据
  2. JavaScript中的普通函数与构造函数比较
  3. vim中Mapping already in use: LocalLeaderis, mode n错误解决的方法解释
  4. docker 和挂载文件一起打包成新镜像_Docker文件系统和数据卷
  5. CodeForces - 1316E Team Building(状压dp)
  6. 曲线 线性回归_GRAPHPAD作图技巧(二)--拟合曲线
  7. 空中最亮的星用计算机弹数字,赤峰——夜空中最亮的星_数字展示在线
  8. IPv6 to IPv4过渡技术——6to4隧道配置实例
  9. RHEL 6.5----haproxy实现负载均衡
  10. Swashbuckle Swagger组件扩展
  11. 软件测试面试题整理(三)之工作/项目流程篇
  12. QT之Tcp数据发送测试工具
  13. Linux快捷键详解
  14. 环形电流计算公式_环形电感的计算公式
  15. 硬盘显示驱动器中的光盘未被格式化的解决方法
  16. Python自动回复微信好友新年祝福
  17. 视频广告播放器(缓存、循环播放)
  18. I3D【Inflated 3D ConvNet】——膨胀卷积网络用于行为识别
  19. java获取excel行数_Java如何利用POI读取Excel行数
  20. Javascript 报错 Uncaught TypeError: x is not a function 常见错误总结

热门文章

  1. scrapy shell 用法(慢慢更新...)
  2. nyoj 78:圈水池 【凸包入门】
  3. 【BZOJ3489】A simple rmq problem(树套树)
  4. BZOJ1110 : [POI2007]砝码Odw
  5. C语言第三节关键字、标识符、注释
  6. 在Microsoft VS.net 2005下开发Linux环境的网站应用
  7. 面试必知的25个经典回答 ,最全的面试干货,没有之一
  8. 二手车没有车险能过户吗?
  9. 为什么办理房产证要经过开发商和物业?
  10. 人究竟能抠到什么程度?