面向对象 - 1.封装之如何实现属性的隐藏/2.封装的意义/3.封装与扩展性/4.property的使用
1.封装之如何实现属性的隐藏
封装: __x=1 # 把数据属性隐藏 (如何实现隐藏) 类定义阶段 __开头发生了变形 __x --> _A__x
特点:
1.在类外部无法直接:obj.__AttrName
2.在类内部是可以直接使用:obj.__AttrName # 为什么会这样?python 如何实现的 !类定义阶段已经变形 #__x --> _A__x #self._A_foo()
3.子类无法覆盖父类__开头的属性 它两根本不是一个名字 #_Foo__func #_Bar__func
总结:
这种变形需要注意的问题:
1.这种机制也并没有真正意义上限制我们从外部直接访问属性,知道了类名和属性名就可以拼出名字:_类名__属性,然后就可以访问了,如a._A__N eg:print(A._A__x)
2.变形的过程只在类的定义时发生一次,在定义后的赋值操作,不会变形 eg: b.__age=18 {'_B__name': 'alice', '__age': 18}
3.在继承中,父类如果不想让子类覆盖自己的方法,可以将方法定义为私有的 eg: def __foo(self): #_A__foo
# class A:
# __x=1 # 把数据属性隐藏 _A__x
#
# def __init__(self,name):
# self.__name=name # _A__name
#
# def __foo(self): # 把函数属性隐藏 def _A__foo(self)
# print('run foo')
#
# def bar(self):
# self.__foo() #self._A_foo() 定义阶段已经变了
# print('from bar') # print(A.__dict__)
# {'__module__': '__main__', '_A__x': 1, '__init__': <function A.__init__ at 0x0000017AC92B0B70>, '_A__foo': <function A.__foo at 0x0000017AC92B0BF8>, '__dict__': <attribute '__dict__' of 'A' objects>, '__weakref__': <attribute '__weakref__' of 'A' objects>, '__doc__': None}
# 类定义阶段 __开头发生了变形 __x --> _A__x # print(A.__x)
# print(A._A__x)
# print(A.__foo)
# print(A._A__foo)
# a=A('alice')
# print(a.__dict__)
# {'_A__name': 'alice'}
# print(a.__name)
# print(a._A__name)
# a.bar() # ---------------------------------------------
# class Foo:
# def __func(self): #_Foo__func
# print('from foo')
#
# class Bar(Foo):
# def __func(self): #_Bar__func
# print('from bar') # b=Bar()
# b.func() # 子类把父类 重名的给覆盖掉了 # ---------------------------------------------
# class B:
# __x=1
#
# def __init__(self,name):
# self.__name=name # # print(B._B__x) # 一般不要这么做!py不让你这么干了
# B.__y=2 # 类定义完之后不会发生变形
# # print(B.__dict__)
# # {'__module__': '__main__', '_B__x': 1, '__init__': <function B.__init__ at 0x00000180405A0B70>, '__dict__': <attribute '__dict__' of 'B' objects>, '__weakref__': <attribute '__weakref__' of 'B' objects>, '__doc__': None, '__y': 2}
#
# b=B('alice')
# print(b.__dict__) # {'_B__name': 'alice'}
#
# b.__age=18
# print(b.__dict__) #{'_B__name': 'alice', '__age': 18} # ---------------------------------------------
# class A:
# def foo(self):
# print('A.foo')
#
# def bar(self):
# print('A.bar')
# self.foo() #b.foo
#
# class B(A):
# def foo(self):
# print('B.foo')
#
# b=B()
# b.bar() # ---------------------------------------------
class A:
def __foo(self): #_A__foo
print('A.foo') def bar(self):
print('A.bar')
self.__foo() #self._A__foo() # 只调自己类的方法 定义时就已经确定好的! class B(A):
def __foo(self): #_B__foo
print('B.foo') b=B()
b.bar()
2.封装的意义
封装数据属性目的: (封装不是单纯意义上的隐藏)
明确的区分内外,控制外部对隐藏的属性的操作行为 封装方法属性目的:
隔离复杂度 # a=ATM() a.withdraw()
# 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,name,age): # 接口 间接的修改 设定规则
# if not isinstance(name,str):
# print('名字必须是字符串类型')
# return
# if not isinstance(age,int):
# print('年龄必须是数字类型')
# return
# self.__name=name
# self.__age=age
#
# p=People('alice','18')
# p.tell_info() # p.set_info('alex',38)
# p.tell_info() # p.set_info('alex','38')
# p.tell_info() # -------------------------------------------------------
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()
3.封装与扩展性
面向对象:可扩展性高
面向对象三大特性:继承 多态 封装
封装的扩展性:
def tell_area(self): # 对使用者来说 不用改变 方式 开发者在类里面修改
class Room:
def __init__(self,name,owner,weight,length,height):
self.name=name
self.owner=owner self.__weight=weight
self.__length=length
self.__height=height def tell_area(self): # 对使用者来说 不用改变 方式
return self.__weight * self.__length * self.__height r=Room('卫生间','alex',10,10,10)
print(r.tell_area())
4.property的使用
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
-------------------------------
property :
@property
def bmi(self): 必须有个返回值
print(p.bmi) 可以使 函数属性 伪装成 数据属性 bmi 是名词 p.bmi=23 # 不能赋值 can't set attribute bmi 实质是个方法 总结:通过计算得来的方法 可以通过@property 伪装成数据属性
@property 查看 必须有返回值
@name.setter 修改
@name.deleter 删除
# 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)
#
# p=People('egon',75,1.81)
# p.bmi=p.weight / (p.height ** 2)
# print(p.bmi) # print(p.bmi()) # 函数 是 去做什么! 是动词 对于使用者 产生误解 调了一个动词 # print(p.bmi) # 使用者可以像 访问数据属性 那样 访问 函数属性
# p.height=1.82
# print(p.bmi)
# p.bmi=23 # 不能赋值 can't set attribute bmi 实质是个方法 # ------------------------------------------------------------
class People:
def __init__(self,name):
self.__name=name @property # 查看
def name(self):
# print('getter')
return self.__name @name.setter # 修改 # 前提是 一定被装饰过 property
def name(self,val):
# print('setter',val)
if not isinstance(val,str):
print('名字必须是字符串')
return
self.__name=val @name.deleter # 删除
def name(self):
# print('deleter')
print('不允许删除') p=People('alice')
# print(p.name) # p.name='alex'
# print(p.name) del p.name
面向对象 - 1.封装之如何实现属性的隐藏/2.封装的意义/3.封装与扩展性/4.property的使用的更多相关文章
- 面向对象编程(封装、封装的意义、封装与扩展性、@property)
1.封装之如何实现属性的隐藏 封装: __x=1 # 把数据属性隐藏 (如何实现隐藏) 类定义阶段 __开头发生了变形 __x --> _A__x特点: 1.在类外部无法直接:obj.__Att ...
- python进阶01 面向对象、类、实例、属性封装、实例方法
python进阶01 面向对象.类.实例.属性封装.实例方法 一.面向对象 1.什么是对象 #一切皆对象,可以简单地将“对象”理解为“某个东西” #“对象”之所以称之为对象,是因为它具有属于它自己的“ ...
- python 3全栈开发-面向对象之绑定方法(classmethod与staticmethod的区别)、多态、封装的特性property
一.面向对象绑定方法 一.类中定义的函数分成两大类 1.绑定方法(绑定给谁,谁来调用就自动将它本身当作第一个参数传入): 1. 绑定到类的方法:用classmethod装饰器装饰的方法. 为类量身定制 ...
- Java:【面向对象:类的定义,静态变量,成员变量,构造函数,封装与私有,this】
本文内容: 什么是面对对象 类的定义与对象实例化 成员变量 成员变量的有效范围 成员变量的赋值 静态变量.方法与实例变量.方法 构造函数 封装与私有 this关键字 注:函数和方法是一样东西.[由于我 ...
- Python之路(第二十二篇) 面向对象初级:概念、类属性
一.面向对象概念 1. "面向对象(OOP)"是什么? 简单点说,“面向对象”是一种编程范式,而编程范式是按照不同的编程特点总结出来的编程方式.俗话说,条条大路通罗马,也就说我们使 ...
- 2.Python封装练习及私有属性
面向对象三大特性 封装:根据职责将属性和方法封装到一个抽象的类中 继承:实现代码的重用,相同代码不需要重复的编写 1.单继承 2.多继承 多态:不同的对象调用相同的方法,产生不同的执行结果,增加代码的 ...
- 04-OC属性的使用、自动释放池、封装和继承
目录: 一.IOS6声明式属性的使用 二.autoreleasepool自动释放池 三.封装.继承 回到顶部 一.IOS6声明式属性的使用 注:声明式属性默认情况下,并没有解决内存问题, 当使用@pr ...
- python基础----封装、封装与扩展性
从封装本身的意思去理解,封装就好像是拿来一个麻袋,把小猫,小狗,小王八,还有egon和alex一起装进麻袋,然后把麻袋封上口子.但其实这种理解相当片面 首先我们要了解 要封装什么 你钱包的有多少钱(数 ...
- python之封装与扩展性
1.封装与扩展性 封装在于明确区分内外,使得类实现者可以修改封装内的东西而不影响外部调用的代码:而外部使用者只知道一个接口(函数),只要接口(函数)名,参数不变,使用者的代码永远无需改变.这就提供了一 ...
随机推荐
- memcahced&redis命令行cmd下的操作
一.memcahced 1.安装 执行memcached.exe -d install 把memcached加入到服务中 执行memcached.exe -d uninstall 卸载memcac ...
- 报错"the microsoft.jet.oledb.4.0 provider is not registered on the local machine"解决方案
报错提示:"the microsoft.jet.oledb.4.0 provider is not registered on the local machine" 错误起因:wi ...
- 用verilog表示两个4x4矩阵的乘法运算?及单个矩阵的求逆
input[63:0] A0, //A0表示A矩阵的第一行 其中A0[63:48] A0 [47:32] A0[31:16] A0 [15:0]分别表示第一行中的四个元素(每个元素16位表示),下同i ...
- Python内置函数之int()
class int(x, base=10) 返回一个整型对象.默认返回0. 参数x可以是字符串,也可以是浮点数. base指x的进制形式,比如2表示2进制,10表示10进制.特别需要注意的是,0表示任 ...
- lua的时间和日期函数
lua的时间和日期函数 -- ::| 分类: Lua | 标签:lua 时间 函数 |举报|字号 订阅 下载LOFTER客户端 --获取当前的时间戳,单位是秒. time=os.time(); pri ...
- Powershell对象条件查询筛选
在 Windows PowerShell 中,与所需的对象数量相比,通常生成的对象数量以及要传递给管道的对象数量要多得多.可以使用 Format cmdlet 来指定要显示的特定对象的属性,但这并不能 ...
- yii2 RESTful API Develop
参考文档:http://www.yiiframework.com/doc-2.0/guide-rest.html 以 DB 中的 news 表为例创建该资源的 RESTful API,最终的测试通过工 ...
- MySQL定义异常和异常处理方法
在MySQL中.特定异常须要特定处理.这些异常可以联系到错误,以及子程序中的一般流程控制.定义异常是事先定义程序运行过程中遇到的问题,异常处理定义了在遇到问题时相应当採取的处理方式.而且保证存储过程或 ...
- Control character in cookie value or attribute
在cookie中添加中文导致静态页面打不开, (1)先清除缓存 (2)使用escape()函数对中文进行编码,获取的时候在对中文进行解码unescape(). cookie.Set("sto ...
- EasyUI combobox 加载JSON数据《一》
Action返回 JSON 格式如下: jsonResult = { total=7,rows=[ {TEXT=技术支持, ID=402894ca4419acf1014419b148a10000}, ...