【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 ...
随机推荐
- springmvc注解式开发
搭建环境 后端控制器无需实现接口,添加相应的注解 springmvc配置文件中无需注册controller springmvc配置文件中添加组件扫描器.注解驱动 涉及常用的注解 @controller ...
- Ecshop 表结构 字段说明
ecs_account_log 用户帐号情况记录表,包括资金和积分等 log_id mediumint 自增ID号user_id mediumint 用户登录后保存在session中的id号,跟use ...
- HDU 3117 Fibonacci Numbers(矩阵)
Fibonacci Numbers [题目链接]Fibonacci Numbers [题目类型]矩阵 &题解: 后4位是矩阵快速幂求,前4位是用log加Fibonacci通项公式求,详见上一篇 ...
- 常用脚本lnmp
3)安装lnmp脚本只供参考需修改相应参数 #!/bin/bash#Function: Install LNMP#Author: wang#Date: 20170809 nginx_install() ...
- word之删除图标目录之间的空行
在生成图表目录时,发现Office word图表目录中多个标题之间的空行无法删除,我是自己建的标签,比如“图1-”.“图2-”…….“表1-”.“表2-”…… 发现“图1-”.“图2-”…….“表1- ...
- org/hibernate/ejb/HibernatePersistence
java.lang.NoClassDefFoundError: org/hibernate/ejb/HibernatePersistence 缺少依赖包:hibernate-entitymanager ...
- linux监控性能和网络的命令
vmstat查看机器实时的综合情况:load,内存,swap,cpu使用率等方面 procs: r:运行队列中进程数量 b:等待IO的进程数量 memory(内存): swpd:使用虚拟内存大小 fr ...
- python中的作用域以及内置函数globals()-全局变量、locals()-局部变量
在python中,函数会创建一个自己的作用域,也称为为命名空间.这意味着在函数内部访问某个变量时,函数会优先在自己的命名空间中寻找. 通过内置函数globals()返回的是python解释器能知道的变 ...
- (Review cs231n)loss function and optimization
分类器需要在识别物体变化时候具有很好的鲁棒性(robus) 线性分类器(linear classifier)理解为模板的匹配,根据数量,表达能力不足,泛化性低:理解为将图片看做在高维度区域 线性分类器 ...
- php 获取ip地址的5种方法,插入用户登录日志实例
php 获取ip地址的5种方法,插入用户登录日志实例,推荐使用第二种方法 <?php //方法1: $ip = $_SERVER["REMOTE_ADDR"]; echo $ ...