请教一个关于 Python 实例化内存地址的基础问题

2020-11-23 14:58:38 +08:00
 foxyier
问题描述: 循环执行以下两个操作: 「 1. 初始化一个实例,2. 输出实例的内存地址」, 发现每次输出的内存地址在两个内存地址中来回跳动, 但是把这两个操作放入函数中调用, 每次输出的内存地址就相同了, 不太懂具体原理, 目前猜测是语言解释的过程中做了缓存? 如果是这样的话, 单例模式的应用场景感觉是不是就很少了?

示例 1:
for i in range(5):
c = Class1()
print id(c)

示例输出:

4467795088
4467795152
4467795088
4467795152
4467795088

示例 2:

def run():
c = Class1()
print id(c)

for i in range(5):
run()

示例输出:

4566115472
4566115472
4566115472
4566115472
4566115472

1. 想请教一下为什么这两个操作在函数中运行时,输出的实例内存地址相同.
2. 想请教一下为什么采用第一种方式的时候,实例的内存地址在两个内存地址中来回跳动
1188 次点击
所在节点    Python
3 条回复
luoleng
2020-11-23 15:20:31 +08:00
示例 1 中,先是创建一个 Class1 实例( m ),然后把它赋予变量 c,新一轮 for 循环的时候,又重新创建一个 Class1 实例( n ),然后又把它赋予变量 c,这个时候因为实例 m 没有被引用了,所以会被销毁。之后的循坏又是在 m 的原地址上创建 m1,然后销毁 n……
示例 2 中,函数执行完后在函数体内创建的变量就被销毁了;之后的实例也是在原来实例的地址上创建的;
至于新实例为什么会在已被销毁的旧实例地址上创建,我姑且认为是 python 自己内存优化的一种结果吧。
gwy15
2020-11-23 15:28:22 +08:00
楼上说得对,要到第二次 c = Class1() 的时候才会失去上一个迭代的引用,然后才会进行 GC 。

如果改成

for i in range(5):
....c = Class1()
....print(f'0x{id(c):x}')
....del c

就会变成一样的地址了
foxyier
2020-11-23 15:41:19 +08:00
@luoleng @gwy15 理解了, 多谢

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/728335

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX