0

0

Python 中 in 运算符在集合和列表中的不同行为

心靈之曲

心靈之曲

发布时间:2025-09-25 17:09:21

|

469人浏览过

|

来源于php中文网

原创

python 中 in 运算符在集合和列表中的不同行为

Python 中 in 运算符在集合和列表中的不同行为

本文深入探讨了 Python 中 in 运算符在集合(set)和列表(list)中的不同行为。通过分析其内部实现机制,解释了为何在特定情况下,使用 in 运算符在列表中会引发错误,而在集合中却能正常运行。此外,还提供了自定义类和 Pytorch 张量的示例,以帮助读者更好地理解和应用 in 运算符,并针对 Pytorch 张量比较问题,提供了一种基于张量大小的解决方案。

在 Python 中,in 运算符用于检查某个元素是否存在于一个集合中。然而,其行为在不同类型的集合(如列表和集合)中有所不同。理解这些差异对于编写高效且健壮的代码至关重要。

in 运算符的工作原理

x in collection 的工作方式取决于所使用的集合类型。具体来说,使用内部哈希表的集合(如集合和字典)与不使用哈希表的集合(如列表和元组)的工作方式不同。

1. 不使用哈希表的集合(列表、元组等)

立即学习Python免费学习笔记(深入)”;

当 collection 是列表或元组时,x in collection 的内部实现类似于以下伪代码:

def is_in(x, collection):
  for c in collection:
      if (x is c or x==c):
          return True
  return False
  • 依次比较 collection 中的每个元素 c 与 x,首先进行身份比较 (is),如果身份不同,则进行相等性比较 (==)。
  • 只要找到第一个匹配项,就返回 True。

2. 使用哈希表的集合(集合、字典等)

当 collection 是集合或字典时,x in collection 的内部实现如下:

def is_in(x, collection):
  # 选择集合中哈希值与 x 相同的元素子集
  subset = get_subset_by_hash(collection, hash(x))
  for c in subset:
      if (x is c or x==c):
          return True
  return False
  • 首先,从 collection 中选择哈希值与 x 相同的元素子集 subset。这是集合查找速度快的原因。
  • 然后,迭代 subset 中的每个元素 c,进行身份比较和相等性比较,并在找到第一个匹配项时停止。即使对于非常大的数据集,subset 通常也只有 1 个或几个元素。
  • 注意:集合中元素的哈希值在添加到集合时计算,而 x 的哈希值在使用 in 运算符时计算。

示例:自定义类

为了更好地理解 in 运算符的行为,我们可以创建一个自定义类 MyObj,并定义其自己的哈希计算逻辑 (hash(x)) 和相等性逻辑 (x == c):

class MyObj:
    def __init__(self, val, hashval):
        self._val = val
        self._hashval = hashval

    def __hash__(self):
        print(f"{str(self)} calling __hash__")
        return self._hashval

    def __eq__(self, other):
        print(f"{str(self)} calling __eq__, {other=}")
        return super().__eq__(other)

    def __repr__(self):
        return f"<{self.__class__.__name__}: {self._val}>"

然后,创建几个 MyObj 实例,并创建一个集合 s 和一个列表 lst:

a = MyObj("a", 123)
b = MyObj("b", 456)
d = MyObj("d", 456)  # 与 b 相同的哈希值!

print("Creating set `s`")
s = set([a, b, d])

print("Creating list `lst`")
lst = [a, b, d]

创建集合时,Python 会计算元素的哈希值。如果存在哈希冲突(例如,b 和 d 具有相同的哈希值),则还需要调用 __eq__。

1. 集合中的 in 运算符

Python精要参考 pdf版
Python精要参考 pdf版

这本书给出了一份关于python这门优美语言的精要的参考。作者通过一个完整而清晰的入门指引将你带入python的乐园,随后在语法、类型和对象、运算符与表达式、控制流函数与函数编程、类及面向对象编程、模块和包、输入输出、执行环境等多方面给出了详尽的讲解。如果你想加入 python的世界,David M beazley的这本书可不要错过哦。 (封面是最新英文版的,中文版貌似只译到第二版)

下载
>>> s
{, , }
>>> b in s
 calling __hash__
True
>>> d in s
 calling __hash__
 calling __eq__, other=
 calling __eq__, other=
True

Python 首先计算 x 的哈希值。如果存在哈希冲突,Python 还需要调用 __eq__,因此也会调用 x == c。

2. 列表中的 in 运算符

>>> lst
[, , ]
>>> a in lst
True
>>> b in lst
 calling __eq__, other=
 calling __eq__, other=
True
>>> d in lst
 calling __eq__, other=
 calling __eq__, other=
 calling __eq__, other=
 calling __eq__, other=
True

Python 首先检查 x is c,如果结果为 True(身份检查),则不需要检查 x == c 的相等性。如果身份检查结果为 False,则 Python 会检查 x == c 的相等性。

Pytorch 张量中的注意事项

在 Pytorch 中,如果尝试比较大小不同的张量 a 和 b,则会引发 RuntimeError。torch.Tensor 的哈希值计算只是返回 id(self)。

当执行 b in list([a, b]) 时,会引发错误,因为逻辑会比较:

  • id(b) is id(a) -> False
  • b == a -> 引发 RuntimeError,因此永远不会将 b 与列表中的 b 进行比较。

当执行 b in set([a, a, b]) 时,不会引发错误,因为集合会转换为类似于 s = {id(a): a, id(b): b} 的哈希表。b in s 执行以下操作:

  • "hash(b) 是否与 s 的哈希表中的任何哈希值相同?是的!"
  • b is b?(in 左侧的 b 是否与集合中哈希值与 b 相同的对象相同?)。这是 True。

总结:b in set([a, b]) 和 b in [a, b] 之间的区别在于,对于列表,将按顺序检查 (x is c or x==c),而对于集合,将首先检查哈希值,然后迭代集合中具有相同哈希值的所有项以检查 (x is c or x==c)。由于 hash(b) != hash(a),几乎永远不会比较 b == a,从而在大多数情况下避免了 RuntimeError。

解决方案

为了解决 Pytorch 张量比较的问题,可以利用 torch.Tensor.size 属性(它是元组的子类),并创建一个集合(或列表)字典,用于存储不同大小的张量。

例如:

import torch

tensors_by_size = {}

a = torch.Tensor(2, 3)
b = torch.Tensor(2)
c = torch.Tensor(2, 3)

def add_tensor(tensor):
    size = tuple(tensor.size())
    if size not in tensors_by_size:
        tensors_by_size[size] = set()
    tensors_by_size[size].add(tensor)

add_tensor(a)
add_tensor(b)
add_tensor(c)

def check_tensor(tensor):
    size = tuple(tensor.size())
    return size in tensors_by_size and tensor in tensors_by_size[size]

print(check_tensor(a)) # True
print(check_tensor(torch.Tensor(2,3))) # False 因为是新的tensor对象
print(check_tensor(b)) # True

通过这种方式,可以避免直接比较大小不同的张量,从而避免 RuntimeError。

相关专题

更多
python开发工具
python开发工具

php中文网为大家提供各种python开发工具,好的开发工具,可帮助开发者攻克编程学习中的基础障碍,理解每一行源代码在程序执行时在计算机中的过程。php中文网还为大家带来python相关课程以及相关文章等内容,供大家免费下载使用。

760

2023.06.15

python打包成可执行文件
python打包成可执行文件

本专题为大家带来python打包成可执行文件相关的文章,大家可以免费的下载体验。

639

2023.07.20

python能做什么
python能做什么

python能做的有:可用于开发基于控制台的应用程序、多媒体部分开发、用于开发基于Web的应用程序、使用python处理数据、系统编程等等。本专题为大家提供python相关的各种文章、以及下载和课程。

763

2023.07.25

format在python中的用法
format在python中的用法

Python中的format是一种字符串格式化方法,用于将变量或值插入到字符串中的占位符位置。通过format方法,我们可以动态地构建字符串,使其包含不同值。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

619

2023.07.31

python教程
python教程

Python已成为一门网红语言,即使是在非编程开发者当中,也掀起了一股学习的热潮。本专题为大家带来python教程的相关文章,大家可以免费体验学习。

1285

2023.08.03

python环境变量的配置
python环境变量的配置

Python是一种流行的编程语言,被广泛用于软件开发、数据分析和科学计算等领域。在安装Python之后,我们需要配置环境变量,以便在任何位置都能够访问Python的可执行文件。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

549

2023.08.04

python eval
python eval

eval函数是Python中一个非常强大的函数,它可以将字符串作为Python代码进行执行,实现动态编程的效果。然而,由于其潜在的安全风险和性能问题,需要谨慎使用。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

579

2023.08.04

scratch和python区别
scratch和python区别

scratch和python的区别:1、scratch是一种专为初学者设计的图形化编程语言,python是一种文本编程语言;2、scratch使用的是基于积木的编程语法,python采用更加传统的文本编程语法等等。本专题为大家提供scratch和python相关的文章、下载、课程内容,供大家免费下载体验。

709

2023.08.11

PHP WebSocket 实时通信开发
PHP WebSocket 实时通信开发

本专题系统讲解 PHP 在实时通信与长连接场景中的应用实践,涵盖 WebSocket 协议原理、服务端连接管理、消息推送机制、心跳检测、断线重连以及与前端的实时交互实现。通过聊天系统、实时通知等案例,帮助开发者掌握 使用 PHP 构建实时通信与推送服务的完整开发流程,适用于即时消息与高互动性应用场景。

11

2026.01.19

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新Python教程 从入门到精通
最新Python教程 从入门到精通

共4课时 | 5万人学习

Django 教程
Django 教程

共28课时 | 3.2万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号