反射:主要是指程序可以访问、检测和修改它本身状态或行为的一种能力(自省)
下列方法适用于类和对象,可以实现自省的函数:
1 hasattr(obj,name) #name为属性字符串,用于查找实例化对象obj中是否有name这个属性,返回bool
2 getattr(obj,name,default = None) #name为函数属性字符串(类中的函数名),用于获取到该函数的内存地址,
给他赋值加上()则可以调用函数,默认值为NONE,也可以的修改,若查找的name不存在,则返回defaul
name也可以的是数据属性,如果name存在则直接返回该属性的值
3 setattr(obj, 'name', value) is equivalent to ``x.y = v'' #给obj对象添加属性和value值
也可以的添加函数属性
4 delattr(x, 'y') is equivalent to ``del x.y'' #删除obj对象的属性
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'}
使用反射的好处
1 如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('处理其他的逻辑')
2 另外一个好处,可以动态导入模块
如m文件下有一个test.py文件,若想要把该文件当做模块导入
1 form m.test import * #使用该种导入方法,若模块文件下存在私有属性方法,则不能调用
2 mo = __import__('m.test') #使用该种导入方法,只是获取到m层面,要是想要调用模块test.py中的方法,还需要
mo.test.func() 才可使用
3 import importlib #使用该种导入方式,需要先导入importlib模块
importlib.import_moudle('m.test') #使用importlib模块下的import_moudle方法,传入导入的模块名字符串
上述方式等同于 __import__('m.test')