Python 类方法实例方法内存地址相同???

问题来源 : 5. 什么是绑定到对象的方法、绑定到类的方法、解除绑定的函数、如何定义,如何调用,给谁用?有什么特性?

从代码表现上来说

class Person(object):
    def func01(self):
        print('绑定到对象的方法')
    
    @classmethod
    def func02(cls):
        print('绑定到类的方法')
    
    @staticmethod
    def func03():
        print('非绑定方法')

从代码调用上来说

p1 = Person()
p1.func01()     # 需要通过对象调用
Person.func02() # 通过类对象调用
Person.func03() # 和类对象调用的方式类似
# 这里可以看到绑定到类对象的函数和非绑定函数的调用方式类似

以上三种函数对象的打印输出为
绑定到对象的函数: <bound method Person.func01 of <__main__.Person object at 0x10b4c9ac8>>
绑定到类的函数: <bound method Person.func02 of <class '__main__.Person'>>
非绑定函数: <function Person.func03 at 0x10b4f0b70>

# 这里出现了个问题 关于python方法调用的流程, 实在是不理解
print(id(p1.func01))        # 4578451592
print(id(p2.func01))        # 4578451592

print(id(Person.func02))    # 4578451592
print(id(Person.func02))    # 4578451592
print(id(Person.func03))    # 4578581360
print(id(Person.func03))    # 4578581360

print(Person().func01 == Person.func02) # False
print(id(Person().func01) == id(Person.func02)) # True

针对上面我不理解的问题 这里做出我的理解

我们先来看看Person.__dict__的输出, 在我定义了对象方法, 类方法, 静态方法后, 这些方法都是在Person这个类的__dict__中的, 一下是输出(我们只关注这个几个方法, 其他的打印不写出):

'func01': <function Person.func01 at 0x109bef9d8>, 
'func02': <classmethod object at 0x109bf7b00>, 
'func03': <staticmethod object at 0x109bf7b70>

这里可以明确的看出这几个方法是什么方法, 且内存地址是不一致的. 到这里完全是可以理解的, 在我的认知中, 不同名, 不同类型的方法确实不应当拥有相同的地址.所以,在方法存储的环节,我认为这些方法在内存的表现是让我满意的.

那么是什么导致我看到的地址相同?既然存储没问题, 那么可能导致内存地址相同的关键是在方法调用的过程中.

先来看看对象方法和类方法的内存地址

print(hex(id(Person().func01))) # 0x10c5a6048
print(hex(id(Person.func02)))   # 0x10c5a6048

print(Person.__dict__['func01'])
# <function Person.func01 at 0x109bef9d8>
print(Person.__dict__['func02'])
# <classmethod object at 0x109bf7b00>

上面的类方法和对象方法的地址是相同的, 但从下面可以看出他们在类对象的__dict__中依旧是各自的地址, 接下来就是问题的关键了!!!

当我们通过实例化对象的时候, Python编辑器会将类方法的对象重新使用描述器包装一下, 然后存储到一个新的内存空间, 因此我们调用Person.func01这样的方法,实际上调用的并不是dict中的方法对象, 而是包装过的一个副本!简单的说就是, 不论我们通过Person.func01还是Person().func02去调用方法, 在这个调用的过程中, Python内部会对Person.__dict__['func01']和Person.__dict__['func02']做一次拷贝, 所以我们调用的只是方法的副本. 那么,为什么会出现相同的地址呢?我们都知道Python采用的是垃圾回收的机制, 当一个内存, 没有对象对其引用的话, 就会立刻销毁这块内存然后对其复用. 这个问题发生的关键就在于这个复用阶段. 我们可以推测, 如果不销毁这块内存, Python自然就会开辟新的内存空间去创建对象了吧?我们做个试验.

# 这里我们要做的就是防止编译器对func01方法立刻销毁, 我们将他加入列表或者用一个变量指向都行
p1_func01 = Person().func01 # 这里用一个变量指向了一下

print(hex(id(p1_func01)))   # 0x102eb20c8
print(hex(id(Person.func02)))   # 0x102eacdc8

现在我们就看到了我们期望的结果了, 对象方法和类方法内存地址不一样了.

结论: 造成这种地址相同错觉的罪魁祸首就是Python的内存管理机制... 实际上他们是不同的对象, 只是编译器对内存做了重用. 以上就是我对这个问题的理解 - - 有错误希望大佬指出!_(:зゝ∠)_

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 1.ios高性能编程 (1).内层 最小的内层平均值和峰值(2).耗电量 高效的算法和数据结构(3).初始化时...
    欧辰_OSR阅读 29,657评论 8 265
  • Python 面向对象Python从设计之初就已经是一门面向对象的语言,正因为如此,在Python中创建一个类和对...
    顺毛阅读 4,250评论 4 16
  • 〇、前言 本文共108张图,流量党请慎重! 历时1个半月,我把自己学习Python基础知识的框架详细梳理了一遍。 ...
    Raxxie阅读 19,084评论 17 410
  • 要实现的样子 好多应用都要这样写,那么有哪几种方法那? 方法一.增加cell的高度(底部还会有10dx空白) 1....
    mkb2阅读 727评论 0 0
  • 轻轻地推开窗 将清晨的第一缕阳光收藏 眺望远方初升的太阳 将夜晚的那一抹悲伤遗忘 尽管门前的小路有些泥泞 尽管街旁...
    雲墨寒阅读 213评论 0 1