跳至主要內容
  • Hostloc 空間訪問刷分
  • 售賣場
  • 廣告位
  • 賣站?

4563博客

全新的繁體中文 WordPress 網站
  • 首頁
  • “a == b” 与”id(a) == id(b)”的本质区别
未分類
15 1 月 2021

“a == b” 与”id(a) == id(b)”的本质区别

“a == b” 与”id(a) == id(b)”的本质区别

資深大佬 : hanssx 6

群友 Jin 发出一段代码,

class Person:     def say(self):         pass      p1 = Person() p2 = Person()  print(p1.say, id(p1.say)) print(p2.say, id(p2.say)) print(p1.say == p2.say)  # False print(p1.say is p2.say)  # False print(p1.say is p1.say)  # False print(id(p1.say) == id(p2.say))  # True 

输出:

<bound method Person.say of <__main__.Person object at 0x7f6739300898>> 140081602502024 <bound method Person.say of <__main__.Person object at 0x7f67393005c0>> 140081602502024 False False False True 

python 官网上对 id 的解释

id(object)  Return the “identity” of an object. This is an integer which is guaranteed to be unique and constant for this object during its lifetime. Two objects with non-overlapping lifetimes may have the same id() value.  CPython implementation detail: This is the address of the object in memory.  Raises an auditing event builtins.id with argument id. 

我理解就是返回内存地址

StackOverFlow 上对 is 的解释: https://stackoverflow.com/questions/132988/is-there-a-difference-between-and-is 也就是引用比较,不就是比较内存地址?

群友 Sweeneys 发了一篇 StackOverFlo 的类似问题帖子: https://stackoverflow.com/questions/2906177/what-is-the-difference-between-a-is-b-and-ida-idb-in-python

回答中有一段话

b.test happens to have the same id as the bound method you had before and it’s allowed to because no other objects have the same id now.

这意思就是说 a.test 被 gc 之后恰好释放的内存,正好被 b.test 所使用?所以才导致 True ?

于是群友 Jin 改成了如下代码,依然输出同样的结果,

import gc gc.disable()  class Person:     def say(self):         pass      p1 = Person() p2 = Person()  print(p1.say, id(p1.say)) print(p2.say, id(p2.say)) print(p1.say == p2.say)  # False print(p1.say is p2.say)  # False print(p1.say is p1.say)  # False print(id(p1.say) == id(p2.say))  # True  

我尝试改成这样,依然输出同样的结果。

class Person:     def say(self):         pass      p1 = Person() p2 = Person()  print(p1.say, id(p1.say)) print(p2.say, id(p2.say)) print(p1.say == p2.say)  # False print(p1.say is p2.say)  # False print(p1.say is p1.say)  # False p1_ref = p1.say p2_ref = p2.say print(id(p1.say) == id(p2.say))  # True 

话说,python 中一个类中的 self 方法本身应该只在内存中占一个位置吧?然后调用的时候把对象地址 self 传进来调用,我记得 C++是这样的,虽然 Python 底层是 c 的 struct 实现的,但是也不至于这么弱智,同个类的不同对象的 self 方法占用多块内存吧?

大佬有話說 (14)

  • 主 資深大佬 : hanssx

    改成群友确实就是 False 了,为啥我上面最后写的那段不行?
    “`python
    class Person:
    def say(self):
    pass

    p1 = Person()
    p2 = Person()

    print(p1.say, id(p1.say))
    print(p2.say, id(p2.say))
    print(p1.say == p2.say) # False
    print(p1.say is p2.say) # False
    print(p1.say is p1.say) # False
    p1_ref = p1.say
    p2_ref = p2.say
    print(id(p1_ref) == id(p2_ref)) # False
    “`

  • 資深大佬 : AoEiuV020

    太复杂,没看出来每个都改了啥,

  • 資深大佬 : forbxy

    class Dog:
    def say(self):
    pass

    d = Dog()

    id(d.say) == id(p2.say) # Ture

  • 資深大佬 : zk8802

    func.__eq__() 可能比较特殊。我猜它会检查 self 是否一致。

  • 資深大佬 : todd7zhang

    不知道,盲猜 id 相等是因为
    p1.say.im_func is p2.say.im_func
    python3 就是 p1.say.__func__ is p2.say.__func__

  • 資深大佬 : AlohaV2

    https://stackoverflow.com/questions/13348031/ids-of-bound-and-unbound-method-objects-sometimes-the-same-for-different-o

  • 資深大佬 : AoEiuV020

    我刚知道,这个 id 居然是每次都不一样的,那就不可能是返回内存地址啥的了,具体还是要看这个方法底层实现,
    "a == b" 与"id(a) == id(b)"的本质区别

  • 主 資深大佬 : hanssx

    @AoEiuV020 是不是每次分配的内存地址不一样?

  • 資深大佬 : AlohaV2

    @AoEiuV020 是内存地址相关的一个东西
    调用 bound method 的话,id(p1.say)其实是新建了一个 bound method, 里面绑定了 Person.say 和 p1 这俩东西。
    所以类似的操作还有
    “`
    class WhatEver:
    def say(self):
    pass
    def run(self):
    pass

    foo = WhatEver()
    print(id(foo.say) == id(foo.run))
    “`

  • 資深大佬 : todd7zhang

    学到了

  • 資深大佬 : zhanglintc

    为啥这个是 False 来着:
    print(p1.say is p1.say) # False

  • 主 資深大佬 : hanssx

    更新群友 Sweeneys 回复:
    ”
    对,我 jio 得大家不能 get 的点在这,即每次 instance.method_name 都会创建一个新的方法对象( Method Object )。还有一点就是生命周期的问题,如果两个对象的生命周期不重叠,那么 id(object)得到的值可能会一样。
    “

  • 資深大佬 : milkpuff

    https://github.com/satwikkansal/wtfpython#-strings-can-be-tricky-sometimes
    github 上的一个专门总结 python 奇特特性的仓库,叫做 wtfpython,有 23k star

  • 主 資深大佬 : hanssx

    @milkpuff 有点意思

文章導覽

上一篇文章
下一篇文章

AD

其他操作

  • 登入
  • 訂閱網站內容的資訊提供
  • 訂閱留言的資訊提供
  • WordPress.org 台灣繁體中文

51la

4563博客

全新的繁體中文 WordPress 網站
返回頂端
本站採用 WordPress 建置 | 佈景主題採用 GretaThemes 所設計的 Memory
4563博客
  • Hostloc 空間訪問刷分
  • 售賣場
  • 廣告位
  • 賣站?
在這裡新增小工具