Python的反射机制

  Python的反射机制,就是反射就是通过字符串的形式,导入模块;通过字符串的形式,去模块寻找指定函数,并执行。利用字符串的形式去对象(模块)中操作(查找/获取/删除/添加)成员,一种基于字符串的事件驱动!

  常用的反射内置函数有四个:hasattr(),getattr(),setattr()和delattr()。我们来看看具体应该怎么使用。

首先定义一个类并实例化,再定义个字符串变量从键盘输入。

 def cry(self):
print("%s is crying!"%self.name)
class Person(object):
def __init__(self,name):
self.name = name
def eat(self):
print("%s is eating "%(self.name))
def talk(self):
print(("%s is talking"%self.name))
p1 = Person("Jack")
action = input(">>>:")

要想用输入的字符串来调用实例中的属性或功能,用p1.action是不行的

>>>:eat
Traceback (most recent call last):
File "D:/python/week7/反射.py", line 14, in <module>
p1.action
AttributeError: 'Person' object has no attribute 'action'

运行结论

最简单暴力的方法有一种:

class Person(object):
def __init__(self,name):
self.name = name
def eat(self):
print("%s is eating "%(self.name))
def talk(self):
print(("%s is talking"%self.name))
p1 = Person("Jack")
action = input(">>>:")
if action == 'eat':
p1.eat()
elif action == 'talk':
p1.talk()

可是如果类里定义了上百中功能,靠if来比较输入字符串显然是不靠谱的,也太low了。这时候,就用上反射了。

 class Person(object):
def __init__(self,name):
self.name = name
def eat(self):
print("%s is eating "%(self.name))
def talk(self):
print(("%s is talking"%self.name))
p1 = Person("Jack")
action = input(">>>:")
if hasattr(p1,action): #hasattr(obj,func_str):判定实例内是否有名字为str的功能或属性
print(getattr(p1,action)) #gatattr(obj,func_str):返回实例内名字为str的功能的地址或属性的值
                     如果action = name,运行结果为Jack,如果action=eat,则返回值为eat()在内存中的地址

通过hasattr()函数来判定实例内是否包含输入名字和字符串一致的属性或功能,返回值为Ture 或False 就可以和if等指令配合使用。

注意getattr()的返回值,如果输入字符串的是属性名字,则返回值为属性的值,而输入的是功能的名字,返回值就是地址,换句话说在地址后加(),并在括号内加上实参,就能运行函数。

 class Person(object):
def __init__(self,name):
self.name = name
def eat(self,food): #这里加了个形参
print("%s is eating %s"%(self.name,food))
def talk(self):
print(("%s is talking"%self.name))
p1 = Person("Jack")
action = input(">>>:")
if hasattr(p1,action):
done = getattr(p1,action)
done("cake") #在这里给形参food传递实参
>>>:eat
Jack is eating cake

运行结果

setattr()用来对实例的属性或功能进行更改或添加。

 def cry(self):          #定义了一个类里没有的函数
print("%s is crying!"%self.name)
class Person(object):
def __init__(self,name):
self.name = name
def eat(self,food):
print("%s is eating %s"%(self.name,food))
def talk(self):
print(("%s is talking"%self.name))
p1 = Person("Jack")
action = input(">>>:")
if hasattr(p1,action):
pass
else:
setattr(p1,action,cry) #setattr(obj,a,b)把名为b的功能添加在obj中,新功能/属性名为a
done = getattr(p1,action)
done(p1) #这里比把p1作为实参传递给新添加的功能
>>>:cry
Jack is crying!

运行结果

在上例中输入字符串为“cry”,实例中没有,就通过setattr()增加新的功能。然后用getattr()调用新添加的功能,这里一定要把实例作为实参传给新添加的功能(因为定义的cry需要调用实例里的属性)

setattr()还可用作更改实例的属性,如果属性内不含输入的字符串,则增加新的属性。

 setattr(p1,action,"Lucy")
print(p1.name)    #在输入name后,p1的name就编程Lucy了。

最后,delattr()用来删除实例中的属性或功能具体用法就是delattr(obj,str)删除boj内名为str的功能或属性。

Python 反射机制的更多相关文章

  1. Python反射机制理解

    Python反射机制用沛齐老师总结的话说就是:利用字符串的形式去对象(模块)中操作(寻找)成员. getattr(object, name) object代表模块,name代表模块中的属性或成员,该函 ...

  2. python反射机制深入分析

    对编程语言比较熟悉的朋友,应该知道“反射”这个机制.Python作为一门动态语言,当然不会缺少这一重要功能.然而,在网络上却很少见到有详细或者深刻的剖析论文.下面结合一个web路由的实例来阐述pyth ...

  3. python 反射机制在实际的应用场景讲解

    剖析python语言中 "反射" 机制的本质和实际应用场景一. 前言 def s1(): print("s1是这个函数的名字!") s = "s1&q ...

  4. Python 反射机制之hasattr()、getattr()、setattr() 、delattr()函数

    反射机制 先看看我对Java中反射机制的通俗理解:反射之中包含了一个“反”的概念,所以要想解释反射就必须先从“正”开始解释,一般而言,当用户使用一个类的时候,应该先知道这个类,而后通过这个类产生实例化 ...

  5. python: 反射机制;

    import comma def run(): inp = input('请输入要调用的函数').strip(); if hasattr(comma,inp): fun = getattr(comma ...

  6. 【Python】python 反射机制在实际的应用场景讲解

    剖析python语言中 "反射" 机制的本质和实际应用场景一. 前言 def s1(): print("s1是这个函数的名字!") s = "s1&q ...

  7. Python面试题之Python反射机制

    0x00 前言 def f1(): print('f1') def f2(): print('f2') def f3(): print('f3') def f4(): print('f4') a = ...

  8. python 反射机制 ( 广泛应用于URL参数)

    web实例 考虑有这么一个场景,根据用户输入的url的不同,调用不同的函数,实现不同的操作,也就是一个url路由器的功能,这在web框架里是核心部件之一.下面有一个精简版的示例: 首先,有一个comm ...

  9. Python ————反射机制

    python中的反射功能是由以下四个内置函数提供:hasattr.getattr.setattr.delattr,改四个函数分别用于对对象内部执行:检查是否含有某成员.获取成员.设置成员.删除成员. ...

随机推荐

  1. ubuntu18换国内源

    编辑/etc/apt/sources.list文件, 在文件最前面添加以下条目(操作前请做好相应备份): ##中科大源 deb https://mirrors.ustc.edu.cn/ubuntu/ ...

  2. metasploit framework(一):基本使用

    它位于/usr/share/metasploit-framework 进入到modules目录,有六大模块 exploits:系统漏洞利用的流程,对系统漏洞注入一些特定的代码,使其覆盖程序执行寄存器, ...

  3. JMeter一次简单的接口测试(转载)

    转载自 http://www.cnblogs.com/yangxia-test 本次接口测试:根据ws查询所有商品的具体的信息.检查商品是否返回成功. 1.  准备测试数据 查询数据库中产品表已上架商 ...

  4. 【Scheme】符号求导

    思路: 定义一个求导算法, 令其在抽象对象上执行求导操作. 可以由以下规约规则完成: dc/dx=0 dx/dx=1 d(u+v)/dx=du/dx+dv/dx d(uv)/dx=u(dv/dx)+v ...

  5. 基于xtrabackup GDIT方式不锁库作主从同步(主主同步同理,反向及可)

    1.安装数据同步工具 注:xtrabackup 在数据恢复的时候比mysqldump要快很多,特别是大数据库的时候,但网络传输的内容要多,压缩需要占用时间. yum install https://w ...

  6. POJ-3414.Pots.(BFS + 路径打印)

    这道题做了很长时间,一开始上课的时候手写代码,所以想到了很多细节,但是创客手打代码的时候由于疏忽又未将pair赋初值,导致一直输出错误,以后自己写代码可以专心一点,可能会在宿舍图书馆或者Myhome, ...

  7. 面向对象中的property装饰器讲解

    面向对象中可以用property来修饰我们的函数,必须下面的例子 class Test(object): def __init__(self,name): self.name = name @prop ...

  8. WorkerMan源码分析(resetStd方法,PHP中STDIN, STDOUT, STDERR的重定向)

    WorkerMan中work.php中 resetStd 方法中代码如下 public static function resetStd() { if (!static::$daemonize || ...

  9. jsplumb踩坑

    一,,关于连线器label 我们全局设置中可以用 getInstance 实例化新对象 也可以通过 importDefaults 实例化新对象 当我想要给连线器上添加标签  每个连线器上的标签都不一样 ...

  10. [Z]haproxy+keepalived高可用群集

    http://blog.51cto.com/13555423/2067131 Haproxy是目前比较流行的一种集群调度工具Haproxy 与LVS.Nginx的比较LVS性能最好,但是搭建相对复杂N ...