【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 ...
随机推荐
- soapUI学习文档(转载)
soapUI 学习文档不是前言的前言记得一个搞开发的同事突然跑来叫能不能做个WebService 性能测试,当时我就凌乱了,不淡定啊,因为我是做测试的,以前连WebService 是什么不知道,毕竟咱 ...
- Flex中如何利用FocusManager类的setFocus函数设置TextInput的焦点的例子
参考:https://blog.csdn.net/liruizhuang/article/details/5876455 <?xml version="1.0" encodi ...
- phpstudy的使用
1.第一步是下载phpstudy,你可以百度去下载,也可以通过下面我分享的网盘下载 链接:https://pan.baidu.com/s/1E_CXIrKv1N-jrlA4KCovZA 密码:mkx9 ...
- BCB Access violateion at Address 0000 0003. Read of address 0000 0003
来自网页:(我的电脑做不到) 运行一个程序,莫名出现一个对话框:access violation at address 0000.. read of address000试了几次问题依旧,网上搜了下解 ...
- equals和==的区别小结
==: == 比较的是变量(栈)内存中存放的对象的(堆)内存地址,用来判断两个对象的地址是否相同,即是否是指相同一个对象.比较的是真正意义上的指针操作. 1.比较的是操作符两端的操作数是否是同一个对象 ...
- DataGridView控件用法一:数据绑定
使用DataGridView控件,可以显示和编辑来自多种不同类型的数据源的表格数据. 将数据绑定到DataGridView控件非常简单和直观,在大多数情况下,只需设置DataSource属性即可.在绑 ...
- Beta冲刺阶段5.0
1. 提供当天站立式会议照片一张 2. 每个人的工作 (有work item 的ID) 成员 昨天已完成的工作 今天计划完成的工作 工作中遇到的困难 具体贡献 郑晓丽 首页活动详情界面的美化 实现首页 ...
- 学习笔记<1>技术体系结构
Android的系统架构采用了分层架构的思想,如上图所示.从上层到底层共包括四层,分别是 1.应用程序程序层 2.应用框架层 3.系统库和Android运行时 4.Linux内核. 每 ...
- IE浏览器解决无法识别js中getElementsByClassName问题
关于ie浏览器无法识别js中getElementsByClassName问题,现通过以下方法,引用如下js /** *打印js对象详细信息 */ function alertObj(obj) { va ...
- html5-新增表单的小结details summary
<!DOCTYPE html><html lang="en"><head> <meta charset="UTF-8&qu ...