page contents

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

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

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

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

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 动态执行

如果你有需要动态执行函数的需要,可以使用 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 库的那两种方法。

attachments-2021-04-gya1gTf8608bc3f1330d9.jpg

  • 发表于 2021-04-30 16:47
  • 阅读 ( 569 )
  • 分类:Python开发

0 条评论

请先 登录 后评论
小柒
小柒

1478 篇文章

作家榜 »

  1. 轩辕小不懂 2403 文章
  2. 小柒 1478 文章
  3. Pack 1135 文章
  4. Nen 576 文章
  5. 王昭君 209 文章
  6. 文双 71 文章
  7. 小威 64 文章
  8. Cara 36 文章