封装和@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. OC 初次接触

    初次接触ObjC时,会发现许多和其它语言不同的地方,会看到很多的+,- ,[ ,] ,@, NS等符号,这些符号在以后的编程中将经常看到,这部分内容在第二节中介绍.先熟悉一下ObjC的代码: #imp ...

  2. Sublime Text 3 免费注册方法(福利)

    对于使用Sublime Text但是又不愿花钱注册的小伙伴,福利到了,免费注册一下你的Sublime吧. 版本3207: 打开Sublime text,然后点击菜单Help->Enter Lis ...

  3. shell脚本中的逻辑判断、文件目录属性判断、if特殊用法、case判断

    7月12日任务 20.5 shell脚本中的逻辑判断20.6 文件目录属性判断20.7 if特殊用法20.8/20.9 case判断 20.5 shell脚本中的逻辑判断 逻辑判断在shell中随处可 ...

  4. xcode8 运行项目时自己打印些东西

    使用 Xcode 8 运行工程的时候,在打印台会发现如下这些奇怪的日志输出: 2016-09-19 10:43:44.001757 Demo[7100:171568] subsystem: com.a ...

  5. ViewGroup dispatchTouchEvent方法中 mFirstTouchTarget标志是否为空的含义

    在ViewGroup dispatchTouchEvent方法中首次出现mFirstTouchTarget的语句为: if (actionMasked == MotionEvent.ACTION_DO ...

  6. ThinkPHP 中 where条件 or,and 同时使用

    ('a'=1 and 'b'=2) or ('c'=3 and 'd'=4) and 'e'=5 $where_1['a'] = 1; $where_1['b'] = 2; $where_2['c'] ...

  7. mybatis中因为不理解$与#而出现的bug

    最近项目中遇到一个bug,正常的流程是这样的:要上传一个应用,首先检查系统中是否已经存在这个应用的更高版本,如果存在,则上传操作将被取消. bug体现为当传入系统中存在的所有应用与新上传的应用的ver ...

  8. 大型情感剧集Selenium:6_selenium中的免密登陆与cookie操作

    网站登录 现在各大平台在反爬虫功能上,可谓花样繁多.总结下来按照破解成功率排名,最高的是滑动解锁.其次是验证码数字.之后是一次点击对应的汉字,最后是想12306之前那种反人类的让你说那些是奶糖吧,哈哈 ...

  9. 掘金转载-手写一个Promise

    目录 一 什么是Promise ? 二 Promises/A+ 规范 2.1 术语 2.2 基本要求 2.2.1. Promise的状态 2.2.2. Then 方法 2.3 简易版实践 2.4 进一 ...

  10. react-native中TextInput在ios平台下不能输入中文

    目录 1. github上相关资料 2.需要满足defultValue和value属性 react-native 0.55.4版本,发现TextInput 在iOS平台上无法输入中文的问题. 1. g ...