随笔 - 139  文章 - 0 评论 - 0 阅读 - 33825
< 2025年3月 >
23 24 25 26 27 28 1
2 3 4 5 6 7 8
9 10 11 12 13 14 15
16 17 18 19 20 21 22
23 24 25 26 27 28 29
30 31 1 2 3 4 5

1、介绍

  反射的概念是由Smith在1982年首次提出的,主要是指程序可以访问、检测和修改它本身状态或行为的一种能力(自省)。这一概念的提出很快引发了计算机科学领域关于应用反射性的研究。它首先被程序语言的设计领域所采用,并在Lisp和面向对象方面取得了成绩。

  python面向对象中的反射:通过字符串的形式操作对象相关的属性。python中的一切事物都是对象(都可以使用反射)

  对类进行反射操作,执行的结果是查询或修改类的属性
  对实例化对象进行反射操作,执行的结果是查询或修改实例化对象的属性

2、四个可以实现自省的函数

hasattr(对象名,'属性名') #判断对象中是否存在该属性
getattr(对象名,'属性名', 默认值可选) #获取对象中的该属性的值,可以添加默认值
setattr(对象名,'属性名', vlaue) #设置对象中的该属性名的值为vlaue
delattr(对象名,'属性名') #删除对象中的该属性

3、例子

1:对象示例(一切皆对象,类本身也是一个对象)

复制代码
class Foo:
    def __init__(self, name, age):
        self.name = name
        self.age = age
        
    def say_hi(self):
        print('hi,%s'%self.name)
        
#检测是否含有某属性
obj = Foo('Tom', 25)
print(hasattr(obj, 'name'))
# True
print(hasattr(obj, 'say_hi'))
# True

#获取属性
name = getattr(obj, 'name')
print(name)
# Tom
print(getattr(obj, 'name1', '不存在'))
# 不存在   #如果不加默认值就会报错
func = getattr(obj, 'say_hi')
if func:
    func()
# hi, Tom

#设置属性
setattr(obj, 'sex', 'man')
print(obj.sex)
# man

#删除属性
delattr(obj, 'sex')
print(obj.__dict__)
{'name': 'Tom', 'age': 25}
delattr(obj, 'name1')
#不存在,则报错,AttributeError: name1
复制代码

 

例子2:对类的示例

复制代码
class Foo(object):
    staticField = "old boy"
 
    def __init__(self):
        self.name = 'wupeiqi'
 
    def func(self):
        return 'func'
 
    @staticmethod
    def bar():
        return 'bar'
 
print(getattr(Foo, 'staticField'))
#old boy
ret1 = getattr(Foo, 'func')
print(ret1(1))
#func 传入参数1,并执行ret1获取的方法
print(getattr(Foo, 'func')(1))
#func 
print(getattr(Foo, 'bar')())
#bar
复制代码

 

例子3:对当前模块的示例

复制代码
import sys
def s1():
    print('s1')

def s2():
    print('s2')

this_module = sys.modules[__name__]

print(hasattr(this_module, 's1'))
# True
getattr(this_module, 's2')()
# s2
复制代码

 

例子4:其他模块的示例

复制代码
#一个模块中的代码
def test():
    print('from the test')
"""
程序目录:
    module_test.py
    index.py
 
当前文件:
    index.py
"""

# 另一个模块中的代码
fs.py文件有如下代码 
n1 = 'Tom'
def func():
    print(666)

class A:
    name = 'Mike'
    def func2(self):
        print('--------IN FUNC2')

test.py文件有如下代码:
import fs
print(getattr(fs, 'n1'))
# Tom 可以从导入的fs模块中获取到n1属性
ret1 = getattr(fs, 'func')
ret1()
# 666  可以从导入的fs模块中获取func方法并执行

ret2 = getattr(fs, 'A')
print(ret2.name)
# Mike  可以从导入的fs模块中获取A类,打印A类的name属性

print(getattr(fs.A, 'name'))
# Mike 可以从导入的fs模块中的A属性中获取name属性并执行

ret3 = getattr(fs.A, 'func2')(1)
# --------IN FUNC2  获取func2方法,并传入参数1执行
复制代码

 

posted on   longfei2021  阅读(118)  评论(0编辑  收藏  举报
编辑推荐:
· AI与.NET技术实操系列:基于图像分类模型对图像进行分类
· go语言实现终端里的倒计时
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
阅读排行:
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 25岁的心里话
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· 闲置电脑爆改个人服务器(超详细) #公网映射 #Vmware虚拟网络编辑器
· ollama系列01:轻松3步本地部署deepseek,普通电脑可用
点击右上角即可分享
微信分享提示