【Python045-魔法方法:属性访问】
一、属性的几种访问方式
1、类.属性名
>>> class C:
def __init__(self):
self.x = 'X-man' >>> c = C()
>>> c.x
'X-man'
2、用内置函数getattr()访问属性
>>> getattr(c,'x','莫有这个属性')
'X-man'
>>> getattr(c,'y','莫有这个属性')
'莫有这个属性'
>>>
3、用property方法访问属性
class C:
def __init__(self,size=10):
self.size=size def getsize(self):
return self.size def setsize(self,value):
self.size = value def delsize(self):
del self.size x=property(getsize,setsize,delsize) 执行结果:
>>> c = C()
>>> c.x
10
>>> c.x=1
>>> c
<__main__.C object at 0x10c012f60>
>>> c.x
1
>>> del c.x
>>> c.x
Traceback (most recent call last):
File "<pyshell#19>", line 1, in <module>
c.x
File "/Users/wufq/Desktop/property.py", line 6, in getsize
return self.size
AttributeError: 'C' object has no attribute 'size'
4、各类内置函数访问属性
* __getattr__(self,name)
定义当用户试图获取一个不存在的属性时的行为
* __getattribute__(self,name)
定义当该类的属性被访问时的行为
* __setattr__(self,name,value)
定义当一个属性被设置时的行为
* __delattr__(self,name)
定义当一个属性被删除时的行为
class C:
def __getattr__(self,nane):
print("__getattr__") def __getattribute__(self,name):
print("__getattribute__")
return super().__getattribute__(name) def __setattr__(self,name,value):
print("__setattr__")
super().__setattr__(name,value) def __delattr__(self,name):
print("__delattr__")
super().__delattr__(name) >>> c = C()
>>> c.x
__getattribute__
__getattr__
>>> c.x =
__setattr__
>>> c.x
__getattribute__ >>> del c.x
__delattr__
>>> c.x '''
代码解析:
|--1、由第一个c.x先后打印出来__getattribute__和__getattr__,可以看出来,程序先找__getattribute__(定义当该类的属性被访问时的行为),然后
|-- 在找__getattr__(属性不存在时的访问行为)
|--2、由c.x =1 打印出来__setattr__,可以得知,当属性被设置时调用的是此方法
|--3、由del c.x=1 打印出来__delattr__可以得知,当属性被删除时调用此方法
'''
5、写一个矩形类,默认有宽和高两个属性
如果为一个叫square的属性赋值,那么说明这是一个正方形,值就是正方形的边长,此时宽和高都应该等于边长
class Rectangle:
def __init__(self,width=0,height=0):
self.width = width
self.height = height
#用square这个属性来标记正方形
def __setattr__(self,name,):
if name == 'square':
self.width = value
self.height = value
else:
self.name = value def getArea(self):
return self.width * self.height >>> r1 = Rectangle(4,5)
Traceback (most recent call last):
File "<pyshell#15>", line 1, in <module>
r1 = Rectangle(4,5)
File "/Users/wufq/Desktop/正方形.py", line 4, in __init__
self.width = width
File "/Users/wufq/Desktop/正方形.py", line 12, in __setattr__
self.name = value
File "/Users/wufq/Desktop/正方形.py", line 12, in __setattr__
self.name = value
File "/Users/wufq/Desktop/正方形.py", line 12, in __setattr__
self.name = value
[Previous line repeated 987 more times]
File "/Users/wufq/Desktop/正方形.py", line 8, in __setattr__
if name == 'square':
RecursionError: maximum recursion depth exceeded in comparison '''
代码解析:
|-- 为什么会报错:原因:首先实例化对象时给width和height赋值,赋值时就调用了__setattr__方法
|-- 进入__setattr__方法,会首先判断name有没有等于square,明显此时没有,则执行else里面的语句
|-- self.name = value,则又一次循环调用默认的__init__方法,依次这样就会进入死循环
'''
改进的方法
#第一种方法:采用super()方法调用基类
#第二种方法:采用字典的方法,__dict__ class Rectangle:
def __init__(self,width=0,height=0):
self.width = width
self.height = height
#用square这个属性来标记正方形
def __setattr__(self,name,value):
if name == 'square':
self.width = value
self.height = value
else:
#self.name = value
#super().__setattr__(name,value)
self.__dict__[name]=value #采用__dict__方法的原因是:给属性square所赋值存储到dict字典内 def getArea(self):
return self.width * self.height
执行结果:
>>> r1 = Rectangle(4,5)
>>> r1.getArea()
20
>>> r1.square = 10
>>> r1.getArea()
100
>>> r1.__dict__
{'width': 10, 'height': 10}
二、练习题
1、对象的属性发生赋值操作时,将实际的的值+1赋值给相应的属性。
def __setattr__(self,name,value):
self.name = value + 1 错误原因:属性被赋值时,__setattr__()会被调用,而里边的self.name = value +1又会再次触发__setattr__()调用,导致无限递归(死循环)
正确的写法
class C:
def __init__(self,name):
self.name = name
def __setattr__(self,name,value): self.__dict__[name]=value+1
print("name=",self.__dict__[name],"value=",value) 执行结果:
>>> c = C(2)
name= 3 value= 2
2、自定义该类的属性被访问的行为,你应该重写那个魔法方法
答:__getattribute__(self,name) :属性被访问时调用这个方法
3、当访问一个不存在的属性时,不报错且提示“该属性不存在”
class Demo:
def __getattr__(self,name):
return "该属性不存在" 执行结果:
>>> demo = Demo()
>>> demo.x
'该属性不存在'
>>>
4、编写一个Demo类,使的下边代码可以正常执行
class Demo:
def __getattr__(self,name):
self.name = 'FishC'
return self.name
def __getattribute__(self,name):
return super().__getattribute__(name)
def __setattr__(self,name,value):
return super().__setattr__(name,value) 执行结果:
>>> d=Demo()
>>> d.x
'FishC'
>>> d.x = "X-man"
>>> d.x
'X-man'
>>>
5、编写一个Counter类,用于实时检查对象有多少个属性
#需求:检测对象有多少个属性
class Counter:
def __init__(self):
super().__setattr__('counter',0)
def __setattr__(self,name,value):
super().__setattr__('counter',self.counter + 1)
super().__setattr__(name,value)
def __delattr__(self,name):
super().__setattr__('counter',self.counter - 1)
super().__delattr__(name) 执行结果:
>>> c = Counter()
>>> c.x=1
>>> c.counter
1
>>> c.y =2
>>> c.z = 3
>>> c.m =4
>>> c.counter
4
>>> del c.m
>>> c.counter
3
>>>
【Python045-魔法方法:属性访问】的更多相关文章
- python魔法方法-属性访问控制
属性访问控制 所谓的属性访问控制就是控制点号访问属性的行为,而且不仅是类的外部,连类的内部也受控制,代码见真章,边看代码边解释: __getattr__(self, item) 定义当访问不存在的属性 ...
- Python魔法方法之属性访问 ( __getattr__, __getattribute__, __setattr__, __delattr__ )
通常情况下,我们在访问类或者实例对象的时候,会牵扯到一些属性访问的魔法方法,主要包括: ① __getattr__(self, name): 访问不存在的属性时调用 ② __getattribute_ ...
- python_魔法方法(四):属性访问
通常可以通过点(.)操作符的形式去访问对象的属性,也可以通过BIF适当地去访问属性,看个例子吧 >>> class A(): def __init__(self): self.x = ...
- Python类属性访问的魔法方法
Python类属性访问的魔法方法: 1. __getattr__(self, name)- 定义当用户试图获取一个不存在的属性时的行为 2. __getattribute__(self, name)- ...
- 《Python基础教程(第二版)》学习笔记 -> 第九章 魔法方法、属性和迭代器
准备工作 >>> class NewStyle(object): more_code_here >>> class OldStyle: more_code_here ...
- python学习笔记之七:魔法方法,属性
在python中,有的名称会在前面和后面加上两个下划线,由这些名字组成的集合所包含的方法称为魔法方法(或者是特殊方法).如果对象实现了这些方法中的某一个,那么这个方法会在特殊的情况下(确切地说是根据名 ...
- Python笔记(二十六)_魔法方法_属性的魔法方法
属性的魔法方法 __getattribute__(self,name):当该类的属性被访问时,自动触发,是最先被触发的属性方法 __setattr__(self,name,value):当一个属性被设 ...
- 【python学习笔记】9.魔法方法、属性和迭代器
[python学习笔记]9.魔法方法.属性和迭代器 魔法方法:xx, 收尾各有两个下划线的方法 __init__(self): 构造方法,创建对象时候自动执行,可以为其增加参数, 父类构造方法不会被自 ...
- Solidity属性和方法的访问权限
属性:默认是internal的类型,外部是不可以访问调用的,如果加上public的话,那么是会自动为这个属性加上一个get的方法的,比如uint public _age; => functi ...
随机推荐
- OEMCC 13.2 集群版本安装部署
之前测试部署过OEMCC 13.2单机,具体可参考之前随笔: OEMCC 13.2 安装部署 当时环境:两台主机,系统RHEL 6.5,分别部署OMS和OMR: OMS,也就是OEMCC的服务端 IP ...
- SQL 2008安装过程(转)
这几天因为需要,一直想安装SQL Server 2008来作为Web后台的数据库进行些实验,但总是没有时间,今天终于有时间了,便安装了SQL Server 2008,以下是我的安装的步骤,希望对于有需 ...
- vue中使用elementui里的table时不被选中设置
情景:例如提现列表,转账失败后转账金额直接返回用户余额,所以当前数据不可以再次操作 直接粘贴代码: <el-table-column type="selection" wid ...
- vue中强制刷新的bug处理
vue是单页面应用,跳转路由也是局部刷新,这里就拿后台管理系统而言,如果你的后台管理系统是左右布局,你不会遇到这样的问题,但是如果你的后台管理系统是上左右布局,你就会遇到这个问题,一级菜单在最上面,二 ...
- jQuery-图片放大镜
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...
- Android Studio -- 关联源码
1,昨天刚把SDK升级到25,然后准备开始 新的一年码代码,结果发现查看源码的时候出现了一堆的“ throw new RuntimeException("Stub!");” 网上搜 ...
- 2017-2018-1 20155228 《信息安全系统设计基础》第六周学习总结&课下作业
20155228 2017-2018-1 <信息安全系统设计基础>第六周学习总结&课下作业 教材学习内容总结 异常及其种类 异常可以分为四类:中断(interrupt) ,陷阱(t ...
- Runtime单例模式类 -- 控制电脑关机
package demo1; import java.io.IOException; public class RunTimeDemo { public static void main(String ...
- java中的神奇"this"
java中的神奇"this",神奇的原因事它能不用new就可以直接创造一个对象出来,后来研究发现,其实java的“this”使用时,也是"new"了一个当前的对 ...
- PCB 布线,直角线,差分线,蛇形线
1.直角线 直角走线的一般标准是PCB布线中要尽量避免的情况,也几乎成为衡量布线好坏的标准之一. 直角走线对信号的影响主要体系那在下面三个方面 1.保教可以等效为传输线是哪个的容性负载,减缓上升时间. ...