Python进阶-----类之反射

反射:主要是指程序可以访问、检测和修改它本身状态或行为的一种能力(自省)  

  下列方法适用于类和对象,可以实现自省的函数:
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')
posted @ 2018-10-14 23:34  Meanwey  阅读(369)  评论(0编辑  收藏  举报