Python动态导入模块和反射机制详解


Posted in Python onFebruary 18, 2020

一、前言

何谓动态导入模块,就是说模块的导入可以根据我们的需求动态的去导入,不是像一般的在代码文件开头固定的导入所需的模块。

何谓反射机制,利用字符串的形式在模块或对象中操作(查找/获取/删除/添加)成员。

下面进入具体实例介绍环节。先创建一个示例文件example.py,简单写入几个加减乘除函数,如下,方便下文讲解使用。

flag = 1  # 此变量在介绍反射机制时会用到
 
def my_sum(a, b):
 
  return a + b
 
def my_sub(a, b):
 
  return a - b

二、动态导入模块

一般,如果我们想从其他文件引用上面的几个函数方法,都会如下使用:

import example as count
 
# 加法
sum = count.my_sum(2, 3)
 
# 减法
sub = count.my_sub(6, 2)
 
print("sum: {}, sub: {}".format(sum, sub))

但现在有这样的需求,我需要动态输入一个模块名,可以随时访问到导入模块中的方法或者变量,怎么做呢?看下面。

imp = input("请输入你需要导入的模块名称:")
count = __import__(imp)  # 这种方式就是通过输入字符串导入你想导入的模块 
 
# 加法
sum = count.my_sum(2, 3)
 
# 减法
sub = count.my_sub(6, 2)
 
print("sum: {}, sub: {}".format(sum, sub))

上面实现了动态输入模块名,从而使我们能够导入模块并且执行里面的函数。但是上面有一个缺点,那就是执行的函数被固定了。那么,我们能不能改进一下,动态输入函数名,并且来执行呢?看下面。

imp = input("请输入你需要导入的模块名称:")
count = __import__(imp)
 
func = input("请输入你需要使用的函数名:")
 
f = getattr(count, func, None)
 
# 加法
sum = f(2, 3)
print(sum)

getattr()方法的作用是:从导入的模块中找到你需要调用的函数func,然后返回一个该函数的引用,没有找到就烦会None。

这样我们就实现了,动态导入一个模块,并且动态输入函数名然后执行相应方法。

不过,上面还存在一点点小问题:那就是我们的模块有可能不是在本级目录中存放着,有可能是如下图存放方式:

Python动态导入模块和反射机制详解

那怎么办呢?看下面。

imp = input("请输入你想导入的模块名称:")
count = __import__('first_level.{}'.format(imp), fromlist=True)
 
fun = input("请输入你想要使用的函数名:")
 
f = getattr(count, fun, None)
 
# 加法
sum = f(2, 3)
print(sum)

三、反射机制(又叫 python自省)

我们先来介绍python的四个内置函数:

1. getattr()

这个函数是Python自省的核心函数,具体使用上面已经介绍了,她不仅可以用于在模块中查找获取相应的方法和变量,也可以在一个对象中查找和获取相应的方法和变量,这里就不距离介绍了。

2、hasattr(object, name)

判断模块(或对象object)是否包含名为name的方法或变量(hasattr是通过调用getattr(ojbect, name)是否抛出异常来实现的)

imp = input("请输入你想导入的模块名称:")
count = __import__('first_level.{}'.format(imp), fromlist=True)
 
print(hasattr(count, "my_sum"))  # 判断模块count中是否存在my_sum方法,存在返回True

3、setattr(object, name, value)

这是相对应的getattr()。参数是一个对象,一个字符串和一个任意值。字符串name可以是对象(object)中一个现有的属性或一个新的属性,这个函数将值(value)赋给属性(name)的。使用示例,setattr(x, y, v)相当于x.y = v。

imp = input("请输入你想导入的模块名称:")
count = __import__('first_level.{}'.format(imp), fromlist=True)
 
setattr(count, "flag", 0)  # 即使example模块中没有flag变量,此处也成立,没有的话相当于给模块中新增一个变量flag
 
print(count.flag)  # 打印出flag的值为0

4、delattr(object, name)

与setattr()相关的一组函数。参数是由一个对象(记住!python中一切皆是对象)和一个字符串(name)组成的。name参数必须是对象属性名之一。该函数删除该对象的一个由字符串(name)指定的属性。delattr(x, y)=del x.y.

imp = input("请输入你想导入的模块名称:")
count = __import__('first_level.{}'.format(imp), fromlist=True)
 
delattr(count, "flag")
 
print(count.flag)  # 此处再打印flag的值将会报错,因为上一步已经将flag属性删除了

需要注意的是getattr,hasattr,setattr,delattr函数对模块的修改都在内存中进行,并不会影响文件中真实内容。

5、基于反射机制模拟获取web框架路由的示例

需求:输入:www.xxx.com/example/my_sum,返回执行my_sum的结果。

# 动态导入模块,并执行其中函数
url = input("url: ")
 
target_module = url.split('/')[-2]  # 分割url,取出模块名
 
module = __import__('first_level.' + target_module, fromlist=True)
 
inp = url.split("/")[-1]  # 分割url,并取出url最后一个字符串
if hasattr(module, inp):  # 判断在commons模块中是否存在inp这个字符串
  target_func = getattr(module, inp)  # 获取inp的引用
  sum_ = target_func(2, 3)  # 执行
  print(sum_)
else:
  print("404")

更多关于Python动态导入模块和反射机制请查看下面的相关文章

Python 相关文章推荐
Python编程入门之Hello World的三种实现方式
Nov 13 Python
深入了解Python数据类型之列表
Jun 24 Python
利用python GDAL库读写geotiff格式的遥感影像方法
Nov 29 Python
Python整数对象实现原理详解
Jul 01 Python
Pycharm简单使用教程(入门小结)
Jul 04 Python
django组合搜索实现过程详解(附代码)
Aug 06 Python
Python数据可视化:泊松分布详解
Dec 07 Python
从训练好的tensorflow模型中打印训练变量实例
Jan 20 Python
基于tensorflow指定GPU运行及GPU资源分配的几种方式小结
Feb 03 Python
Pyspark读取parquet数据过程解析
Mar 27 Python
Python 利用argparse模块实现脚本命令行参数解析
Dec 28 Python
python爬虫--selenium模块
Mar 31 Python
pytorch进行上采样的种类实例
Feb 18 #Python
new_zeros() pytorch版本的转换方式
Feb 18 #Python
对pytorch的函数中的group参数的作用介绍
Feb 18 #Python
基于python3实现倒叙字符串
Feb 18 #Python
Python日期格式和字符串格式相互转换的方法
Feb 18 #Python
Python数组并集交集补集代码实例
Feb 18 #Python
通过python检测字符串的字母
Feb 18 #Python
You might like
浅析PHP原理之变量(Variables inside PHP)
2013/08/09 PHP
PHP实现二维数组按照指定的字段进行排序算法示例
2019/04/23 PHP
JavaScript入门教程(6) Window窗口对象
2009/01/31 Javascript
jquery异步请求实例代码
2011/06/21 Javascript
基于jquery扩展漂亮的CheckBox(自己编写)
2013/11/19 Javascript
基于jquery的文字向上跑动类似跑马灯的效果
2014/09/22 Javascript
Nodejs实现批量下载妹纸图
2015/05/28 NodeJs
javascript实现的多个层切换效果通用函数实例
2015/07/06 Javascript
Angularjs中$http以post请求通过消息体传递参数的实现方法
2016/08/05 Javascript
JavaScript代码里的判断小结
2016/08/22 Javascript
js获取腾讯视频ID的方法
2016/10/03 Javascript
Vue.js组件tabs实现选项卡切换效果
2016/12/01 Javascript
jquery表单提交带错误信息提示效果
2017/03/09 Javascript
JavaScript数据类型的存储方法详解
2017/08/25 Javascript
如何将你的AngularJS1.x应用迁移至React的方法
2018/02/01 Javascript
jQuery实现DIV响应鼠标滑过由下向上展开效果示例【测试可用】
2018/04/26 jQuery
浅谈vue父子组件怎么传值
2018/07/21 Javascript
快速解决angularJS中用post方法时后台拿不到值的问题
2018/08/14 Javascript
小程序实现锚点滑动效果
2019/09/23 Javascript
vue路由传参三种基本方式详解
2019/12/09 Javascript
vue从零实现一个消息通知组件的方法详解
2020/03/16 Javascript
jquery+css3实现的经典弹出层效果示例
2020/05/16 jQuery
vue 使用 sortable 实现 el-table 拖拽排序功能
2020/12/26 Vue.js
vue实现登录功能
2020/12/31 Vue.js
[02:03]完美世界DOTA2联赛10月30日赛事集锦
2020/10/31 DOTA
简单学习Python多进程Multiprocessing
2017/08/29 Python
python使用Plotly绘图工具绘制散点图、线形图
2019/04/02 Python
触发器(trigger)的功能都有哪些?写出一个触发器的例子
2012/09/17 面试题
国外的一些J2EE面试题一
2012/10/13 面试题
护理职业生涯规划书
2014/01/24 职场文书
暑假家长评语大全
2014/04/17 职场文书
2014年军人思想汇报范文
2014/10/12 职场文书
工人先锋号申报材料
2014/12/29 职场文书
员工聘用合同范本
2015/09/21 职场文书
Go语言基础函数基本用法及示例详解
2021/11/17 Golang
Python函数对象与闭包函数
2022/04/13 Python