Python中的反射
admin
2023-07-13 03:02:38
0

什么是反射

反射主要是指程序可以访问、检测和修改它本身状态或行为的一种能力(自省)。
python面向对象中的反射就是通过字符串获取对象或者类的属性,进行操作~,主要是对这4个方法的应用:hasattr,getattr,setattr,delattr。

反射对象的属性和方法

class Person:
    def __init__(self, name, age):
        self.__name = name
        self.__age = age

    def __fun(self):
        print(self.__class__)

    def say(self):
        print(self.__name + ' ' + str(self.__age))

# 判断属性是否存在
p = Person('baby', 18)
print(hasattr(p, 'name'))                    # False
print(hasattr(p, '_Person__name'))   # True
print(hasattr(p, 'say'))                       # True

# 获取属性
fun = getattr(p, 'say')
fun()                # baby 18,执行反射获取的方法
name = getattr(p, '_Person__name')
print(name)    # baby
# 若是属性不存在则报错
# age = getattr(p, 'age')   # 'Person' object has no attribute 'age'

# 设置属性
setattr(p, 'sex', 'male')       # 设置的是对象的属性,存放在对象的名称空间中
# 这里设置的方法是普通方法,存放在对象的名称空间中,self.__name不会变形为 self._Person__name
# setattr(p, 'show_name', lambda self: self.__name)   
setattr(p, 'say_hello', lambda self: 'Hello ' + self._Person__name)
print(p.__dict__)         
# {'_Person__name': 'baby', '_Person__age': 18, 'sex': 'male', 'say_hello':  at 0x10f7bf2f0>}
print(p.say_hello(p))    # 不是绑定方法,需要手动传值

# 删除属性
delattr(p, 'sex')
print(p.__dict__)
# {'_Person__name': 'baby', '_Person__age': 18, 'say_hello':  at 0x10f7bf2f0>}
# 若不存在该属性则报错
# delattr(p, 'name')     # AttributeError: name

Tip:

  • 通过对象设置的属性,不管是变量还是方法,都存放在对象的名称空间中;
  • 通过对象设置的方法仅仅是普通方法,调用的时候也不会自动传值(需要手动传值),且方法中不能使用对象的私有属性

反射类的属性和方法

class Person:
    def __init__(self, name, age):
        self.__name = name
        self.__age = age
        self.city = 'NB'

    def __fun(self):
        print(self.__class__)

    def say(self):
        print(self.__name + ' ' + str(self.__age))

    @classmethod
    def play(cls):
        print(cls.__name__)

    @staticmethod
    def sleep():
        print('sleep...')

# 判断属性是否存在
print(hasattr(Person, 'name'))                    # False
print(hasattr(Person, '_Person__name'))   # False,私有属性 __name,__age 属于对象
print(hasattr(Person, 'say'))                       # True

# 获取属性
fun = getattr(Person, 'say')
p = Person('baby', 18)
fun(p)          # baby 18,等同于Person.say(p),需要手动传递self
# 若是属性不存在
# age = getattr(Person, 'age')   # 报错,'Person' object has no attribute 'age'

# 设置属性
setattr(Person, 'sex', 'male')    # 设置的是类的静态属性
setattr(Person, 'show_city', lambda self: self.city)   # 这里通过类设置的方法为绑定到对象的方法,通过对象调用的时候能够自动传值(self)
print(p.show_city())    # NB
# setattr(Person, 'show_name', lambda self: self.__name)   # self.__name 不会自动转换为 self._Person__name
# print(p.show_name())    # AttributeError: 'Person' object has no attribute '__name'

# 删除属性
delattr(Person, 'sex')   # 删除的是类的静态属性

# 获取类方法
getattr(Person, 'play')()   # 会完成自动传值,默认将Person作为第一个参数传递给play方法

# 获取静态方法
getattr(Person, 'sleep')()

Tip:

  • 通过类设置的属性,存放在类的名称空间中;
  • 通过类设置的方法为绑定到对象的方法,通过对象调用的时候,能够完成自动传值;同样方法内无法访问到私有变量(self.__属性 不会自动转换为self._类名__属性)

反射当前模块成员

import sys

def s1():
    print('s1')

def s2():
    print('s2')

this_module = sys.modules[__name__]

# 判断模块中是否存在 s1 方法
print(hasattr(this_module, 's1'))   # True

# 获取模块中的方法并执行
getattr(this_module, 's2')()           # s2

sys.modules[__name__] 也可以写成 sys.modules['__main__'],但是不建议这么写,因为当前的模块被导入到另外一个模块的时候,这个被导入的模块使用 sys.modules['__main__'] 就获取不到它的内存地址了~
 
操作的对象也可以是导入的模块

# module_test

def test():
    print('from test')

# test.py

import module_test

print(hasattr(module_test,'test'))   # True
getattr(module_test, 'test')()          # from test

isinstance 和 issubclass

isinstance 方法用来判断 一个对象 和 一个类之间的关系,即这个对象是不是由这个类实例化而来

class Person:
    pass

p = Person()
print(isinstance(p, Person))    # True

 
issubclass 用来判断两个类之间是否存在继承关系

class Father:
    pass

class Son(Father):
    pass

print(issubclass(Son, Father))    # True

.................^_^

相关内容

热门资讯

问题居然在实体卡槽上!美版iP... 6月2日消息,日前,又有博主提前把还没发布的iPhone 18 Pro电池参数给曝光了出来,根据爆料...
2026年618手机购机攻略:... 每年618都是手机换新的最佳窗口期,2026年大促叠加数码国补、平台满减、品牌降价三重福利,新机老机...
8.99万打穿底价!史无前例的... 科技的浪漫,不是将人类送入遥不可及的星辰,而是把曾经高不可攀的科幻,变成柴米油盐里的“论斤卖”。 如...
苹果首款折叠屏iPhone U... PChome 6月2日消息,据知名数码博主@i冰宇宙最新曝光的机模谍照,苹果首款折叠屏手机(内部命名...
特朗普:美伊暂停谈判?假新闻! 当地时间6月2日,美国总统特朗普否认伊朗已停止与调解方沟通,称美国与伊朗之间的对话一直在持续进行。特...
白宫记协晚宴改期至7月,特朗普... 当地时间6月2日,美国总统特朗普表示,此前因突发暴力事件而中断的白宫记协晚宴已确定改期至7月24日举...
特朗普,开掉“不听话的情报头子... 苦撑15个月后,美国国家情报总监加巴德即将离开白宫。她对辞职的解释是:丈夫近期确诊罕见骨癌,自己希望...
面粉染头模仿黄仁勋,农村青年走... “先赔偿5000元肖像侵权使用费,再删除所有视频,要么就起诉你。”这是黄仁勋的模仿者杨洋最近频繁收到...
驻马店非遗“大集”开张 把网络... 顶端新闻记者 王丹/文 李思翰 胡楚昊/图红纸剪出“抵制网络谣言”,糖画写下“不信谣不传谣”……5月...
A股三大指数集体收涨 贵金属涨...   A股三大指数集体收涨 贵金属涨幅居前  【A股三大指数集体收涨 贵金属涨幅居前】6月2日,A股三...