Как фильтровать словарь в соответствии с произвольной функцией условия?

У меня есть словарь точек, скажем:

>>> points={'a':(3,4), 'b':(1,2), 'c':(5,5), 'd':(3,3)} 

Я хочу создать новый словарь со всеми точками, чьи значения x и y меньше 5, то есть точки «a», «b» и «d».

Согласно книге , каждый словарь имеет функцию items() , которая возвращает список (key, pair) кортежей:

 >>> points.items() [('a', (3, 4)), ('c', (5, 5)), ('b', (1, 2)), ('d', (3, 3))] 

Поэтому я написал это:

 >>> for item in [i for i in points.items() if i[1][0]<5 and i[1][1]<5]: ... points_small[item[0]]=item[1] ... >>> points_small {'a': (3, 4), 'b': (1, 2), 'd': (3, 3)} 

Есть ли более элегантный способ? Я ожидал, что Python будет иметь супер-awesome dictionary.filter(f) функцию …

7 Solutions collect form web for “Как фильтровать словарь в соответствии с произвольной функцией условия?”

В настоящее время в Python 2.7 и выше вы можете использовать понимание dict:

 {k: v for k, v in points.iteritems() if v[0] < 5 and v[1] < 5} 

И в Python 3:

 {k: v for k, v in points.items() if v[0] < 5 and v[1] < 5} 
 dict((k, v) for k, v in points.items() if all(x < 5 for x in v)) 

Вы можете выбрать вызов .iteritems() вместо .items() если вы находитесь в Python 2, и points могут иметь много записей.

all(x < 5 for x in v) могут быть излишними, если вы точно знаете, что каждая точка всегда будет только 2D (в этом случае вы можете выразить одно и то же ограничение с помощью a and ), но она будет работать нормально ;-).

 points_small = dict(filter(lambda (a,(b,c)): b<5 and c < 5, points.items())) 
 dict((k, v) for (k, v) in points.iteritems() if v[0] < 5 and v[1] < 5) 

Я думаю, что ответ Alex Martelli определенно является самым изящным способом сделать это, но просто хотел добавить способ удовлетворить ваши потребности в супер удивительном dictionary.filter(f) метод в стиле Pythonic:

 class FilterDict(dict): def __init__(self, input_dict): for key, value in input_dict.iteritems(): self[key] = value def filter(self, criteria): for key, value in self.items(): if (criteria(value)): self.pop(key) my_dict = FilterDict( {'a':(3,4), 'b':(1,2), 'c':(5,5), 'd':(3,3)} ) my_dict.filter(lambda x: x[0] < 5 and x[1] < 5) 

В основном мы создаем класс, который наследует от dict , но добавляет метод фильтра. Нам нужно использовать .items() для фильтрации, так как использование .iteritems() то время как разрушающая итерация вызовет исключение.

 >>> points = {'a': (3, 4), 'c': (5, 5), 'b': (1, 2), 'd': (3, 3)} >>> dict(filter(lambda x: (x[1][0], x[1][1]) < (5, 5), points.items())) {'a': (3, 4), 'b': (1, 2), 'd': (3, 3)} 
 dict((k, v) for (k, v) in points.iteritems() if v[0] < 5 and v[1] < 5) 
  • Python - найти самый длинный (большинство слов) ключ в словаре
  • различия между «d = dict ()» и «d = {}»
  • Ввод данных в словарь
  • Python обратное / обратное отображение (но с несколькими значениями для каждого ключа)
  • Python, как сохранить ключи / значения в том же порядке, что и объявлено?
  • Вывод / печать "читаемого" словаря
  • Рекурсивный факториал с использованием dict вызывает RecursionError
  • Python Sqlite3: INSERT INTO table VALUE (словарь идет здесь)
  • Python - лучший язык программирования в мире.