封装和@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.__name def set_name(self,newName): # set方法判断是否需要改名
if type(newName) is str and newName.isdigit()== False:
self.__name = newName
else:
print('不合法的姓名') def area(self):
return self.__length * self.__width W = Room('wm',3,4) # 实例化
print(W.area())
W.set_name('') # 传入新的名字
print(W.get_name()) # 打印最后的名字

运行结果:

12
不合法的姓名
wm

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

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

class Foo:
__key = ''
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
@property
def perimeter(self): # @property后面的self都不能传参数
return 2*pi*self.r
@property
def 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 = name
self.__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下面的函数方法名字必须是相同的,否则就不能做到删除私有属性的操作

												

封装和@property的更多相关文章

  1. 封装之property,多态,鸭子类型,classmethod与staticmethod

    一.封装之Property prooerty是一种特殊的属性,访问时他会执行一段功能(函数)然后返回 '''BMI指数(bmi是计算而来的,但很明显它听起来像是一个属性而非方法,如果我们将其做成一个属 ...

  2. 抽象类,接口类,封装,property,classmetod,statimethod

    抽象类,接口类,封装,property,classmetod,statimethod(类方法,静态方法) 一丶抽象类和接口类 接口类(不崇尚用) 接口类:是规范子类的一个模板,只要接口类中定义的,就应 ...

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

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

  4. day32--面向对象的程序设计之继承实现的原理(继承顺序)、封装、property

    Python之路,Day19 = 面向对象的程序设计之继承实现的原理(继承顺序).封装.property 以下类容参考自:http://www.cnblogs.com/metianzing/p/712 ...

  5. 4月16日 python学习总结 封装之property、多态 、classmethod和staticmethod

    一.封装之property @property把一个函数伪装成一个数据类型  @伪装成数据的函数名.setter   控制该数据的修改,修改该数据时触发 @伪装成数据的函数名.delect  控制该数 ...

  6. 类的封装(property)

    封装 封装程序的主要原因:保护隐私:而封装方法的主要原因是:隔离复杂的执行过程 property的特性 将一个类的函数定义成特性以后,对象再去使用的时候obj.name,根本无法察觉自己的name是执 ...

  7. 封装、property特性及绑定与非绑定方法

    1.封装 (1)什么是封装? 封:属性对外是隐藏的,但对内是开放的: 装:申请一个名称空间,往里面装入一系列名字/属性 (2)为什么要封装? 封装数据属性的目的 首先定义属性的目的就是为了给类外部的使 ...

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

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

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

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

随机推荐

  1. mysql插入的时间莫名的加一秒

    1.问题描述 我获取当天最大的时间: public static Date getEndOfDay(Date date) { LocalDateTime localDateTime = LocalDa ...

  2. 【JMX】jmx结合jmx_exporter实现promethues监控

    JMX JMX的全称为Java Management Extensions. 顾名思义,是管理Java的一种扩展.这种机制可以方便的管理.监控正在运行中的Java程序.常用于管理线程,内存,日志Lev ...

  3. javaSE笔记(重点部分)

    Java 基础篇 数据类型 基本数据类型 由于java是强类型语言,所以要进行有些运算的时候,需要用到类型转换. 低-----------------------------高 byte,short, ...

  4. Oracle SCN 详解

    一.简介 scn,system change number 在某个时间点定义数据库已提交版本的时间戳标记,Oracle为每个已提交事务分配一个唯一的scn,scn值是对数据库进行更改的逻辑时间点.sc ...

  5. Springboot整合webservice

    Springboot整合webservice 2019-12-10 16:34:42 星期二 WebService是什么 WebService是一种跨编程语言和跨操作系统平台的远程调用技术,服务之间的 ...

  6. 小白学 Python 爬虫(15):urllib 基础使用(五)

    人生苦短,我用 Python 前文传送门: 小白学 Python 爬虫(1):开篇 小白学 Python 爬虫(2):前置准备(一)基本类库的安装 小白学 Python 爬虫(3):前置准备(二)Li ...

  7. 浏览器主页锁定之战——IE:我太难了

    精彩回顾: 我是一个explorer的线程 我是一个杀毒软件线程 我是一个IE浏览器线程 比特宇宙-TCP/IP的诞生 产品vs程序员:你知道www是怎么来的吗? Hello, World! 我是一个 ...

  8. (全国多校重现赛一)F-Senior Pan

    Senior Pan fails in his discrete math exam again. So he asks Master ZKC to give him graph theory pro ...

  9. Redis 使用消息隊列

    關鍵函數 ListRightPush  生產消息 ListRightPop   消費消息 這是從右面增或取 左邊亦然

  10. Dubbo加权轮询负载均衡的源码和Bug,了解一下?

    本文是对于Dubbo负载均衡策略之一的加权随机算法的详细分析.从2.6.4版本聊起,该版本在某些情况下存在着比较严重的性能问题.由问题入手,层层深入,了解该算法在Dubbo中的演变过程,读懂它的前世今 ...