TAD de los conjuntos: Transformaciones entre conjuntos y listas
Utilizando el tipo abstracto de datos de los conjuntos definir las funciones
1 2 |
listaAconjunto :: [a] -> Conj a conjuntoAlista :: Conj a -> [a] |
tales que
+ listaAconjunto xs
es el conjunto formado por los elementos de xs
. Por ejemplo,
1 2 |
λ> listaAconjunto [3, 2, 5] {2, 3, 5} |
conjuntoAlista c
es la lista formada por los elementos del conjuntoc
. Por ejemplo,
1 2 |
λ> conjuntoAlista (inserta 5 (inserta 2 (inserta 3 vacio))) [2,3,5] |
Comprobar con QuickCheck que ambas funciones son inversa; es decir,
1 2 |
conjuntoAlista (listaAconjunto xs) = sort (nub xs) listaAconjunto (conjuntoAlista c) = c |
Soluciones
A continuación se muestran las soluciones en Haskell y las soluciones en Python.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 |
import TAD.Conjunto (Conj, vacio, inserta, menor, elimina, pertenece, esVacio) import Data.List (sort, nub) import Test.QuickCheck -- 1ª definición de listaAconjunto -- =============================== listaAconjunto :: Ord a => [a] -> Conj a listaAconjunto [] = vacio listaAconjunto (x:xs) = inserta x (listaAconjunto xs) -- 2ª definición de listaAconjunto -- =============================== listaAconjunto2 :: Ord a => [a] -> Conj a listaAconjunto2 = foldr inserta vacio -- Comprobación de equivalencia -- ============================ -- La propiedad es prop_listaAconjunto :: [Int] -> Bool prop_listaAconjunto xs = listaAconjunto xs == listaAconjunto2 xs -- La comprobación es -- λ> quickCheck prop_listaAconjunto -- +++ OK, passed 100 tests. -- Definición de conjuntoAlista -- ============================ conjuntoAlista :: Ord a => Conj a -> [a] conjuntoAlista c | esVacio c = [] | otherwise = mc : conjuntoAlista rc where mc = menor c rc = elimina mc c -- Comprobación de las propiedades -- =============================== -- La primera propiedad es prop_1_listaAconjunto :: [Int] -> Bool prop_1_listaAconjunto xs = conjuntoAlista (listaAconjunto xs) == sort (nub xs) -- La comprobación es -- λ> quickCheck prop_1_listaAconjunto -- +++ OK, passed 100 tests. -- La segunda propiedad es prop_2_listaAconjunto :: Conj Int -> Bool prop_2_listaAconjunto c = listaAconjunto (conjuntoAlista c) == c -- La comprobación es -- λ> quickCheck prop_2_listaAconjunto -- +++ OK, passed 100 tests. |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 |
from __future__ import annotations from abc import abstractmethod from copy import deepcopy from functools import reduce from typing import Protocol, TypeVar from hypothesis import given from hypothesis import strategies as st from src.TAD.conjunto import (Conj, conjuntoAleatorio, elimina, esVacio, inserta, menor, pertenece, vacio) class Comparable(Protocol): @abstractmethod def __lt__(self: A, otro: A) -> bool: pass A = TypeVar('A', bound=Comparable) # 1ª definición de listaAconjunto # =============================== def listaAconjunto(xs: list[A]) -> Conj[A]: if not xs: return vacio() return inserta(xs[0], listaAconjunto(xs[1:])) # 2ª definición de listaAconjunto # =============================== def listaAconjunto2(xs: list[A]) -> Conj[A]: return reduce(lambda ys, y: inserta(y, ys), xs, vacio()) # 3ª solución de listaAconjunto # ============================= def listaAconjunto3(xs: list[A]) -> Conj[A]: c: Conj[A] = Conj() for x in xs: c.inserta(x) return c # Comprobación de equivalencia # ============================ # La propiedad es @given(st.lists(st.integers())) def test_listaAconjunto(xs: list[int]) -> None: r = listaAconjunto(xs) assert listaAconjunto2(xs) == r assert listaAconjunto3(xs) == r # 1ª definición de conjuntoAlista # =============================== def conjuntoAlista(c: Conj[A]) -> list[A]: if esVacio(c): return [] mc = menor(c) rc = elimina(mc, c) return [mc] + conjuntoAlista(rc) # 2ª definición de conjuntoAlista # =============================== def conjuntoAlista2Aux(c: Conj[A]) -> list[A]: if c.esVacio(): return [] mc = c.menor() c.elimina(mc) return [mc] + conjuntoAlista2Aux(c) def conjuntoAlista2(c: Conj[A]) -> list[A]: c1 = deepcopy(c) return conjuntoAlista2Aux(c1) # 3ª definición de conjuntoAlista # =============================== def conjuntoAlista3Aux(c: Conj[A]) -> list[A]: r = [] while not c.esVacio(): mc = c.menor() r.append(mc) c.elimina(mc) return r def conjuntoAlista3(c: Conj[A]) -> list[A]: c1 = deepcopy(c) return conjuntoAlista3Aux(c1) # Comprobación de equivalencia de las definiciones de conjuntoAlista # ============================================================== @given(c=conjuntoAleatorio()) def test_conjuntoAlista(c: Conj[int]) -> None: r = conjuntoAlista(c) assert conjuntoAlista2(c) == r assert conjuntoAlista3(c) == r # Comprobación de las propiedades # =============================== # La primera propiedad es @given(st.lists(st.integers())) def test_1_listaAconjunto(xs: list[int]) -> None: assert conjuntoAlista(listaAconjunto(xs)) == sorted(list(set(xs))) # La segunda propiedad es @given(c=conjuntoAleatorio()) def test_2_listaAconjunto(c: Conj[int]) -> None: assert listaAconjunto(conjuntoAlista(c)) == c # La comprobación de las propiedades es # > poetry run pytest -v TAD_Transformaciones_conjuntos_listas.py # test_listaAconjunto PASSED # test_conjuntoAlista PASSED # test_1_listaAconjunto PASSED # test_2_listaAconjunto PASSED |