这种是最简单且直观的方法
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:
n = n - 1
s = 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 + 字符串 来执行函数。
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
若把所有的函数是放在类中,并定义成静态方法,那就不需要用 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__()
上面放入类中,只是为了方便使用 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, '', 'exec'))
若你的代码是放在一个 txt 文本中,虽然无法直接导入运行,但仍然可以通过 open 来读取,最后使用 compile 函数编译运行。
with open('source.txt') as f:
source = f.read()
exec(compile(source, 'source.txt', 'exec'))
在 operator 这个内置库中,有一个获取属性的方法,叫 attrgetter
,获取到函数后再执行。
from operator import attrgetter
class People:
def speak(self, dest):
print("Hello, %s" %dest)
p = People()
caller = attrgetter("speak")
caller(p)("明哥")
同样还是 operator 这个内置库,有一个 methodcaller 方法,使用它,也可以做到动态调用实例方法的效果。
from operator import methodcaller
class People:
def speak(self, dest):
print("Hello, %s" %dest)
caller = methodcaller("speak", "明哥")
p = People()
caller(p)
以上就是函数执行的九种方法,很多方法,大家也都知道,但是也有几个方法,几乎是见不到的,尤其是后面使用 operator 库的那两种方法。
审核编辑:汤梓红
全部0条评论
快来发表一下你的评论吧 !