Удалить элемент из словаря, если его ключ неизвестен

Каков наилучший способ удалить элемент из словаря по значению, т.е. когда ключ элемента неизвестен? Здесь простой подход:

for key, item in some_dict.items():
    if item is item_to_remove:
        del some_dict[key]

Есть ли лучшие способы? Что-то не так с изменением (удалением элементов) из словаря при его повторении?

Ответ 1

Помните, что в настоящее время вы тестируете идентификацию объекта (is возвращает только True, если оба операнда представлены одним и тем же объектом в памяти - это не всегда происходит с двумя объектами, которые сравниваются с ==). Если вы делаете это специально, вы можете переписать свой код как

some_dict = {key: value for key, value in some_dict.items() 
             if value is not value_to_remove}

Но это может не делать то, что вы хотите:

>>> some_dict = {1: "Hello", 2: "Goodbye", 3: "You say yes", 4: "I say no"}
>>> value_to_remove = "You say yes"
>>> some_dict = {key: value for key, value in some_dict.items() if value is not value_to_remove}
>>> some_dict
{1: 'Hello', 2: 'Goodbye', 3: 'You say yes', 4: 'I say no'}
>>> some_dict = {key: value for key, value in some_dict.items() if value != value_to_remove}
>>> some_dict
{1: 'Hello', 2: 'Goodbye', 4: 'I say no'}

Итак, вы, вероятно, хотите != вместо is not.

Ответ 2

Метод dict.pop(key[, default]) позволяет вам удалять элементы, когда вы знаете ключ. Он возвращает значение в ключе, если удаляет элемент, в противном случае он возвращает то, что передано по default. Смотрите документы.

Пример:

>>> dic = {'a':1, 'b':2}
>>> dic
{'a': 1, 'b': 2}
>>> dic.pop('c', 0)
0
>>> dic.pop('a', 0)
1
>>> dic
{'b': 2}

Ответ 3

a = {'name': 'your_name','class': 4}
if 'name' in a: del a['name']

Ответ 4

Простое сравнение между del и pop():

import timeit
code = """
results = {'A': 1, 'B': 2, 'C': 3}
del results['A']
del results['B']
"""
print timeit.timeit(code, number=100000)
code = """
results = {'A': 1, 'B': 2, 'C': 3}
results.pop('A')
results.pop('B')
"""
print timeit.timeit(code, number=100000)

результат:

0.0329667857143
0.0451040902256

Итак, del быстрее, чем pop().

Ответ 5

items() возвращает список, и это тот список, который вы повторяете, поэтому мутация dict в цикле здесь не имеет значения. Если бы вы использовали iteritems() вместо этого, мутация dict в цикле была бы проблематичной, а также для viewitems() в Python 2.7.

Я не могу придумать лучший способ удалить элементы из dict по значению.

Ответ 6

Я бы создал список ключей, которые нужно удалить, а затем удалите их. Он прост, эффективен и позволяет избежать любой проблемы одновременного итерации и мутации dict.

keys_to_remove = [key for key, value in some_dict.iteritems()
                  if value == value_to_remove]
for key in keys_to_remove:
    del some_dict[key]

Ответ 8

Нет ничего плохого в удалении элементов из словаря во время итерации, как вы предлагали. Будьте внимательны в отношении нескольких потоков, используя один и тот же словарь одновременно, что может привести к возникновению KeyError или других проблем.

Конечно, см. документы в http://docs.python.org/library/stdtypes.html#typesmapping

Ответ 9

y={'username':'admin','machine':['a','b','c']}
if 'c' in y['machine'] : del y['machine'][y['machine'].index('c')]

Ответ 10

Вот как бы я это сделал.

for key in some_dict.keys():
    if some_dict[key] == item_to_remove:
        some_dict.pop(key)
        break