Convertir une liste d'objets en une liste d'entiers et une table de correspondance

StackOverflow https://stackoverflow.com/questions/1401721

  •  05-07-2019
  •  | 
  •  

Question

Pour illustrer ce que je veux dire par ceci, voici un exemple

messages = [
  ('Ricky',  'Steve',  'SMS'),
  ('Steve',  'Karl',   'SMS'),
  ('Karl',   'Nora',   'Email')
]

Je souhaite convertir cette liste et une définition de groupes en une liste d'entiers et un dictionnaire de références afin que chaque élément du groupe reçoive un identifiant unique. Cet identifiant doit correspondre à l'élément dans la table de recherche comme ceci

messages_int, lookup_table = create_lookup_list(
              messages, ('person', 'person', 'medium'))

print messages_int
[ (0, 1, 0),
  (1, 2, 0),
  (2, 3, 1) ]

print lookup_table
{ 'person': ['Ricky', 'Steve', 'Karl', 'Nora'],
  'medium': ['SMS', 'Email']
}

Je me demande s’il existe une solution élégante et pythonique à ce problème.

Je suis également ouvert à une meilleure terminologie que create_lookup_list etc.

Était-ce utile?

La solution

La combinaison de

defaultdict et de la méthode itertools.count (). next est un bon moyen d'attribuer des identificateurs à des éléments uniques. Voici un exemple d'application de ce principe à votre cas:

from itertools import count
from collections import defaultdict

def create_lookup_list(data, domains):
    domain_keys = defaultdict(lambda:defaultdict(count().next))
    out = []
    for row in data:
        out.append(tuple(domain_keys[dom][val] for val, dom in zip(row, domains)))
    lookup_table = dict((k, sorted(d, key=d.get)) for k, d in domain_keys.items())
    return out, lookup_table

Modifier: notez que count (). next devient count () .__ next __ ou lambda: next (count ()) en Python 3.

Autres conseils

Le mien a à peu près la même longueur et la même complexité:

import collections

def create_lookup_list(messages, labels):

    # Collect all the values
    lookup = collections.defaultdict(set)
    for msg in messages:
        for l, v in zip(labels, msg):
            lookup[l].add(v)

    # Make the value sets lists
    for k, v in lookup.items():
        lookup[k] = list(v)

    # Make the lookup_list
    lookup_list = []
    for msg in messages:
        lookup_list.append([lookup[l].index(v) for l, v in zip(labels, msg)])

    return lookup_list, lookup

Dans la réponse d'Otto (ou de quelqu'un d'autre avec des dictes de chaîne- > id), je remplacerais (si obsédé par la vitesse est votre truc):

# create the lookup table
lookup_dict = {}
for group in indices:
    lookup_dict[group] = sorted(indices[group].keys(),
            lambda e1, e2: indices[group][e1]-indices[group][e2])

par

# k2i must map keys to consecutive ints [0,len(k2i)-1)
def inverse_indices(k2i):
    inv=[0]*len(k2i)
    for k,i in k2i.iteritems():
        inv[i]=k
    return inv

lookup_table = dict((g,inverse_indices(gi)) for g,gi in indices.iteritems()) 

Cela est préférable, car l'attribution directe à chaque élément du tableau inverse est plus rapide que le tri.

Voici ma propre solution - je doute que ce soit la meilleure

def create_lookup_list(input_list, groups):
    # use a dictionary for the indices so that the index lookup 
    # is fast (not necessarily a requirement)
    indices = dict((group, {}) for group in groups) 
    output = []

    # assign indices by iterating through the list
    for row in input_list:
        newrow = []
        for group, element in zip(groups, row):
            if element in indices[group]:
                index = indices[group][element]
            else:
                index = indices[group][element] = len(indices[group])
            newrow.append(index)
        output.append(newrow)

    # create the lookup table
    lookup_dict = {}
    for group in indices:
        lookup_dict[group] = sorted(indices[group].keys(),
                lambda e1, e2: indices[group][e1]-indices[group][e2])

    return output, lookup_dict

C’est un peu plus simple et plus direct.

from collections import defaultdict

def create_lookup_list( messages, schema ):
    def mapped_rows( messages ):
        for row in messages:
            newRow= []
            for col, value in zip(schema,row):
                if value not in lookups[col]:
                    lookups[col].append(value)
                code= lookups[col].index(value)
                newRow.append(code)
            yield newRow
    lookups = defaultdict(list)
    return list( mapped_rows(messages) ), dict(lookups)  

Si les recherches étaient des dictionnaires appropriés, et non des listes, cela pourrait être simplifié davantage.
Créez votre " table de consultation " avoir la structure suivante

{ 'person': {'Ricky':0, 'Steve':1, 'Karl':2, 'Nora':3},
  'medium': {'SMS':0, 'Email':1}
}

Et la complexité peut encore être réduite.

Vous pouvez transformer cette copie de travail des recherches en son inverse comme suit:

>>> lookups = { 'person': {'Ricky':0, 'Steve':1, 'Karl':2, 'Nora':3},
      'medium': {'SMS':0, 'Email':1}
    }
>>> dict( ( d, dict( (v,k) for k,v in lookups[d].items() ) ) for d in lookups )
{'person': {0: 'Ricky', 1: 'Steve', 2: 'Karl', 3: 'Nora'}, 'medium': {0: 'SMS', 1: 'Email'}}

Voici ma solution, ce n'est pas mieux - c'est juste différent:)

def create_lookup_list(data, keys):
  encoded = []
  table = dict([(key, []) for key in keys])

  for record in data:
      msg_int = []
      for key, value in zip(keys, record):
          if value not in table[key]:
              table[key].append(value)
          msg_int.append(table[key].index(value))  
      encoded.append(tuple(msg_int))

  return encoded, table

Voici la mienne, la fonction interne me permet d’écrire le tuple d’index comme générateur.

def create_lookup_list( data, format):
    table = {}
    indices = []
    def get_index( item, form ):
        row = table.setdefault( form, [] )
        try:
            return row.index( item )
        except ValueError:
            n = len( row )
            row.append( item )
            return n
    for row in data:
        indices.append( tuple( get_index( item, form ) for item, form in zip( row, format ) ))

    return table, indices
Licencié sous: CC-BY-SA avec attribution
Non affilié à StackOverflow
scroll top