Python 字典理解 [重复]

2024-11-27 10:43:00
admin
原创
144
摘要:问题描述:是否有可能在 Python 中创建字典推导(针对键)?如果没有列表推导,你可以使用如下方法:l = [] for n in range(1, 11): l.append(n) 我们可以将其缩短为列表推导:l = [n for n in range(1, 11)]。但是,假设我想将字典的键设置...

问题描述:

是否有可能在 Python 中创建字典推导(针对键)?

如果没有列表推导,你可以使用如下方法:

l = []
for n in range(1, 11):
    l.append(n)

我们可以将其缩短为列表推导:l = [n for n in range(1, 11)]

但是,假设我想将字典的键设置为相同的值。我可以这样做:

d = {}
for n in range(1, 11):
     d[n] = True # same value for each

我尝试过这个:

d = {}
d[i for i in range(1, 11)] = True

但是,我得到SyntaxErrorfor

此外(我不需要这部分,只是想知道),您是否可以将字典的键设置为一堆不同的值,如下所示:

d = {}
for n in range(1, 11):
    d[n] = n

通过字典理解可以实现这一点吗?

d = {}
d[i for i in range(1, 11)] = [x for x in range(1, 11)]

这也引发了一个SyntaxError问题for


解决方案 1:

Python 2.7+ 中提供了字典推导,但它们的工作方式并不像您尝试的那样。与列表推导一样,它们会创建一个字典;您不能使用它们向现有字典添加键。此外,您必须指定键和值,当然,如果您愿意,也可以指定一个虚拟值。

>>> d = {n: n**2 for n in range(5)}
>>> print d
{0: 0, 1: 1, 2: 4, 3: 9, 4: 16}

如果你想将它们全部设置为 True:

>>> d = {n: True for n in range(5)}
>>> print d
{0: True, 1: True, 2: True, 3: True, 4: True}

您似乎想要的是一种在现有字典中同时设置多个键的方法。没有直接的捷径。您可以像已经展示的那样循环,也可以使用字典推导来创建一个具有新值的新字典,然后将oldDict.update(newDict)新值合并到旧字典中。

解决方案 2:

您可以使用dict.fromkeys类方法......

>>> dict.fromkeys(range(5), True)
{0: True, 1: True, 2: True, 3: True, 4: True}

这是创建所有键都映射到相同值的字典的最快方法。

但不要其与可变对象一起使用:

d = dict.fromkeys(range(5), [])
# {0: [], 1: [], 2: [], 3: [], 4: []}
d[1].append(2)
# {0: [2], 1: [2], 2: [2], 3: [2], 4: [2]} !!!

如果您实际上不需要初始化所有键,defaultdict那么也可能有用:

from collections import defaultdict
d = defaultdict(True)

为了回答第二部分,你所需要的就是一个字典理解:

{k: k for k in range(10)}

您可能不应该这样做,但您也可以创建一个子类dict,其工作方式类似于defaultdict如果您覆盖__missing__

>>> class KeyDict(dict):
...    def __missing__(self, key):
...       #self[key] = key  # Maybe add this also?
...       return key
... 
>>> d = KeyDict()
>>> d[1]
1
>>> d[2]
2
>>> d[3]
3
>>> print(d)
{}

解决方案 3:

>>> {i:i for i in range(1, 11)}
{1: 1, 2: 2, 3: 3, 4: 4, 5: 5, 6: 6, 7: 7, 8: 8, 9: 9, 10: 10}

解决方案 4:

我真的很喜欢@mgilson 的评论,因为如果你有两个可迭代对象,一个对应键,另一个对应值,你也可以执行以下操作。

keys = ['a', 'b', 'c']
values = [1, 2, 3]
d = dict(zip(keys, values))

给予

d = {'a': 1, 'b': 2, 'c': 3}

解决方案 5:

考虑一下这个使用字典理解来计算列表中单词出现次数的例子

my_list = ['hello', 'hi', 'hello', 'today', 'morning', 'again', 'hello']
my_dict = {k:my_list.count(k) for k in my_list}
print(my_dict)

结果是

{'again': 1, 'hi': 1, 'hello': 3, 'today': 1, 'morning': 1}

解决方案 6:

列表推导的主要目的是基于另一个列表创建一个新列表,而不改变或破坏原始列表。

而不是写作

l = []
for n in range(1, 11):
    l.append(n)

或者

l = [n for n in range(1, 11)]

你应该只写

l = range(1, 11)

在上面的两段代码中,你创建了一个新列表,遍历它并返回每个元素。这只是创建列表副本的一种昂贵方法。

要获取一个基于另一个字典且所有键都设置为相同值的新字典,请执行以下操作:

old_dict = {'a': 1, 'c': 3, 'b': 2}
new_dict = { key:'your value here' for key in old_dict.keys()}

你收到一个 SyntaxError,因为当你写

d = {}
d[i for i in range(1, 11)] = True

你基本上是在说:“将我的键‘i for i in range(1, 11)’设置为 True”,而“i for i in range(1, 11)”不是有效的键,这只是语法错误。如果 dicts 支持列表作为键,你可以这样做

d[[i for i in range(1, 11)]] = True

而不是

d[i for i in range(1, 11)] = True

但是列表不可哈希,所以不能将它们用作字典键。

解决方案 7:

在元组列表上使用 dict(),此解决方案允许您在每个列表中拥有任意值,只要它们的长度相同

i_s = range(1, 11)
x_s = range(1, 11)
# x_s = range(11, 1, -1) # Also works
d = dict([(i_s[index], x_s[index], ) for index in range(len(i_s))])

解决方案 8:

字典推导式与列表推导式非常相似,但是我们在字典末尾得到了一个字典,因此我们需要分配键值对而不是仅仅分配值。

假设我们有一个用户列表,其中每个用户的信息都存储在一个元组中。所以我们有一个包含四个用户元组的列表。其中,它们有一个 ID、每个用户的唯一标识号、一个用户名和一个密码。

因此,我们希望创建用户名与用户信息的映射。这是您经常会做的事情,尤其是在开发 Web 应用程序之类的东西时。

users = [
    (0, "Bob", "password"),
    (1, "code", "python"),
    (2, "Stack", "overflow"),
    (3, "username", "1234"),
]

username_mapping = {user[1]: user for user in users}
userid_mapping = {user[0]: user for user in users}

print(username_mapping)

"""
Why can this be helpful?

Well, imagine you know a user's username,and you want to get their information out.
You just access, let's say, "Bob," in your username_mapping, and you've got the information out.
"""
print(username_mapping["Bob"])  # (0, "Bob", "password")

# -- Can be useful to log in for example --

username_input = input("Enter your username: ")
password_input = input("Enter your password: ")

_, username, password = username_mapping[username_input]

if password_input == password:
    print("Your details are correct!")
else:
    print("Your details are incorrect.")

这是使用此结构(即字典理解)执行某种登录的一个例子。

这确实很有用,因为它可以节省您在这里执行另一个 for 循环的时间,以确保您使用正确的用户名进行输入。

解决方案 9:

你不能对这样的列表进行哈希处理。试试这个,它使用元组

d[tuple([i for i in range(1,11)])] = True
相关推荐
  政府信创国产化的10大政策解读一、信创国产化的背景与意义信创国产化,即信息技术应用创新国产化,是当前中国信息技术领域的一个重要发展方向。其核心在于通过自主研发和创新,实现信息技术应用的自主可控,减少对外部技术的依赖,并规避潜在的技术制裁和风险。随着全球信息技术竞争的加剧,以及某些国家对中国在科技领域的打压,信创国产化显...
工程项目管理   1565  
  为什么项目管理通常仍然耗时且低效?您是否还在反复更新电子表格、淹没在便利贴中并参加每周更新会议?这确实是耗费时间和精力。借助软件工具的帮助,您可以一目了然地全面了解您的项目。如今,国内外有足够多优秀的项目管理软件可以帮助您掌控每个项目。什么是项目管理软件?项目管理软件是广泛行业用于项目规划、资源分配和调度的软件。它使项...
项目管理软件   1354  
  信创国产芯片作为信息技术创新的核心领域,对于推动国家自主可控生态建设具有至关重要的意义。在全球科技竞争日益激烈的背景下,实现信息技术的自主可控,摆脱对国外技术的依赖,已成为保障国家信息安全和产业可持续发展的关键。国产芯片作为信创产业的基石,其发展水平直接影响着整个信创生态的构建与完善。通过不断提升国产芯片的技术实力、产...
国产信创系统   21  
  信创生态建设旨在实现信息技术领域的自主创新和安全可控,涵盖了从硬件到软件的全产业链。随着数字化转型的加速,信创生态建设的重要性日益凸显,它不仅关乎国家的信息安全,更是推动产业升级和经济高质量发展的关键力量。然而,在推进信创生态建设的过程中,面临着诸多复杂且严峻的挑战,需要深入剖析并寻找切实可行的解决方案。技术创新难题技...
信创操作系统   27  
  信创产业作为国家信息技术创新发展的重要领域,对于保障国家信息安全、推动产业升级具有关键意义。而国产芯片作为信创产业的核心基石,其研发进展备受关注。在信创国产芯片的研发征程中,面临着诸多复杂且艰巨的难点,这些难点犹如一道道关卡,阻碍着国产芯片的快速发展。然而,科研人员和相关企业并未退缩,积极探索并提出了一系列切实可行的解...
国产化替代产品目录   28  
热门文章
项目管理软件有哪些?
云禅道AD
禅道项目管理软件

云端的项目管理软件

尊享禅道项目软件收费版功能

无需维护,随时随地协同办公

内置subversion和git源码管理

每天备份,随时转为私有部署

免费试用