Как отфильтровать словарь по произвольной условной функции? - PullRequest
174 голосов
/ 16 мая 2010

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

>>> 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 будет какая-то супер-классная dictionary.filter(f) функция ...

Ответы [ 7 ]

368 голосов
/ 16 мая 2013

В настоящее время, в 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}
106 голосов
/ 16 мая 2010
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 (в этом случае вы можете выразить то же ограничение с помощью and), но она будет работать нормально; -).

20 голосов
/ 16 мая 2010
points_small = dict(filter(lambda (a,(b,c)): b<5 and c < 5, points.items()))
10 голосов
/ 23 февраля 2015
>>> 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)}
9 голосов
/ 16 мая 2010
dict((k, v) for (k, v) in points.iteritems() if v[0] < 5 and v[1] < 5)
7 голосов
/ 16 мая 2013

Я думаю, что ответ Алекса Мартелли, безусловно, является самым элегантным способом сделать это, но я просто хотел добавить способ удовлетворить вашу потребность в супер удивительном 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() при деструктивной итерации вызовет исключение.

6 голосов
/ 16 мая 2010
dict((k, v) for (k, v) in points.iteritems() if v[0] < 5 and v[1] < 5)
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...