Combinar, contar e iterar con eficiencia

Cómo escribir código Python eficiente

Logan Thomas

Scientific Software Technical Trainer, Enthought

Resumen de Pokémon

  • Entrenadores (coleccionan Pokémon)

alt=”Ash Ketchum; uno de los muchos personajes (entrenadores) del videojuego Pokémon de Nintendo”

Cómo escribir código Python eficiente

Resumen de Pokémon

  • Pokémon (personajes animales ficticios)

alt=”Squirtle, Pikachu, Bulbasaur y Charmander; algunos de los muchos Pokémon del videojuego de Nintendo”

Cómo escribir código Python eficiente

Resumen de Pokémon

  • Pokédex (almacena Pokémon capturados)

alt=”Pokédex, herramienta que usan los entrenadores para guardar Pokémon capturados”

Cómo escribir código Python eficiente

Descripción de Pokémon

alt=”El Pokémon Squirtle con sus metadatos”

Cómo escribir código Python eficiente

Descripción de Pokémon

alt=”El Pokémon Squirtle con sus metadatos; campos Name y Generation resaltados”

Cómo escribir código Python eficiente

Descripción de Pokémon

alt=”El Pokémon Squirtle con sus metadatos; campos Type y Legendary resaltados”

Cómo escribir código Python eficiente

Descripción de Pokémon

alt=”El Pokémon Squirtle con sus metadatos; campos Health Points, Attack, Defense, Special Attack, Special Defense, Speed y Total resaltados”

Cómo escribir código Python eficiente

Combinar objetos

names = ['Bulbasaur', 'Charmander', 'Squirtle']
hps = [45, 39, 44]
combined = []

for i,pokemon in enumerate(names):
    combined.append((pokemon, hps[i]))

print(combined)
[('Bulbasaur', 45), ('Charmander', 39), ('Squirtle', 44)]
Cómo escribir código Python eficiente

Combinar con zip

names = ['Bulbasaur', 'Charmander', 'Squirtle']
hps = [45, 39, 44]
combined_zip = zip(names, hps)

print(type(combined_zip))
<class 'zip'>
combined_zip_list = [*combined_zip]

print(combined_zip_list)
[('Bulbasaur', 45), ('Charmander', 39), ('Squirtle', 44)]
Cómo escribir código Python eficiente

El módulo collections

  • Parte de la biblioteca estándar de Python (módulo integrado)
  • Tipos de contenedores especializados
    • Alternativas a dict, list, set y tuple de propósito general
  • Destaca:
    • namedtuple: subclases de tuple con campos con nombre
    • deque: contenedor tipo lista con inserciones/extracciones rápidas
    • Counter: dict para contar objetos hashables
    • OrderedDict: dict que conserva el orden de inserción
    • defaultdict: dict que llama a una fábrica para valores faltantes
Cómo escribir código Python eficiente

El módulo collections

  • Parte de la biblioteca estándar de Python (módulo integrado)
  • Tipos de contenedores especializados
    • Alternativas a dict, list, set y tuple de propósito general
  • Destaca:
    • namedtuple: subclases de tuple con campos con nombre
    • deque: contenedor tipo lista con inserciones/extracciones rápidas
    • Counter: dict para contar objetos hashables
    • OrderedDict: dict que conserva el orden de inserción
    • defaultdict: dict que llama a una fábrica para valores faltantes
Cómo escribir código Python eficiente

Contar con bucle

# Tipo de cada Pokémon (720 en total)
poke_types = ['Grass', 'Dark', 'Fire', 'Fire', ...]

type_counts = {}
for poke_type in poke_types: if poke_type not in type_counts: type_counts[poke_type] = 1 else: type_counts[poke_type] += 1
print(type_counts)
{'Rock': 41, 'Dragon': 25, 'Ghost': 20, 'Ice': 23, 'Poison': 28, 'Grass': 64,
 'Flying': 2, 'Electric': 40, 'Fairy': 17, 'Steel': 21, 'Psychic': 46, 'Bug': 65,
 'Dark': 28, 'Fighting': 25, 'Ground': 30, 'Fire': 48,'Normal': 92, 'Water': 105}
Cómo escribir código Python eficiente

collections.Counter()

# Tipo de cada Pokémon (720 en total)
poke_types = ['Grass', 'Dark', 'Fire', 'Fire', ...]

from collections import Counter
type_counts = Counter(poke_types)
print(type_counts)
Counter({'Water': 105, 'Normal': 92, 'Bug': 65, 'Grass': 64, 'Fire': 48,
         'Psychic': 46, 'Rock': 41, 'Electric': 40, 'Ground': 30,
         'Poison': 28, 'Dark': 28, 'Dragon': 25, 'Fighting': 25, 'Ice': 23,
         'Steel': 21, 'Ghost': 20, 'Fairy': 17, 'Flying': 2})
Cómo escribir código Python eficiente

El módulo itertools

  • Parte de la biblioteca estándar de Python (módulo integrado)
  • Herramientas funcionales para crear y usar iteradores
  • Destaca:
    • Iteradores infinitos: count, cycle, repeat
    • Iteradores finitos: accumulate, chain, zip_longest, etc.
    • Generadores de combinaciones: product, permutations, combinations
Cómo escribir código Python eficiente

El módulo itertools

  • Parte de la biblioteca estándar de Python (módulo integrado)
  • Herramientas funcionales para crear y usar iteradores
  • Destaca:
    • Iteradores infinitos: count, cycle, repeat
    • Iteradores finitos: accumulate, chain, zip_longest, etc.
    • Generadores de combinaciones: product, permutations, combinations
Cómo escribir código Python eficiente

Combinaciones con bucle

poke_types = ['Bug', 'Fire', 'Ghost', 'Grass', 'Water']

combos = [] for x in poke_types: for y in poke_types: if x == y: continue if ((x,y) not in combos) & ((y,x) not in combos): combos.append((x,y))
print(combos)
[('Bug', 'Fire'), ('Bug', 'Ghost'), ('Bug', 'Grass'), ('Bug', 'Water'),
 ('Fire', 'Ghost'), ('Fire', 'Grass'), ('Fire', 'Water'),
 ('Ghost', 'Grass'), ('Ghost', 'Water'), ('Grass', 'Water')]
Cómo escribir código Python eficiente

itertools.combinations()

poke_types = ['Bug', 'Fire', 'Ghost', 'Grass', 'Water']

from itertools import combinations
combos_obj = combinations(poke_types, 2)
print(type(combos_obj))
<class 'itertools.combinations'>
combos = [*combos_obj]
print(combos)
[('Bug', 'Fire'), ('Bug', 'Ghost'), ('Bug', 'Grass'), ('Bug', 'Water'),
 ('Fire', 'Ghost'), ('Fire', 'Grass'), ('Fire', 'Water'),
 ('Ghost', 'Grass'), ('Ghost', 'Water'), ('Grass', 'Water')]
Cómo escribir código Python eficiente

¡Vamos a practicar!

Cómo escribir código Python eficiente

Preparing Video For Download...