当前位置: 首页 > 面试题库 >

对Python的id()感到困惑

刘辰钊
2023-03-14
问题内容

我可以理解以下定义:

每个对象都有一个标识,一个类型和一个值。一旦创建了对象,其身份就永远不会改变。您可能会认为它是对象在内存中的地址。所述is操作者比较两个对象的身份;
id()函数返回一个表示其身份的整数。

我认为上面的定义在创建“某物”时起作用,例如:

>>> a = 0
>>> id(a)
1720438480

但是我不理解:

>>> id(1)
1720438512
>>> b = 1
>>> id(b)
1720438512

我还没有创建任何东西。那么整数“ 1”如何具有ID?这是否意味着只要我在Python Shell中“提及”
1,便立即将其分配给内存地址?另外,这是否意味着因为ID在其生命周期内从未发生变化,并且由于我的计算机内存有限,所以如果我反复询问唯一事物的id(),我最终会得到类似“内存不足”的消息吗?(它不能重新分配内存,因为其他对象的生命期尚未结束。)

或者,从相反的方向显示我的耳朵:

>>> id(something_1)
some unique memory address
>>> id(something_2)
some unique memory address
>>> ...

在什么时候重新分配内存?那就是在那个时候

>>> my_variable = something_1
>>> id(my_variable)

会提供与的ID不同的ID id(something_1)


问题答案:

通常,一旦您使用整数或字符串或任何其他文字,Python就会在您的内存中创建一个新对象。保证id在对象的生存期内具有相同的值,即,其引用计数不为零。

当您编写类似的内容时:

>>> id(1000)
140497411829680

Python创建整数1000并返回其id(CPython中对象的内存地址)。完成此操作后,整数对象1000的引用计数为零,并将其删除。这样可以确保您不能仅通过写入id(something)(也不能将任何变量名绑定到对象)来保持填充内存。

通常,您无法预测何时会发生重用,但是在我的Python Shell中,它会相当一致地发生:

>>> id(1000)
140697307078576
>>> id(1001)
140697307078576
>>> id(1002)
140697307078576
>>> id(1003)
140697307078576

您会看到在创建每个新整数时,一次又一次使用相同的内存地址。但是,如果您防止引用计数下降到零,则可以看到使用了新的内存:

>>> a = 1000
>>> id(a)
140697307078576
>>> b = 1001
>>> id(b)
140697306008368

在CPython中,整数-5到255是特殊情况,因为它们 始终
存在(因此id在Python运行时始终保持相同)。这是为了避免重复创建和破坏常用整数而进行的优化。



 类似资料:
  • 我正在尝试提出一种解决方案,它涉及在连接操作之后应用一些逻辑,从多个中的中选择一个事件。这类似于reduce函数,但它只返回1个元素,而不是递增地返回。因此最终结果将是单个(,对,而不是一个 每个键保证只到达一次。 假设像上面这样的连接操作,它用4个生成了1个,成功地连接并收集在。现在,我想做的是,立即访问这些值,并执行一些逻辑以将正确匹配到一个。例如,对于上面的数据集,我需要(,和)。 将为每个

  • 所以我一直在读Kafka的语义学,我对它的工作原理有点困惑。 我理解生产者如何避免发送重复的消息(以防代理的ack失败),但我不明白的是,在消费者处理消息但在提交偏移量之前崩溃的情况下,一次是如何工作的。Kafka不会在这种情况下重试吗?

  • 问题内容: 来自Java背景,我知道这类似于toString的Python版本(虽然我确实意识到Python是较老的语言)。 因此,我定义了一个小类以及一个如下的方法: 然后,我创建它的一些实例: 现在,尝试打印这些对象之一时的预期行为是将打印其关联的值。这也会发生。 产量 但是当我执行以下操作时: 接着 我懂了 我期望的地方 我想念什么?而我在做什么其他值得冒犯的东西呢?:) 问题答案: Pyt

  • 问题内容: 我已经在eclipse中创建了一个项目,并添加了Maven依赖项。在Eclipse中,它表示我正在使用JRE 1.5。一切在Eclipse中都可以正常运行,例如,我可以运行测试。 当我尝试从终端运行时,出现以下错误。 …在-source 1.3中不支持泛型(使用-source 5或更高版本来启用泛型)… 看来,Maven认为我正在使用JRE 1.3,并且无法识别泛型或for-each循

  • 问题内容: 在碰到此链接http://www.javacodegeeks.com/2013/01/java-thread-pool-example-using- executors-and-threadpoolexecutor 之后,这是我第一次为新项目使用Java线程池。 .html ,我对此更加困惑,这是页面中的代码, 在代码中,创建了一个固定大小的池并创建了10个工作线程,对吗? 线程池应该

  • 问题内容: 与此代码有点混淆。 我在pg-go 仓库中找到了这段代码,不知道为什么这样声明。请解释一下用这种方式声明变量的用例是什么。 问题答案: 这在运行时不会执行任何操作,但是除非类型满足接口要求,否则编译将失败。这是一种静态断言。