TAD de los conjuntos: Reconocimiento de subconjuntos
Utilizando el tipo abstracto de datos de los conjuntos definir la función
1 |
subconjunto :: Ord a => Conj a -> Conj a -> Bool |
tal que subconjunto c1 c2
se verifica si todos los elementos de c1
pertenecen a c2
. Por ejemplo,
1 2 3 4 5 6 7 |
λ> ej1 = inserta 5 (inserta 2 vacio) λ> ej2 = inserta 3 (inserta 2 (inserta 5 vacio)) λ> ej3 = inserta 3 (inserta 4 (inserta 5 vacio)) λ> subconjunto ej1 ej2 True λ> subconjunto ej1 ej3 False |
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 60 61 |
import TAD.Conjunto (Conj, vacio, inserta, menor, elimina, pertenece, esVacio) import Transformaciones_conjuntos_listas (conjuntoAlista) import Test.QuickCheck -- 1ª solución -- =========== subconjunto :: Ord a => Conj a -> Conj a -> Bool subconjunto c1 c2 | esVacio c1 = True | otherwise = pertenece mc1 c2 && subconjunto rc1 c2 where mc1 = menor c1 rc1 = elimina mc1 c1 -- 2ª solución -- =========== subconjunto2 :: Ord a => Conj a -> Conj a -> Bool subconjunto2 c1 c2 = and [pertenece x c2 | x <- conjuntoAlista c1] -- La función conjuntoAlista está definida en el ejercicio -- "Transformaciones entre conjuntos y listas" que se encuentra en -- https://bit.ly/3RexzxH -- 3ª solución -- =========== subconjunto3 :: Ord a => Conj a -> Conj a -> Bool subconjunto3 c1 c2 = all (`pertenece` c2) (conjuntoAlista c1) -- 4ª solución -- =========== subconjunto4 :: Ord a => Conj a -> Conj a -> Bool subconjunto4 c1 c2 = sublista (conjuntoAlista c1) (conjuntoAlista c2) -- (sublista xs ys) se verifica si xs es una sublista de ys. Por -- ejemplo, -- sublista [5, 2] [3, 2, 5] == True -- sublista [5, 2] [3, 4, 5] == False sublista :: Ord a => [a] -> [a] -> Bool sublista [] _ = True sublista (x:xs) ys = elem x ys && sublista xs ys -- Comprobación de equivalencia -- ============================ -- La propiedad es prop_subconjunto :: Conj Int -> Conj Int -> Bool prop_subconjunto c1 c2 = all (== subconjunto c1 c2) [subconjunto2 c1 c2, subconjunto3 c1 c2, subconjunto4 c1 c2] -- La comprobación es -- λ> quickCheck prop_subconjunto -- +++ 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 |
from __future__ import annotations from abc import abstractmethod from copy import deepcopy from typing import Protocol, TypeVar from hypothesis import given from src.TAD.conjunto import (Conj, conjuntoAleatorio, elimina, esVacio, inserta, menor, pertenece, vacio) from src.TAD_Transformaciones_conjuntos_listas import conjuntoAlista class Comparable(Protocol): @abstractmethod def __lt__(self: A, otro: A) -> bool: pass A = TypeVar('A', bound=Comparable) # 1ª solución # =========== def subconjunto(c1: Conj[A], c2: Conj[A]) -> bool: if esVacio(c1): return True mc1 = menor(c1) rc1 = elimina(mc1, c1) return pertenece(mc1, c2) and subconjunto(rc1, c2) # 2ª solución # =========== def subconjunto2(c1: Conj[A], c2: Conj[A]) -> bool: return all((pertenece(x, c2) for x in conjuntoAlista(c1))) # La función conjuntoAlista está definida en el ejercicio # "Transformaciones entre conjuntos y listas" que se encuentra en # https://bit.ly/3RexzxH # 3ª solución # =========== # (sublista xs ys) se verifica si xs es una sublista de ys. Por # ejemplo, # sublista [5, 2] [3, 2, 5] == True # sublista [5, 2] [3, 4, 5] == False def sublista(xs: list[A], ys: list[A]) -> bool: if not xs: return True return xs[0] in ys and sublista(xs[1:], ys) def subconjunto3(c1: Conj[A], c2: Conj[A]) -> bool: return sublista(conjuntoAlista(c1), conjuntoAlista(c2)) # 4ª solución # =========== def subconjunto4(c1: Conj[A], c2: Conj[A]) -> bool: while not esVacio(c1): mc1 = menor(c1) if not pertenece(mc1, c2): return False c1 = elimina(mc1, c1) return True # 5ª solución # =========== def subconjunto5Aux(c1: Conj[A], c2: Conj[A]) -> bool: while not c1.esVacio(): mc1 = c1.menor() if not c2.pertenece(mc1): return False c1.elimina(mc1) return True def subconjunto5(c1: Conj[A], c2: Conj[A]) -> bool: _c1 = deepcopy(c1) return subconjunto5Aux(_c1, c2) # Comprobación de equivalencia # ============================ # La propiedad es @given(c1=conjuntoAleatorio(), c2=conjuntoAleatorio()) def test_subconjunto(c1: Conj[int], c2: Conj[int]) -> None: r = subconjunto(c1, c2) assert subconjunto2(c1, c2) == r assert subconjunto3(c1, c2) == r assert subconjunto4(c1, c2) == r assert subconjunto5(c1, c2) == r # La comprobación de las propiedades es # > poetry run pytest -q TAD_subconjunto.py # 1 passed in 0.37s |