zoukankan      html  css  js  c++  java
  • 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')
  • 相关阅读:
    在 Java 应用程序中绑定 Bean 和数据
    JAVA的23种设计模式
    Windows 2012 Server评估版本安装问题处理
    从程序员到CTO的Java技术路线图
    MyBatis 学习
    CRT【p3868】[TJOI2009]猜数字
    线段树+扫描线【bzoj1645】[USACO07OPEN]城市的地平线City Horizon
    数学【CF743C】Vladik and fractions
    贪心【CF1029E】Tree with Small Distances
    线段树+二进制位拆分【CF242E】XOR on Segment
  • 原文地址:https://www.cnblogs.com/Meanwey/p/9788787.html
Copyright © 2011-2022 走看看