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

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

2019年03月19日 18:22:52 IT界的游骑兵 阅读数 56

 版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/SCF_1104/article/details/88669251

一、前言

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

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

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

 
  1. flag = 1 # 此变量在介绍反射机制时会用到

  2.  
  3.  
  4. def my_sum(a, b):

  5.  
  6. return a + b

  7.  
  8.  
  9. def my_sub(a, b):

  10.  
  11. return a - b

二、动态导入模块

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

 
  1. import example as count

  2.  
  3. # 加法

  4. sum = count.my_sum(2, 3)

  5.  
  6. # 减法

  7. sub = count.my_sub(6, 2)

  8.  
  9.  
  10. print("sum: {}, sub: {}".format(sum, sub))

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

 
  1. imp = input("请输入你需要导入的模块名称:")

  2. count = __import__(imp) # 这种方式就是通过输入字符串导入你想导入的模块

  3.  
  4. # 加法

  5. sum = count.my_sum(2, 3)

  6.  
  7. # 减法

  8. sub = count.my_sub(6, 2)

  9.  
  10.  
  11. print("sum: {}, sub: {}".format(sum, sub))

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

 
  1. imp = input("请输入你需要导入的模块名称:")

  2. count = __import__(imp)

  3.  
  4. func = input("请输入你需要使用的函数名:")

  5.  
  6. f = getattr(count, func, None)

  7.  
  8. # 加法

  9. sum = f(2, 3)

  10. print(sum)

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

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

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

那怎么办呢?看下面。

 
  1. imp = input("请输入你想导入的模块名称:")

  2. count = __import__('first_level.{}'.format(imp), fromlist=True)

  3.  
  4. fun = input("请输入你想要使用的函数名:")

  5.  
  6. f = getattr(count, fun, None)

  7.  
  8. # 加法

  9. sum = f(2, 3)

  10. print(sum)

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

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

1. getattr()

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

2、hasattr(object, name)

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

 
  1. imp = input("请输入你想导入的模块名称:")

  2. count = __import__('first_level.{}'.format(imp), fromlist=True)

  3.  
  4. print(hasattr(count, "my_sum")) # 判断模块count中是否存在my_sum方法,存在返回True,不存在返回False

3、setattr(object, name, value)

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

 
  1. imp = input("请输入你想导入的模块名称:")

  2. count = __import__('first_level.{}'.format(imp), fromlist=True)

  3.  
  4. setattr(count, "flag", 0) # 即使example模块中没有flag变量,此处也成立,没有的话相当于给模块中新增一个变量flag

  5.  
  6. print(count.flag) # 打印出flag的值为0

4、delattr(object, name)

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

 
  1. imp = input("请输入你想导入的模块名称:")

  2. count = __import__('first_level.{}'.format(imp), fromlist=True)

  3.  
  4. delattr(count, "flag")

  5.  
  6. print(count.flag) # 此处再打印flag的值将会报错,因为上一步已经将flag属性删除了

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

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

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

 
  1. # 动态导入模块,并执行其中函数

  2. url = input("url: ")

  3.  
  4. target_module = url.split('/')[-2] # 分割url,取出模块名

  5.  
  6. module = __import__('first_level.' + target_module, fromlist=True)

  7.  
  8. inp = url.split("/")[-1] # 分割url,并取出url最后一个字符串

  9. if hasattr(module, inp): # 判断在commons模块中是否存在inp这个字符串

  10. target_func = getattr(module, inp) # 获取inp的引用

  11. sum_ = target_func(2, 3) # 执行

  12. print(sum_)

  13. else:

  14. print("404")

好了,到这里就介绍结束了。有疑议或不明白的地方可以再评论区留言哦。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值