集美阅读大全是一个以文章句子为主题的在线阅读网站。内含有各种经典好文章,爱情美文,诗歌散文,情感句子说说,范文资料等。读好文章,尽在集美阅读大全!!!
当前位置:集美阅读大全 >互联网 >建站 > 正文

什么是 Python 集合?为什么以及如何使用它?

Python 配备了几种内置数据类型来帮我们组织数据。这些结构包括列表、字典、元组和集合。

什么是 Python 集合?为什么以及如何使用它?

根据 Python 3 文档:

集合是一个无序集合,没有重复元素。基本用途包括成员测试和消除重复的条目。集合对象还支持数学运算,如并集、交集、差集和对等差分。

在本文中,我们将回顾并查看上述定义中列出的每个要素的示例。让我们马上开始,看看如何创建它。

 

初始化一个集合

有两种方法可以创建一个集合:一个是给内置函数 set() 提供一个元素列表,另一个是使用花括号 {}

1、使用内置函数 set() 来初始化一个集合:

  1. >>> s1 = set([1, 2, 3])
  2. >>> s1
  3. {1, 2, 3}
  4. >>> type(s1)
  5. <class 'set'>

2、使用花括号初始化集合 {}

  1. >>> s2 = {3, 4, 5}
  2. >>> s2
  3. {3, 4, 5}
  4. >>> type(s2)
  5. <class 'set'>
  6. >>>

如你所见,这两种方法都是有效的。但问题是,如果我们想要一个空的集合呢?

  1. >>> s = {}
  2. >>> type(s)
  3. <class 'dict'>

没错,如果我们使用空花括号,我们将得到一个字典而不是一个集合。=)

值得一提的是,为了简单起见,本文中提供的所有示例都将使用整数集合,但集合可以包含 Python 支持的所有 可哈希的 数据类型。换句话说,即整数、字符串和元组,而不是列表或字典这样的可变类型。

  1. >>> s = {1, 'coffee', [4, 'python']}
  2. Traceback (most recent call last):
  3.   File "<stdin>", line 1, in <module>
  4. TypeError: unhashable type: 'list'

既然你知道了如何创建一个集合以及它可以包含哪些类型的元素,那么让我们继续看看为什么我们总是应该把它放在我们的工具箱中。

 

为什么你需要使用它

写代码时,你可以用不止一种方法来完成它。有些被认为是相当糟糕的,另一些则是清晰的、简洁的和可维护的,或者是 “Python 式的”。

根据 Hitchhiker 对 Python 的建议:

当一个经验丰富的 Python 开发人员(Python 人)调用一些不够 “Python 式的” 的代码时,他们通常认为着这些代码不遵循通用指南,并且无法被认为是以一种好的方式(可读性)来表达意图。

让我们开始探索 Python 集合那些不仅可以帮助我们提高可读性,还可以加快程序执行时间的方式。

 

无序的集合元素

首先你需要明白的是:你无法使用索引访问集合中的元素。

  1. >>> s = {1, 2, 3}
  2. >>> s[0]
  3. Traceback (most recent call last):
  4.   File "<stdin>", line 1, in <module>
  5. TypeError: 'set' object does not support indexing

或者使用切片修改它们:

  1. >>> s[0:2]
  2. Traceback (most recent call last):
  3.   File "<stdin>", line 1, in <module>
  4. TypeError: 'set' object is not subscriptable

但是,如果我们需要删除重复项,或者进行组合列表(与)之类的数学运算,那么我们可以,并且应该始终使用集合。

我不得不提一下,在迭代时,集合的表现优于列表。所以,如果你需要它,那就加深对它的喜爱吧。为什么?好吧,这篇文章并不打算解释集合的内部工作原理,但是如果你感兴趣的话,这里有几个链接,你可以阅读它:

  • 时间复杂度
  • set() 是如何实现的?
  • Python 集合 vs 列表
  • 在列表中使用集合是否有任何优势或劣势,以确保独一无二的列表条目?

 

没有重复项

写这篇文章的时候,我总是不停地思考,我经常使用 for 循环和 if 语句检查并删除列表中的重复元素。记得那时我的脸红了,而且不止一次,我写了类似这样的代码:

  1. >>> my_list = [1, 2, 3, 2, 3, 4]
  2. >>> no_duplicate_list = []
  3. >>> for item in my_list:
  4. ...     if item not in no_duplicate_list:
  5. ...             no_duplicate_list.append(item)
  6. ...
  7. >>> no_duplicate_list
  8. [1, 2, 3, 4]

或者使用列表解析:

  1. >>> my_list = [1, 2, 3, 2, 3, 4]
  2. >>> no_duplicate_list = []
  3. >>> [no_duplicate_list.append(item) for item in my_list if item not in no_duplicate_list]
  4. [None, None, None, None]
  5. >>> no_duplicate_list
  6. [1, 2, 3, 4]

但没关系,因为我们现在有了武器装备,没有什么比这更重要的了:

  1. >>> my_list = [1, 2, 3, 2, 3, 4]
  2. >>> no_duplicate_list = list(set(my_list))
  3. >>> no_duplicate_list
  4. [1, 2, 3, 4]
  5. >>>

现在让我们使用 timeit 模块,查看列表和集合在删除重复项时的执行时间:

  1. >>> from timeit import timeit
  2. >>> def no_duplicates(list):
  3. ...     no_duplicate_list = []
  4. ...     [no_duplicate_list.append(item) for item in list if item not in no_duplicate_list]
  5. ...     return no_duplicate_list
  6. ...
  7. >>> # first, let's see how the list perform:
  8. >>> print(timeit('no_duplicates([1, 2, 3, 1, 7])', globals=globals(), number=1000))
  9. 0.0018683355819786227
  1. >>> from timeit import timeit
  2. >>> # and the set:
  3. >>> print(timeit('list(set([1, 2, 3, 1, 2, 3, 4]))', number=1000))
  4. 0.0010220493243764395
  5. >>> # faster and cleaner =)

使用集合而不是列表推导不仅让我们编写更少的代码,而且还能让我们获得更具可读性高性能的代码。

注意:请记住集合是无序的,因此无法保证在将它们转换回列表时,元素的顺序不变。

 

Python 之禅:

优美胜于丑陋

明了胜于晦涩

简洁胜于复杂

扁平胜于嵌套

集合不正是这样美丽、明了、简单且扁平吗?

 

成员测试

每次我们使用 if 语句来检查一个元素,例如,它是否在列表中时,意味着你正在进行成员测试:

  1. my_list = [1, 2, 3]
  2. >>> if 2 in my_list:
  3. ...     print('Yes, this is a membership test!')
  4. ...
  5. Yes, this is a membership test!

在执行这些操作时,集合比列表更高效:

  1. >>> from timeit import timeit
  2. >>> def in_test(iterable):
  3. ...     for i in range(1000):
  4. ...             if i in iterable:
  5. ...                     pass
  6. ...
  7. >>> timeit('in_test(iterable)',
  8. ... setup="from __main__ import in_test; iterable = list(range(1000))",
  9. ... number=1000)
  10. 12.459663048726043
  1. >>> from timeit import timeit
  2. >>> def in_test(iterable):
  3. ...     for i in range(1000):
  4. ...             if i in iterable:
  5. ...                     pass
  6. ...
  7. >>> timeit('in_test(iterable)',
  8. ... setup="from __main__ import in_test; iterable = set(range(1000))",
  9. ... number=1000)
  10. 0.12354438152988223
  11. >>>

注意:上面的测试来自于这个 StackOverflow 话题。

因此,如果你在巨大的列表中进行这样的比较,尝试将该列表转换为集合,它应该可以加快你的速度。

 

如何使用

现在你已经了解了集合是什么以及为什么你应该使用它,现在让我们快速浏览一下,看看我们如何修改和操作它。

 

添加元素

根据要添加的元素数量,我们要在 add() 和 update() 方法之间进行选择。

add() 适用于添加单个元素:

  1. >>> s = {1, 2, 3}
  2. >>> s.add(4)
  3. >>> s
  4. {1, 2, 3, 4}

update() 适用于添加多个元素:

  1. >>> s = {1, 2, 3}
  2. >>> s.update([2, 3, 4, 5, 6])
  3. >>> s
  4. {1, 2, 3, 4, 5, 6}

请记住,集合会移除重复项。

 

移除元素

如果你希望在代码中尝试删除不在集合中的元素时收到警报,请使用 remove()。否则,discard() 提供了一个很好的选择:

  1. >>> s = {1, 2, 3}
  2. >>> s.remove(3)
  3. >>> s
  4. {1, 2}
  5. >>> s.remove(3)
  6. Traceback (most recent call last):
  7.   File "<stdin>", line 1, in <module>
  8. KeyError: 3

discard() 不会引起任何错误:

  1. >>> s = {1, 2, 3}
  2. >>> s.discard(3)
  3. >>> s
  4. {1, 2}
  5. >>> s.discard(3)
  6. >>> # nothing happens!

我们也可以使用 pop() 来随机丢弃一个元素:

  1. >>> s = {1, 2, 3, 4, 5}
  2. >>> s.pop()  # removes an arbitrary element
  3. 1
  4. >>> s
  5. {2, 3, 4, 5}

或者 clear() 方法来清空一个集合:

  1. >>> s = {1, 2, 3, 4, 5}
  2. >>> s.clear()  # discard all the items
  3. >>> s
  4. set()

 

union()

union() 或者 | 将创建一个新集合,其中包含我们提供集合中的所有元素:

  1. >>> s1 = {1, 2, 3}
  2. >>> s2 = {3, 4, 5}
  3. >>> s1.union(s2)  # or 's1 | s2'
  4. {1, 2, 3, 4, 5}

 

intersection()

intersection 或 & 将返回一个由集合共同元素组成的集合:

  1. >>> s1 = {1, 2, 3}
  2. >>> s2 = {2, 3, 4}
  3. >>> s3 = {3, 4, 5}
  4. >>> s1.intersection(s2, s3)  # or 's1 & s2 & s3'
  5. {3}

 

difference()

使用 diference() 或 - 创建一个新集合,其值在 “s1” 中但不在 “s2” 中:

  1. >>> s1 = {1, 2, 3}
  2. >>> s2 = {2, 3, 4}
  3. >>> s1.difference(s2)  # or 's1 - s2'
  4. {1}

 

symmetric_diference()

symetric_difference 或 ^ 将返回集合之间的不同元素。

  1. >>> s1 = {1, 2, 3}
  2. >>> s2 = {2, 3, 4}
  3. >>> s1.symmetric_difference(s2)  # or 's1 ^ s2'
  4. {1, 4}

 

结论

我希望在阅读本文之后,你会知道集合是什么,如何操纵它的元素以及它可以执行的操作。知道何时使用集合无疑会帮助你编写更清晰的代码并加速你的程序。

如果你有任何疑问,请发表评论,我很乐意尝试回答。另外,不要忘记,如果你已经理解了集合,它们在 Python Cheatsheet 中有自己的一席之地,在那里你可以快速参考并重新认知你已经知道的内容。

80%的人都看过的文章

  • Python 或将成为法国高中的官方编程教学语言
  • PHP 5.6.38、7.0.32、7.1.22和7.2.10发布
  • Python 3.7.1 和 3.6.7 正式发布
  • PHP 5.6.40, 7.1.26, 7.3.1 & 7.2.14 安全修复版本发布
  • 微软推出 Python 免费在线视频教程
  • Python新手常犯的10个错误
  • Stack Overflow调查:哪门编程语言最赚钱?
  • 7月PYPL IDE指数榜Visual Studio Code 击败 Xcode

课后答案张九龄《望月怀远》阅读答案及全诗翻译赏析

望月怀远张九龄海上生明月,天涯共此时。情人怨遥夜,竟夕起相思。灭烛怜光满,披衣觉露滋。不堪盈手赠,还寝梦佳期。注释⑴怀远:怀念远方的亲人。⑵最前面两句:辽阔无边的大海上升起一轮明月,使人想起了远在天涯……
2023-11-22 04:53暂无评论阅读详情

课后答案王安石《次韵唐公三首其三旅思》阅读答案

次韵唐公三首其三旅思王安石此身南北老,愁见问征途。地大蟠三楚,天低入五湖。看云心共远,步月影同孤。慷慨秋风起,悲歌不为鲈②。注:①张壤,字唐公,北宋嘉佑六年契丹国母生辰使,王安石友人。②《晋书&mid……
2023-11-22 04:52暂无评论阅读详情

笔记心得各级干部学习执法为民心得体会

  &ldquo;各级干部都要牢固树立全心全意为人民服务的思想和真心实意对人民负责的精神,做到心里装着群众,凡事想着群众,工作依靠群众,一切为了群众。要坚持权为民所用,情为民所系,利为民所谋,为群众诚……
2023-11-22 04:12暂无评论阅读详情

笔记心得寒假大学生社会实践心得体会

  自从走进了大学,就业问题就似乎总是围绕在我们的身边,成了说不完的话题。在现今社会,招聘会上的大字报都总写着&ldquo;有经验者优先&rdquo;,可还在校园里面的我们这班学子社会经验又会拥有多少……
2023-11-22 04:08暂无评论阅读详情

协议书济南市某美容院转让协议第2篇

&nbsp;&nbsp;__________美容院根据中华人民共和国国务院劳动法规和________市私营企业劳动管理实施办法,结合本美容院经营的具体所需今制订此劳动合同书。&nbsp;&nbsp;双……
2023-11-22 02:36暂无评论阅读详情

剧本劳模宣传短剧剧本《阿咪也想当劳模》

  1、机械厂门卫处,日,外。  清早,机械厂班长李玉伟开着别克赛欧小汽车驶进厂区,门卫室内的保安一边按开电动门,一边朝李玉伟摆手。  李玉伟:(摇下车窗,笑着打招呼)小秦,早。  保安小秦:(笑着)……
2023-11-22 02:11暂无评论阅读详情

教程灰雀说课稿

灰雀说课稿  灰雀说课稿(一):  《灰雀》说课稿  一、说教材  《灰雀》是义务教育课程标准实验教科书,小学语文第五册第二单元的一篇讲读课文。这篇课文记叙了列宁在莫斯科郊外养病期间爱护灰雀的故事。列……
2023-11-22 00:41暂无评论阅读详情

课件“吴隐之字处默,濮阳鄄城人”阅读答案及原文

吴隐之字处默,濮阳鄄城人。美姿容,善谈论,博涉文史,以儒雅标名。弱冠而介立,有清操,虽儋石无储,不取非其道。事母孝谨,及其执丧,哀毁过礼。与太常韩康伯邻居,康伯母,贤明妇人也,每闻隐之哭声,辍餐投箸,……
2023-11-22 00:38暂无评论阅读详情

标签