方法一:直接调用函数运行
这种是最简单且直观的方法
- def task():
- print("running task")
- task()
如果是在类中,也是如此
- class Task:
- def task(self):
- print("running task")
- Task().task()
方法二:使用偏函数来执行
在 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
方法三:使用 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)()
运行效果如下:
- $ python demo.py pre_task task post_task
- running pre_task
- running task
- running post_task
方法四:使用 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()
方法五:使用类本身的字典
我们都知道对象都有一个 __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__()
方法六:使用 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)()
方法七:从文本中编译运行
先定义一个字符串,内容是你函数的内容,比如上面的 pre_task ,再通过 compile 函数编进 编译,转化为字节代码,最后再使用 exec 去执行它。
- pre_task = """
- print("running pre_task")
- """
- exec(compile(pre_task, '<string>', 'exec'))
若你的代码是放在一个 txt 文本中,虽然无法直接导入运行,但仍然可以通过 open 来读取,最后使用 compile 函数编译运行。
- with open('source.txt') as f:
- source = f.read()
- exec(compile(source, 'source.txt', 'exec'))
方法八:使用 attrgetter 获取执行
在 operator 这个内置库中,有一个获取属性的方法,叫 attrgetter ,获取到函数后再执行。
- from operator import attrgetter
- class People:
- def speak(self, dest):
- print("Hello, %s" %dest)
- p = People()
- caller = attrgetter("speak")
- caller(p)("明哥")
方法九:使用 methodcaller 执行
同样还是 operator 这个内置库,有一个 methodcaller 方法,使用它,也可以做到动态调用实例方法的效果。
- from operator import methodcaller
- class People:
- def speak(self, dest):
- print("Hello, %s" %dest)
- caller = methodcaller("speak", "明哥")
- p = People()
- caller(p)
以上就是我总结的函数执行的十种方法,很多方法,大家也都知道,但是也有几个方法,几乎是见不到的,尤其是后面使用 operator 库的那两种方法。