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根据出生年份简单计算生肖的方法
Mar 27 Python
Python下rrdtool模块的基本使用方法
Nov 13 Python
pandas将DataFrame的列变成行索引的方法
Apr 10 Python
python+opencv实现摄像头调用的方法
Jun 22 Python
python的一些加密方法及python 加密模块
Jul 11 Python
Django用户认证系统 Web请求中的认证解析
Aug 02 Python
pycharm 2019 最新激活方式(pycharm破解、激活)
Sep 22 Python
Django REST framwork的权限验证实例
Apr 02 Python
如何基于python对接钉钉并获取access_token
Apr 21 Python
Python几种常见算法汇总
Jun 02 Python
python3.6.5基于kerberos认证的hive和hdfs连接调用方式
Jun 06 Python
python 利用百度API识别图片文字(多线程版)
Dec 14 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下一个阿拉伯数字转中文数字的函数
2007/07/16 PHP
php 模拟 asp.net webFrom 按钮提交事件实例
2014/10/13 PHP
php文件操作小结(删除指定文件/获取文件夹下的文件名/读取文件夹下图片名)
2016/05/09 PHP
Prototype1.5 rc2版指南最后一篇之Position
2007/01/10 Javascript
基于jquery跨浏览器显示的file上传控件
2011/10/24 Javascript
jquery选择器之属性过滤选择器详解
2014/01/27 Javascript
后端接收不到AngularJs中$http.post发送的数据原因分析及解决办法
2016/07/05 Javascript
JavaScript基于对象去除数组重复项的方法
2016/10/09 Javascript
javascript加载xml 并解析各节点的值(实现方法)
2016/10/12 Javascript
ES6概念 Symbol toString()方法
2016/12/25 Javascript
利用JS实现简单的日期选择插件
2017/01/23 Javascript
Nodejs 发送Post请求功能(发短信验证码例子)
2017/02/09 NodeJs
jQuery Chosen通用初始化
2017/03/07 Javascript
React注册倒计时功能的实现
2018/09/06 Javascript
Vue监听数据渲染DOM完以后执行某个函数详解
2018/09/11 Javascript
Echart折线图手柄触发事件示例详解
2018/12/16 Javascript
VUE安装使用教程详解
2019/06/03 Javascript
ES6 let和const定义变量与常量的应用实例分析
2019/06/27 Javascript
简单分析Python中用fork()函数生成的子进程
2015/05/04 Python
Python中splitlines()方法的使用简介
2015/05/20 Python
分享Python文本生成二维码实例
2016/01/06 Python
详解使用pymysql在python中对mysql的增删改查操作(综合)
2017/01/18 Python
python删除不需要的python文件方法
2018/04/24 Python
Tensorflow全局设置可见GPU编号操作
2020/06/30 Python
Django Admin后台模型列表页面如何添加自定义操作按钮
2020/11/11 Python
matplotlib制作雷达图报错ValueError的实现
2021/01/05 Python
幼师自我鉴定
2014/02/01 职场文书
应届生求职信范文
2014/06/30 职场文书
一份恶作剧的检讨书
2014/09/13 职场文书
二手房购房协议书范本
2014/10/05 职场文书
个人作风建设总结
2014/10/23 职场文书
交警正风肃纪剖析材料
2014/10/29 职场文书
2014年安全保卫工作总结
2014/11/13 职场文书
大学生求职自荐信
2015/03/24 职场文书
公司放假通知范文
2015/04/14 职场文书
python之django路由和视图案例教程
2021/07/26 Python