一、反射
 python 中用字符串的方式操作对象的相关属性,python 中一切皆对象,都可以使用反射
 用eval  有安全隐患,用 反射就很安全
 1.反射对象中的属性和方法



class A:
    a_class = '反射类静态属性'
    @classmethod
    def func_class(cls):
        return '反射类方法'
    def __init__(self, name):
        self.name = name
        print('%s是对象属性' % (self.name))
    def func(self):
        return '反射对象方法'



2、getatter   获取
1. 反射对象中的属性和方法
a = A('zzy')    # zzy是对象属性   实例化的时候会触发init,打印zzy是对象属性



原版:a.name
ret = getattr(a, 'name')   #
print(ret)      # zzy




PYTHON RE flag python re flags_python

PYTHON RE flag python re flags_类方法_02

原版:a.func()
ret2 = getattr(a, 'func')
print(ret2)     #<bound method A.func of <__main__.A object at 0x01B5EBB0>>打印的是函数地址,并没有调用
ret2()     #加括号就表示调用,打印‘对象方法’
简洁版
print(getattr(a, 'name'))      #zzy是对象属性,zzy
print(getattr(a,'func')())     #反射对象方法


例二


2.反射类的静态属性和方法



#原版:A.func_class()
print(getattr(A,'func_class')())     #zzy是对象属性
print(getattr(a,'func_class')())    #反射类方法---->>用对象调用类方法也可以,但是不推荐,
#原版:A.a_class
print(getattr(A,'a_class'))    #反射类静态属性



"""
注释:getattr (对象/类,‘属性/方法’)    属性/方法必须是字符串形式
getattr 反射方法的时候,要在末尾加(),才能算是调用,否则打印的只是函数地址
"""



要反射的函数有参数怎么办?



# print(time.strftime('%Y-%m-%d %H:%M:S'))
# print(getattr(time,'strftime')('%Y-%m-%d %H:%M:S'))



3、hasattr 有



#判断需要反射的对象是否存在,和getattr是夫妻档,判断存在的时候,才会执行getattr
if hasattr(a,'name'):
    print(getattr(a,'name'))         #   zzy是对象属性   zzy



4、setattr 设置修改变量




PYTHON RE flag python re flags_python

PYTHON RE flag python re flags_类方法_02

class A:
    pass
a = A()
setattr(a,'name','nezha')      #在对象a中新建属性name属性,并且赋值'nezha'
setattr(A,'name','alex')        #在A类中新建静态属性name属性,并且赋值'alex'
print(A.name)   #alex
print(a.name)   #'nezha


setattr


5、delattr 删除一个变量




PYTHON RE flag python re flags_python

PYTHON RE flag python re flags_类方法_02

delattr(a,'name')     #删除对象a中新建属性name属性
print(a.name)       #删除后打印会报错
delattr(A,'name')   #删除在A类中新建静态属性name属性
print(a.name)     #删除后打印会报错


delattr


6、进阶使用1,涉及用户交互



变量名=input('请输入:')
print(getattr(a,变量名))   #zzy



练习,需求:用户输入time 和 asctime打印出time.time/asctime




PYTHON RE flag python re flags_python

PYTHON RE flag python re flags_类方法_02

import time
t=input('请输入:')   #输入time / asctime
print(getattr(time,t)())    #  因为输入的是time / asctime,而
#注释:说明内置函数也是可以反射的


练习


7、进阶使用2,涉及其他导入模块的反射



(1)反射其他模块



import sys
import model
b=model.B('zzy')
#反射模块的类属性
原版:print(b.name)
print(getattr(b,'name'))   #zzy
#反射模块类方法
原版: b.func1()
print(getattr(b,'func1')())    #你好



(2)反射模块中变量



import model



#原版:model.soga       #   model模块 中有soga=input('请输入:')

print(getattr(model,'soga')) #打印你输入的内容 #反射模块中的函数
# 原版model.foo()
print(getattr(model,'foo')('zxc')) #zxc 函数中有参数的话,就可以在括号里('zxc')传进去



(3)反射模块中的函数



# 原版model.foo()
# print(getattr(model,'foo')('zxc'))    #zxc  函数中有参数的话,就可以在括号里('zxc')传进去



8、进阶3:反射自己模块



(1)导入自己模块的变量



msg=1+2print(getattr(sys.modules['__main__'],'msg')



(2)导入自己模块的函数



def foo (a):
    return a
print(getattr(sys.modules[__name__],'foo')('zxc'))        #zxc



(3)反射自己模块的类




PYTHON RE flag python re flags_python

PYTHON RE flag python re flags_类方法_02

class C:
    a_class = '反射模块类静态属性'
    @classmethod
    def func_class(cls):
        return '反射模块类方法'
    def __init__(self, name):
        self.name = name
    @property
    def func_c(self):
        return '反射模块对象方法'
    @staticmethod
    def func():
        return '我是一个普通的函数'


类函数



PYTHON RE flag python re flags_python

PYTHON RE flag python re flags_类方法_02

import sys
c=C('ZZY')
print(getattr(sys.modules['__main__'],'C'))     #<class '__main__.C'>  类的函数地址
print(getattr(sys.modules['__main__'],'c'))     #<__main__.C object at 0x01B77210>   对象的函数地址
print(getattr(sys.modules['__main__'],'c').name)     #ZZY    对象.name  就可以调用自己模块中类的方法名了
print(getattr(sys.modules['__main__'],'C').func_class())    #反射模块类方法
print(getattr(sys.modules['__main__'],'c').func_c)     #反射模块对象方法
print(getattr(sys.modules['__main__'],'C').func())    #我是一个普通的函数
#注释:sys.modules['__main__']找到当前运行的模块
#最好用sys.modules['__name__'],因为__name__ 永远获取的是__name__所在的模块


反射


9、比较区别
先在一个reflect.py文件中打印
print(sys.modules[__name__])
print(sys.modules['__main__'])
再在另一个model.py文件在中执行下列



import reflect      #导入reflect模块了,就会执行reflect模块中的程序

#reflect  中运行了print(sys.modules[__name__])
就会打印:<module 'reflect' from 'D:\\Users\\ZhaoYun Zhu\\PycharmProjects\\reflect.py'>

# reflect  中运行了print(sys.modules[__main__])
就会打印:<module '__main__' from 'D:/Users/ZhaoYun Zhu/PycharmProjects//model.py'>

# 总结:sys.modules['__main__']找到当前运行的模块
# 最好用sys.modules['__name__'],因为__name__ 永远获取的是__name__所在的模块