替换一个实例方法,没你想的那么简单

开发 后端
当你想对类实例的方法进行替换时,你可能想到的是直接对他进行粗暴地替换,但当你试着执行这段代码的时候,就会发现行不通,它提示我们要传入 self 参数。

思路一:简单地替换

[[390247]]

当你想对类实例的方法进行替换时,你可能想到的是直接对他进行粗暴地替换:

  1. class People: 
  2.     def speak(self): 
  3.         print("hello, world") 
  4.  
  5.  
  6. def speak(self): 
  7.     print("hello, python") 
  8.  
  9. p = People() 
  10. p.speak = speak 
  11. p.speak() 

但当你试着执行这段代码的时候,就会发现行不通,它提示我们要传入 self 参数:

  1. Traceback (most recent call last): 
  2.   File "/Users/MING/Code/Python/demo.py", line 12, in <module> 
  3.     p.speak() 
  4. TypeError: speak() missing 1 required positional argument: 'self' 

不对啊~ self 不是实例本身吗?函数不是一直就这么写的?

实际上你这么替换,speak 就变成了一个 function,而不是一个和实例绑定的 method ,你可以把替换前后的 speak 打印出来

  1. p = People() 
  2. print(p.speak) 
  3. p.speak = speak 
  4. print(p.speak) 

输出结果如下,区别非常明显

  1. <bound method People.speak of <__main__.People object at 0x10cfa7fd0>> 
  2. <function speak at 0x10ca10040> 

这种方法,只能用在替换不与实例绑定的静态方法上,不然你每次调用的时候,就得手动传入实例本身,但这样调用就会变得非常怪异。

思路二:利用 im_func

有 Python 2 使用经验的朋友,可以会知道类实例的方法,都有 im_func 和 im_class 属性,分别指向了该方法的函数和类。

很抱歉的是,这些在 Python3 中全都取消了,意味你无法再使用 im_func 和 im_class 。

但即使你身处 Python 2 的环境下,你想通过 im_func 去直接替换函数,也仍然是有问题的。

因为在 Python2 中不推荐普通用户对类实例的方法进行替换,所以 Python 给类实例的方法赋予了只读属性

思路三:非常危险的字节码替换

表层不行,但这个方法在字节码层面却是可行的

这种方法,非常的粗暴且危险,他会直接影响到使用 People 的所有实例的 speak 方法,因此这种方法千万不要使用。

思路四:利用 types 绑定方法

在 types 中有一个 MethodType,可以将普通方法与实例进行绑定。

绑定后,就可以直接替换掉原实例的 speak 方法了,完整代码如下:

  1. import types 
  2.  
  3. class People: 
  4.     def speak(self): 
  5.         print("hello, world") 
  6.  
  7.  
  8. def speak(self): 
  9.     print("hello, python") 
  10.  
  11. p = People() 
  12. p.speak = types.MethodType(speak, p) 
  13. p.speak() 

这种方法,最为安全,不会影响其他实例。并且 Python 2 和 Python 3 都适用,是官方推荐的一种做法。

总结一下

  • 直接替换:只适用于静态方法
  • 使用 im_func 替换:行不通
  • 使用 im_func.func_code 替换字节码:非常危险,请不要使用
  • 使用 types.MethodType 进行方法绑定:安全且有效,推荐使用

 

责任编辑:赵宁宁 来源: Python编程时光
相关推荐

2015-04-30 10:12:13

开源云平台OpenStack

2017-08-09 14:49:03

WebHTTPS浏览器

2024-10-31 11:49:41

Kafka管理死信队列

2014-08-25 10:17:54

数据中心管理

2020-03-26 10:41:02

API网关大公司

2014-03-14 09:35:56

内存优化软件内存优化

2016-01-07 10:17:48

2015-06-24 10:32:13

讯鸟云计算会展

2021-08-02 15:24:19

Windows 11Windows微软

2014-03-21 15:30:06

产品经理PM能力

2023-12-28 12:07:21

2011-09-08 13:41:53

Widget

2013-01-15 10:09:43

Windows Ser

2016-07-25 12:58:07

SDN路由故障排查

2014-07-09 09:06:33

SDN自动化

2010-08-04 09:20:31

JavaScript

2021-12-14 17:19:15

存储数据

2009-06-22 14:02:00

2019-05-17 09:33:50

图像识别三维重建文本识别

2012-08-07 11:28:13

卸载linux
点赞
收藏

51CTO技术栈公众号