深圳幻海软件技术有限公司 欢迎您!

Python炫技操作:执行函数的九种方法

2023-02-27

 方法一:直接调用函数运行这种是最简单且直观的方法 复制def task():      print("running task")  task() 1.2.3.如果是在

 方法一:直接调用函数运行

这种是最简单且直观的方法 

def task():  
    print("running task")  
task() 
  • 1.
  • 2.
  • 3.

如果是在类中,也是如此 

class Task:  
    def task(self):  
        print("running task")  
Task().task() 
  • 1.
  • 2.
  • 3.
  • 4.

方法二:使用偏函数来执行

在 functools 这个内置库中,有一个 partial 方法专门用来生成偏函数。 

def power(x, n):  
    s = 1  
    while n > 0:  
        nn = n - 1  
        ss = s * x  
    return s  
from functools import partial  
power_2=partial(power, n=2 
power_2(2)  # output: 4  
power_2(3)  # output: 9 
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.

方法三:使用 eval 动态执行

如果你有需要动态执行函数的需要,可以使用 eval + 字符串 来执行函数。 

import sys  
def pre_task():  
    print("running pre_task")  
def task():  
    print("running task")  
def post_task():  
    print("running post_task") 
argvs = sys.argv[1:]  
for action in argvs:  
    eval(action)() 
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.

运行效果如下:

$ python demo.py pre_task task post_task  
running pre_task  
running task  
running post_task 
  • 1.
  • 2.
  • 3.
  • 4.

方法四:使用 getattr 动态获取执行

若把所有的函数是放在类中,并定义成静态方法,那就不需要用 eval 了,接着使用 getattr 去获取并调用。 

import sys  
class Task:  
    @staticmethod  
    def pre_task():  
        print("running pre_task")  
    @staticmethod  
    def task():  
        print("running task")  
    @staticmethod  
    def post_task():  
        print("running post_task")  
argvs = sys.argv[1:]  
task = Task()  
for action in argvs:  
    func = getattr(task, action)  
    func() 
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.

方法五:使用类本身的字典

我们都知道对象都有一个 __dict__() 的魔法方法,存放所有对象的属性及方法。

到这里,大家可以思考一下, 如果还是上面的代码,我直接取实例的 __dict__() 能不能取到函数呢?

我相信很多人都会答错。

上面我们定义的是静态方法,静态方法并没有与实例进行绑定,因此静态方法是属于类的,但是不是属于实例的,实例虽然有使用权(可以调用),但是并没有拥有权。

因此要想通过 __dict__  获取函数,得通过类本身 Task,取出来的函数,调用方法和平时的也不一样,必须先用 __func__ 获取才能调用。 

import sys  
class Task:  
    @staticmethod  
    def pre_task():  
        print("running pre_task")  
func = Task.__dict__.get("pre_task") 
 func.__func__() 
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.

方法六:使用 global() 获取执行

上面放入类中,只是为了方便使用 getattr 的方法,其实不放入类中,也是可以的。此时你需要借助 globals() 或者 locals() ,它们本质上就是一个字典,你可以直接 get 来获得函数。 

import sys  
def pre_task():  
    print("running pre_task") 
def task():  
    print("running task")  
def post_task():  
    print("running post_task")  
argvs = sys.argv[1:]  
for action in argvs:  
    globals().get(action)() 
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.

方法七:从文本中编译运行

先定义一个字符串,内容是你函数的内容,比如上面的 pre_task ,再通过 compile 函数编进 编译,转化为字节代码,最后再使用 exec 去执行它。 

pre_task = "" 
print("running pre_task")  
"""  
exec(compile(pre_task, '<string>', 'exec')) 
  • 1.
  • 2.
  • 3.
  • 4.

若你的代码是放在一个 txt 文本中,虽然无法直接导入运行,但仍然可以通过 open 来读取,最后使用 compile 函数编译运行。 

with open('source.txt') as f:  
    source = f.read()  
    exec(compile(source, 'source.txt', 'exec')) 
  • 1.
  • 2.
  • 3.

方法八:使用 attrgetter 获取执行

在 operator 这个内置库中,有一个获取属性的方法,叫 attrgetter ,获取到函数后再执行。 

from operator import attrgetter  
class People:  
    def speak(self, dest):  
        print("Hello, %s" %dest)  
p = People()  
caller = attrgetter("speak")  
caller(p)("明哥") 
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.

方法九:使用 methodcaller 执行

同样还是 operator 这个内置库,有一个 methodcaller 方法,使用它,也可以做到动态调用实例方法的效果。 

from operator import methodcaller  
class People:  
    def speak(self, dest):  
        print("Hello, %s" %dest)  
caller = methodcaller("speak", "明哥")  
p = People()  
caller(p) 
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.

以上就是我总结的函数执行的十种方法,很多方法,大家也都知道,但是也有几个方法,几乎是见不到的,尤其是后面使用 operator 库的那两种方法。