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. spingboot @EnableScheduling

    springboot让开发更简单!springmvc中启用定时任务还得需要在xml中进行配置启用并且要配置扫描器,但是springboot只需要一个注解就可以. @EnableScheduling 无 ...

  2. day26 面向对象的常用方法 和 元类的使用

    1. 面向对象中的常用方法 ***** isinstance() # 判断某个对象是不是某个类的实例 # 判断stu对象是不是Student类的实例 是不是子类 issubclass() class ...

  3. Redis集群部署及命令

    一.简介 redis集群是一个无中心的分布式Redis存储架构,可以在多个节点之间进行数据共享,解决了Redis高可用.可扩展等问题. redis集群提供了以下两个好处: 将数据自动切分(split) ...

  4. cloudstack4.11+KVM+4网卡bond5+briage 交换机不作配置

    网卡绑定配置 # cat ifcfg-em1TYPE=EthernetBOOTPROTO=noneDEVICE=em1ONBOOT=yesMASTER=bond0SLAVE=yes# cat ifcf ...

  5. c++ - Linking problems due to symbols with abi::cxx11?

    看错误内容: /data/projects/LipReadingSDKGPU/lib/cwlibs/libLipReading.so: undefined reference to `tensorfl ...

  6. LinearLayout 线性布局

    android:orientation 设置布局管理器内组件的排列方式,可设置为 horizontal (水平排列).vertical (垂直排列) android:gravity 设置布局管理器内组 ...

  7. Mobile Game Development with Unity Build Once, Deploy Anywhere

    本书从自上而下的角度介绍了Unity游戏引擎的功能,并提供了具体的.面向项目的指导,说明了如何在真实的游戏场景中使用这些功能,以及如何从头开始构建让玩家爱不释手的2D和3D游戏.主要内容有:探索Uni ...

  8. vue-app项目,将px自动转化为rem

    1. 安装lib-flexible: npm install --save lib-flexible 2.安装postcss-loader和postcss-px2rem: npm install -- ...

  9. layer使用

    1引入js <script src="${pageContext.request.contextPath }/js/jquery-1.9.1.min.js" type=&qu ...

  10. Javascript(es2016) import和require用法和区别

    http://blog.csdn.net/chinaycheng/article/details/52559439 其中这个总结很到位: 加载方式 规范 命令 特点 运行时加载 CommonJS/AM ...