Удалить элемент из словаря, если его ключ неизвестен
Каков наилучший способ удалить элемент из словаря по значению, т.е. когда ключ элемента неизвестен? Здесь простой подход:
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]
Ответ 7
c - новый словарь, a - ваш исходный словарь, {'z', 'w'} - ключи, которые вы хотите удалить из
c = {key:a[key] for key in a.keys() - {'z', 'w'}}
Также проверьте: https://www.safaribooksonline.com/library/view/python-cookbook-3rd/9781449357337/ch01.html
Ответ 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