web-dev-qa-db-fra.com

Python: Trouver la valeur min, max dans une liste de tuples

alist = [(1,3),(2,5),(2,4),(7,5)]

Je dois obtenir la valeur min max pour chaque position dans Tuple.

Exemple Fox: La sortie attendue de alist est

min_x = 1
max_x = 7

min_y = 3
max_y = 5

Y a-t-il un moyen facile de faire?

26
user483144
map(max, Zip(*alist))

Cette première décompresse votre liste, puis trouve le maximum pour chaque position de tuple

>>> alist = [(1,3),(2,5),(2,4),(7,5)]
>>> Zip(*alist)
[(1, 2, 2, 7), (3, 5, 4, 5)]
>>> map(max, Zip(*alist))
[7, 5]
>>> map(min, Zip(*alist))
[1, 3]

Cela fonctionnera également pour les n-uplets de toute longueur dans une liste.

60
cobbal
>>> from operator import itemgetter
>>> alist = [(1,3),(2,5),(2,4),(7,5)]
>>> min(alist)[0], max(alist)[0]
(1, 7)
>>> min(alist, key=itemgetter(1))[1], max(alist, key=itemgetter(1))[1]
(3, 5)
8
John La Rooy

Au moins avec Python 2.7 , le "Zip" n’est pas nécessaire, donc cela se simplifie pour map(max, *data) (où data est un itérateur sur des n-uplets ou des listes de même longueur).

3
tiwo

Une approche généralisée ressemblerait à ceci:

alist = [(1,6),(2,5),(2,4),(7,5)]

temp = map(sorted, Zip(*alist))
min_x, max_x, min_y, max_y = temp[0][0], temp[0][-1], temp[1][0], temp[1][-1]

Pour Python 3, il vous faudrait changer la ligne qui crée tempto:

temp = Tuple(map(sorted, Zip(*alist)))

L'idée peut être résumée dans une fonction qui fonctionne à la fois en Python 2 et 3:

from __future__ import print_function
try:
    from functools import reduce  # moved into functools in release 2.6
except ImportError:
    pass

# readable version
def minmaxes(seq):
    pairs = Tuple()
    for s in map(sorted, Zip(*seq)):
        pairs += (s[0], s[-1])
    return pairs

# functional version
def minmaxes(seq):
    return reduce(Tuple.__add__, ((s[0], s[-1]) for s in map(sorted, Zip(*seq))))

alist = [(1,6), (2,5), (2,4), (7,5)]
min_x, max_x, min_y, max_y = minmaxes(alist)
print(' '.join(['{},{}']*2).format(*minmaxes(alist)))  # 1,7 4,6

triplets = [(1,6,6), (2,5,3), (2,4,9), (7,5,6)]
min_x, max_x, min_y, max_y, min_z, max_z = minmaxes(triplets)
print(' '.join(['{},{}']*3).format(*minmaxes(triplets)))  # 1,7 4,6 3,9
3
martineau

Une autre solution utilisant la compréhension énumérée et listée

alist = [(1,3),(2,5),(2,4),(7,5)]

for num, k in enumerate(['X', 'Y']):
    print 'max_%s' %k, max([i[num] for i in alist])
    print 'min_%s' %k, min([i[num] for i in alist])
0
toppare