草庐IT

如何从 Python 字典中删除键?

HuntsBot 2023-04-13 原文

问:

是否有一种在不引发 KeyError 的情况下从字典中删除键的单行方法?

if 'key' in my_dict:
    del my_dict['key']

答1:

huntsbot.com聚合了超过10+全球外包任务平台的外包需求,寻找外包任务与机会变的简单与高效。

要删除某个键(无论它是否在字典中),请使用 dict.pop() 的两个参数形式:

my_dict.pop('key', None)

如果字典中存在 key,则返回 my_dict[key],否则返回 None。如果未指定第二个参数(即 my_dict.pop(‘key’))且 key 不存在,则引发 KeyError。

要删除保证存在的键,您还可以使用:

del my_dict['key']

如果键不在字典中,这将引发 KeyError。

有时使用 pop() 而不是 del 的优势:它返回该键的值。这样,您可以在一行代码中从字典中获取和删除条目。

在问题中,不需要保留该值。这只会增加不必要的复杂性。 @zigg(下)的答案要好得多。

@SalvatoreCosentino 我无法理解你的论点。这个答案中的代码比另一个答案中的代码复杂吗?

@SalvatoreCosentino 不,忽略函数的返回值一点也不低效。恰恰相反 - 如果密钥不存在,此解决方案比 try/except 解决方案快得多。您可能会发现其中一个或另一个更易于阅读,这很好。两者都是惯用的 Python,因此请选择您喜欢的任何内容。但是声称这个答案更复杂或更低效是没有意义的。

@user5359531 我不明白。这是怎么回事? Python 内置类型的所有方法都没有返回 self,所以如果这个方法返回,那就太令人惊讶了。

答2:

huntsbot.com洞察每一个产品背后的需求与收益,从而捕获灵感

具体回答“有没有一种方法可以做到这一点?”

if 'key' in my_dict: del my_dict['key']

…好吧,你问了;-)

不过,您应该考虑,这种从 dict 中删除对象的方式是 not atomic — ‘key’ 可能在 if 语句期间位于 my_dict 中,但可能在 del 之前被删除} 被执行,在这种情况下 del 将失败并返回 KeyError。鉴于此,使用 use dict.pop 或类似的东西是最安全的

try:
    del my_dict['key']
except KeyError:
    pass

当然,这绝对不是单行的。

是的,pop 绝对更简洁,尽管这样做有一个关键优势:立即清楚它在做什么。

try/except 语句更昂贵。引发异常很慢。

@ChrisBarker 我发现如果密钥存在,try 会稍微快一些,但如果不存在,try 确实会慢很多。 pop 相当一致,但比使用不存在键的 try 慢。请参阅gist.github.com/zigg/6280653。最终,这取决于您期望密钥实际出现在字典中的频率,以及您是否需要原子性——当然,还取决于您是否进行过早优化;)

我认为不应该忽视清晰的价值。为此+1。

关于 try/except 的费用,您也可以使用 if 'key' in mydict: #then del...。我需要从字典中提取一个键/值来正确解析,pop 不是一个完美的解决方案。

答3:

huntsbot.com聚合了超过10+全球外包任务平台的外包需求,寻找外包任务与机会变的简单与高效。

我花了一些时间才弄清楚 my_dict.pop(“key”, None) 到底在做什么。因此,我将添加此作为答案以节省其他人的谷歌搜索时间:

pop(key[, default]) 如果 key 在字典中,删除它并返回它的值,否则返回默认值。如果未给出默认值且 key 不在字典中,则会引发 KeyError。

Documentation

只需在 python 解释器中输入 help(dict.pop)。

help() 和 dir() 当你需要知道某事做什么时,可以成为你的朋友。

或 IPython 中的 dict.pop?。

答4:

一个优秀的自由职业者,应该有对需求敏感和精准需求捕获的能力,而huntsbot.com提供了这个机会

del my_dict[key] 比 my_dict.pop(key) 在键存在时从字典中删除键稍快

>>> import timeit
>>> setup = "d = {i: i for i in range(100000)}"

>>> timeit.timeit("del d[3]", setup=setup, number=1)
1.79e-06
>>> timeit.timeit("d.pop(3)", setup=setup, number=1)
2.09e-06
>>> timeit.timeit("d2 = {key: val for key, val in d.items() if key != 3}", setup=setup, number=1)
0.00786

但是当密钥不存在时,if key in my_dict: del my_dict[key] 比 my_dict.pop(key, None) 稍快。在 try/except 语句中,两者都至少比 del 快三倍:

>>> timeit.timeit("if 'missing key' in d: del d['missing key']", setup=setup)
0.0229
>>> timeit.timeit("d.pop('missing key', None)", setup=setup)
0.0426
>>> try_except = """
... try:
...     del d['missing key']
... except KeyError:
...     pass
... """
>>> timeit.timeit(try_except, setup=setup)
0.133

@Boris - 这对于一般练习很有用。

@daisy 我的意思是,您应该选择最易读的语法,而不是快 300 纳秒的操作(这实际上是上面第一组时序中 del 和 pop 之间的区别)

这些操作也非常快,以至于这些时间不可靠。

答5:

一个优秀的自由职业者,应该有对需求敏感和精准需求捕获的能力,而huntsbot.com提供了这个机会

如果您需要在一行代码中从字典中删除很多键,我认为使用 map() 非常简洁且 Python 可读:

myDict = {'a':1,'b':2,'c':3,'d':4}
map(myDict.pop, ['a','c']) # The list of keys to remove
>>> myDict
{'b': 2, 'd': 4}

如果您需要捕获弹出不在字典中的值的错误,请在 map() 中使用 lambda,如下所示:

map(lambda x: myDict.pop(x,None), ['a', 'c', 'e'])
[1, 3, None] # pop returns
>>> myDict
{'b': 2, 'd': 4}

或在 python3 中,您必须改用列表推导:

[myDict.pop(x, None) for x in ['a', 'c', 'e']]

有用。即使 myDict 没有“e”键,“e”也不会导致错误。

这在 Python 3 中不起作用,因为 map 和朋友现在很懒惰并返回迭代器。将 map 用于副作用通常被认为是不好的做法;标准的 for ... in 循环会更好。有关详细信息,请参阅 Views And Iterators Instead Of Lists。

无论品味和实践风格如何,列表推导在 Py3 [myDict.pop(i, None) for i in ['a', 'c']] 中仍然可以工作,因为它们提供了 map(和 filter)的一般替代方案。

@MichaelEkoka 你不应该使用列表推导来解决它们的副作用,使用常规的 for ... in 循环。

@Boris 你可能是对的。我的回答特别与使用 map() 相关,因为它的副作用经常被使用。 Python 中推荐的替代方法是 列表推导式,在我看来,它仍然是非常易读的,并且作为单行代码在认知上很轻(参见问题)。仅用于它们的副作用,这两种结构确实会导致一个无用的列表,这可能是低效的。从 Python3 开始,我不知道有一个内置函数可以安全、优雅地遍历生成器表达式,而没有代价高昂的副产品,例如 loop(d.pop(k) for k in ['a', 'b'])。

@MichaelEkoka “内置函数”是一个 for 循环:for k in ['a', 'b']: d.pop(k)。为什么你需要一种不同的方式来做到这一点?如果您需要它只占用一行那么严重,那么您可以将 for 循环的第二部分放在同一行上。

答6:

huntsbot.com精选全球7大洲远程工作机会,涵盖各领域,帮助想要远程工作的数字游民们能更精准、更高效的找到对方。

您可以使用 dictionary comprehension 创建一个删除该键的新字典:

>>> my_dict = {k: v for k, v in my_dict.items() if k != 'key'}

可以按条件删除。如果 key 不存在,则不会出错。

这个答案与其他答案不同,因为它没有副作用(它不会改变原始字典)。

虽然这可能也是我会这样做的方式,但这会在内存中创建一个全新的字典,将对象复制(引用)到新字典。然后将其保存为旧名称。对于大型词典,这可能需要一些时间。 del dict[key] 或 dict.pop(key) 在所有情况下都会更快。

del 和 pop 更快,但有时您只是不想修改原始字典。如果在上面的示例中,将理解结果分配给另一个变量会更有意义。

答7:

huntsbot.com聚合了超过10+全球外包任务平台的外包需求,寻找外包任务与机会变的简单与高效。

使用“del”关键字:

del dict[key]

答8:

huntsbot.com洞察每一个产品背后的需求与收益,从而捕获灵感

我们可以通过以下一些方法从 Python 字典中删除一个键。

使用 del 关键字;这与您所做的几乎相同 -

 myDict = {'one': 100, 'two': 200, 'three': 300 }
 print(myDict)  # {'one': 100, 'two': 200, 'three': 300}
 if myDict.get('one') : del myDict['one']
 print(myDict)  # {'two': 200, 'three': 300}

或者

我们可以这样做:

但请记住,在此过程中,实际上它不会删除字典中的任何键,而是从该字典中排除特定键。此外,我观察到它返回的字典与 myDict 的排序不同。

myDict = {'one': 100, 'two': 200, 'three': 300, 'four': 400, 'five': 500}
{key:value for key, value in myDict.items() if key != 'one'}

如果我们在 shell 中运行它,它会执行类似 {‘five’: 500, ‘four’: 400, ‘three’: 300, ‘two’: 200} 的操作 - 请注意它与 myDict 的顺序不同。同样,如果我们尝试打印 myDict,那么我们可以看到所有键,包括我们通过这种方法从字典中排除的键。但是,我们可以通过将以下语句分配给变量来创建一个新字典:

var = {key:value for key, value in myDict.items() if key != 'one'}

现在,如果我们尝试打印它,那么它将遵循父顺序:

print(var) # {'two': 200, 'three': 300, 'four': 400, 'five': 500}

或者

使用 pop() 方法。

myDict = {'one': 100, 'two': 200, 'three': 300}
print(myDict)

if myDict.get('one') : myDict.pop('one')
print(myDict)  # {'two': 200, 'three': 300}

del 和 pop 的区别在于,使用 pop() 方法,如果需要,我们实际上可以存储 键的值,如下所示:

myDict = {'one': 100, 'two': 200, 'three': 300}
if myDict.get('one') : var = myDict.pop('one')
print(myDict) # {'two': 200, 'three': 300}
print(var)    # 100

Fork this gist 以供将来参考,如果您觉得这很有用。

huntsbot.com聚合了超过10+全球外包任务平台的外包需求,寻找外包任务与机会变的简单与高效。

不要使用 if myDict.get('one') 检查密钥是否存在!如果 myDict['one'] 有一个错误的值,它会失败。此外,dicts 没有固有的顺序,因此提及它是没有意义的。

@Rob 字典按从 CPython 3.6 开始的插入顺序和从 3.7 开始的所有其他 Python 实现进行排序。

答9:

huntsbot.com提供全网独家一站式外包任务、远程工作、创意产品分享与订阅服务!

如果您想非常详细,可以使用异常处理:

try: 
    del dict[key]

except KeyError: pass

但是,如果密钥不存在,这将比 pop() 方法慢。

my_dict.pop('key', None)

几个键无关紧要,但如果你反复这样做,那么后一种方法是更好的选择。

最快的方法是这样的:

if 'key' in dict: 
    del myDict['key']

但是这种方法很危险,因为如果在两行之间删除 ‘key’,则会引发 KeyError。

答10:

huntsbot.com – 高效赚钱,自由工作

字典 数据类型有一个名为 dict_name.pop(item) 的方法,可用于从字典中删除 key:value 对。

a={9:4,2:3,4:2,1:3}
a.pop(9)
print(a)

这将使输出为:

{2: 3, 4: 2, 1: 3}

这样,您可以在一行中从字典中删除一个项目。

答11:

保持自己快人一步,享受全网独家提供的一站式外包任务、远程工作、创意产品订阅服务–huntsbot.com

另一种方法是使用 items() + dict理解。

items() 与dict理解相结合也可以帮助我们完成键值对删除的任务,但它的缺点是不是一个到位的dict技术。实际上,如果创建了一个新的字典,除了我们不希望包含的键。

test_dict = {"sai" : 22, "kiran" : 21, "vinod" : 21, "sangam" : 21}

# Printing dictionary before removal
print ("dictionary before performing remove is : " + str(test_dict))

# Using items() + dict comprehension to remove a dict. pair
# removes  vinod
new_dict = {key:val for key, val in test_dict.items() if key != 'vinod'}

# Printing dictionary after removal
print ("dictionary after remove is : " + str(new_dict))

输出:

dictionary before performing remove is : {'sai': 22, 'kiran': 21, 'vinod': 21, 'sangam': 21}
dictionary after remove is : {'sai': 22, 'kiran': 21, 'sangam': 21}

这实际上并没有做所要求的 - 它创建一个新的字典,删除了不需要的键。此外,遍历所有键以删除一个键是在 O(N) 时间内执行 O(1) 操作。

原文链接:https://www.huntsbot.com/qa/JAnJ/how-can-i-remove-a-key-from-a-python-dictionary?lang=zh_CN&from=csdn

huntsbot.com – 程序员副业首选,一站式外包任务、远程工作、创意产品分享订阅平台。

有关如何从 Python 字典中删除键?的更多相关文章

  1. ruby - 如何使用 Nokogiri 的 xpath 和 at_xpath 方法 - 2

    我正在学习如何使用Nokogiri,根据这段代码我遇到了一些问题:require'rubygems'require'mechanize'post_agent=WWW::Mechanize.newpost_page=post_agent.get('http://www.vbulletin.org/forum/showthread.php?t=230708')puts"\nabsolutepathwithtbodygivesnil"putspost_page.parser.xpath('/html/body/div/div/div/div/div/table/tbody/tr/td/div

  2. ruby - 如何从 ruby​​ 中的字符串运行任意对象方法? - 2

    总的来说,我对ruby​​还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用

  3. python - 如何使用 Ruby 或 Python 创建一系列高音调和低音调的蜂鸣声? - 2

    关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。

  4. ruby-on-rails - 如何验证 update_all 是否实际在 Rails 中更新 - 2

    给定这段代码defcreate@upgrades=User.update_all(["role=?","upgraded"],:id=>params[:upgrade])redirect_toadmin_upgrades_path,:notice=>"Successfullyupgradeduser."end我如何在该操作中实际验证它们是否已保存或未重定向到适当的页面和消息? 最佳答案 在Rails3中,update_all不返回任何有意义的信息,除了已更新的记录数(这可能取决于您的DBMS是否返回该信息)。http://ar.ru

  5. ruby-on-rails - 'compass watch' 是如何工作的/它是如何与 rails 一起使用的 - 2

    我在我的项目目录中完成了compasscreate.和compassinitrails。几个问题:我已将我的.sass文件放在public/stylesheets中。这是放置它们的正确位置吗?当我运行compasswatch时,它不会自动编译这些.sass文件。我必须手动指定文件:compasswatchpublic/stylesheets/myfile.sass等。如何让它自动运行?文件ie.css、print.css和screen.css已放在stylesheets/compiled。如何在编译后不让它们重新出现的情况下删除它们?我自己编译的.sass文件编译成compiled/t

  6. ruby - 如何将脚本文件的末尾读取为数据文件(Perl 或任何其他语言) - 2

    我正在寻找执行以下操作的正确语法(在Perl、Shell或Ruby中):#variabletoaccessthedatalinesappendedasafileEND_OF_SCRIPT_MARKERrawdatastartshereanditcontinues. 最佳答案 Perl用__DATA__做这个:#!/usr/bin/perlusestrict;usewarnings;while(){print;}__DATA__Texttoprintgoeshere 关于ruby-如何将脚

  7. ruby - 如何指定 Rack 处理程序 - 2

    Rackup通过Rack的默认处理程序成功运行任何Rack应用程序。例如:classRackAppdefcall(environment)['200',{'Content-Type'=>'text/html'},["Helloworld"]]endendrunRackApp.new但是当最后一行更改为使用Rack的内置CGI处理程序时,rackup给出“NoMethodErrorat/undefinedmethod`call'fornil:NilClass”:Rack::Handler::CGI.runRackApp.newRack的其他内置处理程序也提出了同样的反对意见。例如Rack

  8. ruby - 如何每月在 Heroku 运行一次 Scheduler 插件? - 2

    在选择我想要运行操作的频率时,唯一的选项是“每天”、“每小时”和“每10分钟”。谢谢!我想为我的Rails3.1应用程序运行调度程序。 最佳答案 这不是一个优雅的解决方案,但您可以安排它每天运行,并在实际开始工作之前检查日期是否为当月的第一天。 关于ruby-如何每月在Heroku运行一次Scheduler插件?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/8692687/

  9. ruby-on-rails - 如何从 format.xml 中删除 <hash></hash> - 2

    我有一个对象has_many应呈现为xml的子对象。这不是问题。我的问题是我创建了一个Hash包含此数据,就像解析器需要它一样。但是rails自动将整个文件包含在.........我需要摆脱type="array"和我该如何处理?我没有在文档中找到任何内容。 最佳答案 我遇到了同样的问题;这是我的XML:我在用这个:entries.to_xml将散列数据转换为XML,但这会将条目的数据包装到中所以我修改了:entries.to_xml(root:"Contacts")但这仍然将转换后的XML包装在“联系人”中,将我的XML代码修改为

  10. ruby - 我可以使用 Ruby 从 CSV 中删除列吗? - 2

    查看Ruby的CSV库的文档,我非常确定这是可能且简单的。我只需要使用Ruby删除CSV文件的前三列,但我没有成功运行它。 最佳答案 csv_table=CSV.read(file_path_in,:headers=>true)csv_table.delete("header_name")csv_table.to_csv#=>ThenewCSVinstringformat检查CSV::Table文档:http://ruby-doc.org/stdlib-1.9.2/libdoc/csv/rdoc/CSV/Table.html

随机推荐