如何在 Python 中动态调用函数,九种方法有大用处,特别是在框架设计的时候

1. 直接调用函数(__call__

最简单最直接的使用方法:

def func():
    print('Hello, world!')
func() # Hello, world!

func.__call__() # 一样的

2. partial 函数

在python的内置库functools中有一个partial函数,可以让我们可以把一个函数的一部分参数填入,然后调用。看起来没什么用,遇到的时候有大用。

from functools import partial
# 请仔细品!
def func(domain, user):
    echo = f"Hello, {user}@{domain}!"
    print(echo)


func_userA = partial(func, user="userA")
func_userB = partial(func, user="userB")
func_userA("example.com") # Hello, [email protected]!
func_userB("example.com") # Hello, [email protected]!

 

3. eval 函数

如果需要动态执行函数,可以使用 eval 来执行动态代码。

import sys


def pre_task():
    print("running pre_task")


def task():
    print("running task")


def post_task():
    print("running post_task")


cmdList = ["pre_task()", "task()", "post_task()"]
for cmd in cmdList:
    eval(cmd)  # 执行函数

# 或者使用
cmdList = [pre_task, task, post_task]
for cmd in cmdList:
    cmd()  # 执行函数
# running pre_task
# running task
# running post_task

4. 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")

#⚠️ 没有括号的字符串。
cmdList = ["pre_task", "task", "post_task"]

task = Task()

for cmd in cmdList:
    func = getattr(task, cmd)
    func()

 

5. dict 方法

首先我们需要知道,每个python对象都有一个内置的__dict__()方法,这个方法返回一个字典,包含了对象的所有属性。 如下图,我们可以看到list的__dict__()方法返回的所有属性,其中红框内的,你是否有些熟悉?

扫描二维码关注公众号,回复: 14598656 查看本文章
class Task:
    @staticmethod
    def pre_task():
        print("running pre_task")

    @staticmethod
    def task():
        print("running task")

    @staticmethod
    def post_task():
        print("running post_task")


func = Task.__dict__.get("pre_task")
func.__func__() # running pre_task
func() # 为什么不这样用?

 

6. globals 函数

import sys

def pre_task():
    print("running pre_task")

def task():
    print("running task")

def post_task():
    print("running post_task")

cmdList = ["pre_task", "task", "post_task"]

for cmd in cmdList:
    func = globals().get(cmd)
    func()
# running pre_task
# running task
# running post_task

 

7. exec 函数

你可以在一个字符串中定义你的函数,并使用compile函数将它编译成字节码,然后使用exec来执行它。

# 方式1
pre_task = """
print("running pre_task")
"""
exec(compile(pre_task, '', 'exec'))
# running pre_task

# 方式2
with open('./source.txt') as f:
    source = f.read()
    exec(compile(source, 'source.txt', 'exec'))

 

8. attrgetter 函数

在内置库operator中,有一个获取属性的方法,叫做attrgetter,我们可以通过它获取函数后执行。

from operator import attrgetter

class People:
    def speak(self, dest):
        print("Hello, %s" %dest)

p = People()
caller = attrgetter("speak")
caller(p)("Tony") # Hello, Tony
# 本文第四条
caller2 = getattr(p, "speak")
caller2("Tony") # Hello, Tony

 

9. methodcaller 函数 

from operator import methodcaller

class People:
    def speak(self, dest):
        print(f"Hello, {dest}")

caller = methodcaller("speak", "Tony")
p = People()
caller(p)

 

本文分享了使用函数的9种方法: getattrpartialeval__dict__globalsexecattrgettermethodcaller, 和 __call__

请仔细品味,思考下他们的使用场景。其中一些函数调用方法,在函数式编程或元编程场景中大量使用。相信你在今后的学习或工作中会遇到!

猜你喜欢

转载自blog.csdn.net/weixin_44786530/article/details/129876506