• Python如何动态替换对象的方法?


    如何用动态替换一个对象的实例方法,简化代码如下:

    python
    class Test:
        def __init__(self, name):
            self.name = name
    
        def work(self, job):
            print(f'{self.name}正在{job}')
    
    def work(self, job1, job2):
    
    
     print(f'{self.name}正在同时做两个工作,分别是{job1}{job2}')
    
    
    t = Test('kingname')
    t.work = work
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16

    当我们在替换之前,直接运行t.work(‘job’),效果如下:
    在这里插入图片描述

    这个同学期望在替换以后,运行t.work(‘job1’, ‘job2’),能够输出:kingname正在同时做两个工作,分别是job1和job2。但上面的代码,直接运行以后会报错,如下图所示:

    图片

    这说明,替换以后,在调用t.work的时候,Python 不会自动把self传入到第一个参数。

    在以前的文章里面,我们已经讲过,实例方法的第一个参数self,就是这个实例对象自身。我们可以写一段代码来验证这一点:

    class Test:
        def __init__(self, name):
            self.name = name
    
        def work(self, job):
            print(f'{self.name}正在{job}')
        
        def check(self, instance):
            print(f'self的内存地址是:{id(self)}')
            print(f'instance 的内存地址是:{id(instance)}')
            print('self与 instance 就是同一个对象:', self is instance)
    
    
    t = Test('kingname')
    t.check(t)
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15

    运行效果如下图所示:

    在这里插入图片描述

    知道这一点以后,要解决动态替换以后报错的问题,最简单的方法就是手动把实例对象作为第一个参数传入进去,如下图所示:

    在这里插入图片描述

    但这样做显然很麻烦,每次都要手动传入第一个实例对象。有没有什么办法能省略它呢?这个时候,如果你记得我公众号里面的这篇文章偏函数:在Python中设定默认参数的另一种办法,那么你就有办法了。使用偏函数,提前把第一个参数固定下来,就能解决问题:

    from functools import partial
    def work(self, job1, job2):
        print(f'{self.name}正在同时做两个工作,分别是{job1}{job2}')
    
    simple_work = partial(work, t)
    t.work = simple_work
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6

    运行效果如下图所示:

    图片

  • 相关阅读:
    自然语言处理-BERT处理框架-transformer
    springboot与flowable(10):网关服务(排他网关)
    并查集合并、计算集合个数、每个集合的元素
    【Git】Gitbash使用ssh 上传本地项目到github
    Vue3中的refs使用
    经典动态规划:最长递增子序列
    [云原生K8S] Yaml文件详解
    15天获取20万用户,小程序如何做到?
    2023.9.11 关于传输层协议 UDP和TCP 详解
    NoSQL数据库之Redis2
  • 原文地址:https://blog.csdn.net/z099164/article/details/126252537