反射:主要是指程序可以访问、检测和修改它本身状态或行为的一种能力(自省)
下列方法适用于类和对象,可以实现自省的函数:
1
hasattr(obj,name)
name为属性字符串,用于查找实例化对象obj中是否有name这个属性,返回bool2
getattr(obj,name,default = None)
name
为函数属性字符串(类中的函数名),用于获取到该函数的内存地址,
给他赋值加上()则可以调用函数,若查找的name
不存在,则返回defaul的值默认的是None可以改成你想要的值。
name
也可以的是数据属性,如果name存在则直接返回该属性的值3
setattr(obj, 'name', value)
给obj对象添加属性和value值也可以的添加函数属性4
delattr(obj, 'name')
删除obj对象的name属性
class A:
num = 1
def __init__(self,name):
self.name = name
def low(self):
print(self.name)
a = A('大写字母A')
print(hasattr(a,'num')) #True
print(getattr(a,'num')) # '1'
fun = getattr(a,'low')
print(fun) #<bound method A.low of <__main__.A object at 0x00000225A2885518>>
fun() #“大写字母A”
print(getattr(a,'loow','没有这个属性')) #没有这个属性
setattr(a,'addr','中国') #给实例化对象a设置'addr'数据属性,值为'中国'
print(a.addr) # '中国'
setattr(a,'funct',lambda x:x+1) #添加一个funct函数 lambda x :x+1
print(a.funct(10)) # 11
delattr(a,'addr') #删掉了上面setatter设置的addr属性
print(a.__dict__)
#{'name': '大写字母A', 'funct': <function <lambda> at 0x000000000208C1E0>}
使用反射的好处
如A写了一个功能,但是还没有完善,部分功能不可用
class FtpClient:
'ftp客户端,但是还么有实现具体的功能'
def __init__(self,addr):
print('正在连接服务器[%s]' %addr)
self.addr=addr
#但是B需要使用该模块
#from module import FtpClient #将A写的功能模块导入
f1=FtpClient('192.168.1.1') #实例化
if hasattr(f1,'get'): #判断功能对否存在
func_get=getattr(f1,'get')
func_get() #存在则调用该功能
else:
print('---->不存在此方法') #不存在也不影响后续处理逻辑
print('处理其他的逻辑')