Sucesión de Cantor de números innombrables
Un número es innombrable si es divisible por 7 o alguno de sus dígitos es un 7. Un juego infantil consiste en contar saltándose los números innombrables:
1 |
1 2 3 4 5 6 ( ) 8 9 10 11 12 13 ( ) 15 16 ( ) 18 ... |
La sucesión de Cantor se obtiene llenando los huecos de la sucesión anterior
como se indica a continuación:
1 2 3 4 5 6 |
1 2 3 4 5 6 (1) 8 9 10 11 12 13 (2) 15 16 (3) 18 19 20 (4) 22 23 24 25 26 (5) (6) 29 30 31 32 33 34 (1) 36 (8) 38 39 40 41 (9) 43 44 45 46 (10) 48 (11) 50 51 52 53 54 55 (12) (13) 58 59 60 61 62 (2) 64 65 66 (15) 68 69 (16) (3) (18) (19) (20) (4) (22) (23) (24) (25) 80 81 82 83 (26) 85 86 (5) 88 89 90 (6) 92 93 94 95 96 (29) (30) 99 100 |
Definir la sucesión
1 |
sucCantor :: [Integer] |
cuyos elementos son los términos de la sucesión de Cantor. Por ejemplo,
1 2 3 4 5 6 7 8 9 10 11 12 13 |
λ> take 100 sucCantor [1,2,3,4,5,6, 1 ,8,9,10,11,12,13, 2, 15,16, 3, 18,19,20, 4, 22,23,24,25,26, 5 , 6 ,29,30,31,32,33,34, 1 ,36 , 8 ,38,39, 40,41, 9 ,43,44,45,46, 10 ,48, 11 ,50,51,52,53,54,55 , 12 , 13, 58,59,60,61,62, 2 ,64,65,66, 15 ,68,69, 16 , 3 , 18, 19, 20, 4, 22, 23, 24 ,25 ,80,81,82,83, 26 ,85,86, 5 ,88,89,90, 6, 92,93,94,95,96, 29, 30 ,99,100] λ> sucCantor2 !! (5+10^6) 544480 λ> sucCantor2 !! (6+10^6) 266086 |
Soluciones
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 |
-- 1ª solución -- =========== sucCantor1 :: [Integer] sucCantor1 = map fst $ scanl f (1,0) [2..] where f (a,i) x | esInnombrable x = (sucCantor1 !! i, i+1) | otherwise = (x,i) esInnombrable :: Integer -> Bool esInnombrable x = rem x 7 == 0 || '7' `elem` show x -- 2ª solución -- =========== sucCantor2 :: [Integer] sucCantor2 = aux 0 1 where aux i x | esInnombrable x = sucCantor2 !! i : aux (i+1) (x+1) | otherwise = x : aux i (x+1) |
Referencia
Basado en Cantor’s unspeakable numbers de
CodeGolf.
(más eficiente usar
Int
)