-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathsandbox.hs
577 lines (492 loc) · 23.2 KB
/
sandbox.hs
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
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
{-# LANGUAGE DatatypeContexts #-}
module Sandbox where
-- Ejemplos y ejercicios siguiendo el libro: "Programacion Funcional" de Jeroem Fokker
-- http://people.cs.uu.nl/jeroen/
-- Algunas de las soluciones copiadas de ALBERTO RODRIGUEZ CALVO-}
import Data.List as List
import Data.Array as Array
import Data.Char
import Data.Ord
import Control.Monad.Writer
fac n = product [1..n]
comb n k = fac n / (fac k * fac (n-k))
n !^! k=comb n k
formulaWhere a b c = [(-b+d)/n, (-b-d)/n ]
where d= sqrt (b*b-4.0*a*c)
n= 2.0*a
facrec n |n==0 = 1
|n>0 = n * fac (n-1)
--Ejemplos de patrones en la declaracion (destructuracion en la llamada)
facrec2 0 = 1
--facrec2 (n+1)=(n+1)*fac n
{-Es posible crear listas de funciones, si estas funciones (como numeros, valores booleanos y listas) son de un
mismo tipo, es posible hacer listas de funciones.
:type [sin,cos,tan]-}
{-Esta permitido escribir el tipo de una funcion en el
programa. La definicion de funcion se realizarla de la siguiente forma:-}
mysum :: [Int] -> Int
mysum [] = 0
mysum (x:xs) = x + mysum xs
{-Aunque la declaracion del tipo es superflua, tiene dos ventajas:
* se comprueba si la funcion tiene el tipo que esta declarado.
* la declaracion del tipo ayuda a entender la funcion.-}
add5 :: [Int] -> [Int]
add5 [] = []
add5 (x:xs) = (x+5):add5(xs)
add6 :: Int -> Int
add6 n = n+6
lpad :: Int -> Char -> [Char] -> [Char]
lpad 0 c cad = cad
lpad n c cad = lpad (n-1) c (c:cad)
{- Ejercicio 1.1.
Escriba una funcion que cuente cuantos numeros negativos existen en una lista.-}
countNeg0 xs = foldl (\x y -> if y<0 then x+1 else x) 0 xs
countNeg1 xs=sum [1 | x <- xs,x<0]
countNeg2 [] = 0
countNeg2 (h:t) | h < 0 = 1 + countNeg2 t
| otherwise = countNeg2 t
{- Ejercicio 1.2
Escriba una funcion diag que tenga una lista de caracteres como parametro
y que de como resultado los caracteres
en una diagonal.-}
diag str=putStrLn $ reduce 0 str
where reduce n []=""
reduce n (h:t)= replicate n ' '
++ [h] ++ reduce (n+1) t
{- Ejercicio 1.3
Escriba una funcion cuadrado que dada una lista de caracteres, presente tantas
copias de esta serie de caracteres (cada copia en una nueva linea), de manera
que el numero de las letras en horizontal sea igual al numero de las letras
que hay verticalmente. Tenga en cuenta que una cadena de caracteres es en realidad
una lista de caracteres -}
cuadrado str= putStrLn.unlines $ replicate (length str) str
{- Ejercicio 1.4
Escriba una funcion dividir, de manera que dada una lista de caracteres de como
resultado otra lista, pero ahora dividida en lineas. Cada vez que haya dos
caracteres seguidos que sean iguales se insertar en el resultado una
nueva linea (entre los dos caracteres iguales) -}
dividir lst=let line x (h:t) |x==h=x:'\n':h:t
|otherwise=x:h:t
f (h:t)=foldr line [last (h:t)] $ init (h:t)
f []=[]
in putStrLn $ f lst
{- Ejercicio 3.1
Escriba una funcion aproxseno que, dados dos numeros eps y x
(el primero mayor que 0, el segundo cualquiera),
de como resultado el numero y con la propiedad de que
| sin x - y | < eps
Use la siguiente regla matematica:
(-1)^n * x ^(2*n+1) /(fromIntegral $ fac (2*n+1))
Escriba dos veces una definicion para aproxseno:
una vez usando la funcion iterate y otra con until.
-}
aproxseno x eps = head $ until (\(y:_)-> abs (sin x-y) < eps)
(\(fst:snd:tail) -> fst+snd:tail)
$ termsTaylor x
termsTaylor x= map (term x) [0..]
term x n= (-1)^n * x ^(2*n+1) /(fromIntegral $ fac (2*n+1))
aproxseno2 x eps= head $ dropWhile (\y-> abs (sin x-y) >= eps)
$ map head $ iterate (\(fst:snd:tail) -> fst+snd:tail)
$ termsTaylor x
{- Ejercicio 3.4
¿Que funcion f y que lista a cumplen la siguiente regla?
map (+1) . reverse = foldl f a -}
assert 3.4 f lst=(map (+ 1).reverse $ lst)
== (f lst)
test 3.41 = assert 3.4 (foldl (\a b->b+1:a) []) [0,1,2,3]
{- Ejercicio 3.5
Defina una funcion esta que controle si existe cierto elemento en una lista de elementos. Defina la funcion de las
siguientes maneras:
1 Tome todos los elementos iguales al elemento buscado y coloque estos en una lista. Compruebe despues si
esta lista esta vacia o no.
2 Haga una nueva lista en la que todos los elementos iguales al elemento buscado sean reemplazados por 1 y los
otros elementos por 0. Sume los elementos de la lista resultante y compruebe si el resultado es igual a 0 o no.
3 Compruebe para cada elemento si es igual al elemento buscado o no. Despues compruebe si uno de estos tests
devolvio True.-}
esta 0 x lst = not.null.(filter (== x)) $ lst
esta 1 x lst = 0 < (sum $ map (\y->if y==x then 1 else 0) lst)
esta 2 x lst = or $ map (== x) lst
esta 3 x lst = any (== x) lst
{-Ejercicio 3.6
Escriba una funcion posiciones que devuelva una lista de indices de las posiciones de un elemento determinado en
una lista de elementos.
Por ejemplo:
? posiciones 4 [1,4,3,7,4,2]
[2,5]
? posiciones [3,5] [[3,6],[2,5]]
[] -}
posiciones x lst= let acc (i,is) y | x==y = (i+1,is++[i])
| otherwise=(i+1,is)
in snd $ foldl acc (0,[]) lst
{-Ejercicio 3.7
Escriba una funcion ndedc (numero de elementos distintos creciente), que dada una lista no decreciente de numeros,
decida cuantos numeros distintos hay en la lista.
Use el dato de que la lista esta ordenada.-}
ndedc:: (Eq a) => [a] -> Int
ndedc lista = let norep [] n=[n]
norep (x:xs) n | x == n = x:xs
| otherwise = n:x:xs
in length $ foldl norep [] lista
{-Ejercicio 3.8
Escriba una funcion nded (numero de elementos distintos), que, dada una lista cualquiera de numeros, devuelva
cuantos numeros distintos existen en la lista.
Una posibilidad de resolver este problema es contar solamente la primera ocurrencia de cada numero en la lista.
-}
nded :: (Eq a)=>[a]->Int
nded = length.nub
nded2::(Eq a)=>[a]->Int
nded2 = let cont (x,xs) y | elem y xs = (x,xs)
| otherwise = (x+1,y:xs)
in fst .(foldl cont (0,[]))
{-Ejercicio 3.9
Escriba una funcion segmento, que, dados una lista xs y dos numeros i y j, devuelva una sublista de xs desde el
indice i+1 hasta el indice j.
No se puede usar el operador !!.
Antes de contestar esta pregunta, se debe especificar que pasa si j <= i, j > #xs y si i > #s.-}
segmento2 i j lst
| j > (length lst)=segmento i (length lst) lst
| i>=j = segmento j i lst
| otherwise= let acc (c,xs) x | c>=i && c<=j = (c+1,xs++[x])
| otherwise = (c+1,xs)
in snd $ foldl acc (0,[]) lst
segmento:: Int -> Int -> [a] -> [a]
segmento i j lista | i >= j = segmento2 j i lista
| i < 0=segmento2 0 j lista
| j > l=segmento2 i l lista
| otherwise=take (j-i) (drop i lista)
where l=length lista
{-Ejercicio 3.10
Escriba una funcion esSegmento, que, dadas dos listas xs y ys devuelva True si xs es segmento de ys,
y False si no.
Una lista xs es sublista de ys cuando ys = hs ++ xs ++ ts, con hs, ts listas de cero o mas elementos.
Se puede usar la funcion segmento del ejercicio anterior.-}
esSegmento xs ys
| length xs > length ys = False
| h1==h2 && xs == take (length xs) ys = True
| otherwise = esSegmento xs t2
where (h1:_,h2:t2)=(xs,ys)
esSegmento2 xs ys | length xs > (length ys) = False
| xs == take (length xs) ys = True
| otherwise = esSegmento2 xs (tail ys)
{-Ejercicio 3.11
Escriba una funcion scdosa (sigue concatenando los dos anteriores), que, dadas dos listas xs y ys
del mismo tipo, devuelva una lista infinita de listas, con las siguientes propiedades:
*Los primeros dos elementos son respectivamente xs y ys.
*Para cada n > 0 el n+2-esimo elemento es la concatenacion del n-esimo elemento con el n+1-esimo elemento.
Use la funcion iterate.-}
scdosa xs ys =xs:(map last $ iterate (\lst -> lst++[(last.init $ lst) ++ (last lst)]) [xs,ys])
-- La buena
acdosa2 xs ys = map fst $ iterate (\(xs,ys)->(ys,xs++ys)) (xs, ys)
{-Ejercicio 3.12
Escriba una funcion sssp (sigue sumando el segmento previo), que, dada una lista finita de numeros ns con un
tamaño k > 0, devuelva una lista infinita ms que cumpla con las siguientes propiedades:
* ns = take k ms
* Para todo n >= k : ms!!(n+1) = (sum . drop (n-k) . take n) ms
Por ejemplo:
sssp [0,0,1] = [0,0,1,1,2,4,7,13,24,44..
Use la funcion iterate.-}
sssp lst = let k=length lst
f xs=let n=length xs
in xs++[(sum.drop(n-k).take n) xs]
in init lst ++ (map last $ iterate f lst)
sssp2:: [Int] -> [Int]
sssp2 xs = xs ++ map fst ( iterate f (sum xs, xs) )
where f (suma, y:ys) = (sum zs, zs) where zs = ys ++ [suma]
-- La buena
sssp3 ns = map head (iterate f ns)
where f ns = (tail ns) ++ [sum ns]
{-Ejercicio 3.13
Escriba una funcion elimDobles, que, dada una lista (que puede ser infinita), devuelva una nueva lista, con solamente
una ocurrencia de cada elemento de la lista original. El problema en este ejercicio es que la lista puede ser infinita.
Por eso, no puede usar las funciones foldr y foldl.-}
elimDobles []=[]
elimDobles (x:xs) = x:(elimDobles (filter (/=x) xs))
{-Ejercicio 3.14
Un valor x se denomina extremo interno con indice i en la lista xs, si i es un indice con las siguientes propiedades:
1 < i < length xs
xs!!i = x
existen una j y una k , con j < i y k > i con xs!!j /= x y xs!!k /= x
la mayor j (j < i) y la menor k (k > i) con xs!!j /= x y xs!!k /= x cumplen con la condicion que
o xs!!j > x y xs!!k > x
o xs!!j < x y xs!!k < x
Dos extremos internos con indices i y j en una lista son vecinos si no existe otro extremo con indice k y i < k < j
o j < k < i.
Escriba una funcion extremos, que calcule los extremos internos de una lista.
Use la funcion foldl.-}
-- La mia
extremos [h]= []
extremos [h,t]=[]
extremos (h:t)=
let f (acc,p) x
| x>p && (head acc)>p ||
x<p && (head acc)<p=(p:acc,x)
| otherwise= (acc,x)
in init.fst $ foldl f ([h],h) t
extremos1 [h]= []
extremos1 [h,t]=[]
extremos1 (h:t)=let f (acc,p) x = (acc++e,x)
where e | x>p && (last acc)>p ||
x<p && (last acc)<p=[p]
| otherwise=[]
in tail.fst $ foldl f ([h],h) t
--Otra
extremos2:: [Int] -> [Int]
extremos2 lista = fst (foldl f ([], []) lista)
where f ([], []) n = ([], [n])
f (extremos, (y:ys)) n | ys == [] && y == n =(extremos, (y:ys ))
| ys == [] && y /= n =(extremos, (y:[n]))
| head ys == n =(extremos, (y:ys))
| (y < head ys)==((head ys) < n)=(extremos, y:[n])
| otherwise =(extremos ++ [head ys], (head ys):[n])
{-Ejercicio 3.15
Escriba una funcion distanciaExtr que calcule la maxima distancia entre dos extremos vecinos. (Ver el ejercicio
3.14 para la definicion de extremos.) Si no existen dos extremos vecinos en la lista, entonces el resultado sera 0.-}
distanciaExtr (h:t)=
let f ((i,max),(acc,p)) x
| x>p && (head acc)>p ||
x<p && (head acc)<p=((0,max'),(p:acc,x))
| otherwise= ((i',max),(acc,x))
where i'=i+1
max' | max<0=0 | i'>max=i'
|otherwise=max
in snd.fst $ foldl f ((0,-1),([h],h)) t
-- Entendi mal no es la distancia segun el indice del extremo sino la diferencia entre valores:
distanciaExtr2:: [Int] -> Int
distanciaExtr2 lista = maxDif (extremos lista)
-- Maxima diferencia absoluta entre dos elementos consecutivos
maxDif:: [Int] -> Int
maxDif (x:xs) | length (x:xs) < 2=0
| otherwise=fst ( foldl f (0, x) xs )
where f (a, b) n=( max (abs(b-n))a, n )
{-Ejercicio 3.16
Escriba una funcion recursiva sc (sublistas crecientes), que, dada una lista, devuelva una lista de listas que existan
en todas las sublistas no decrecientes de la lista. Escriba tambien una definicion de sc usando foldr.
Por ejemplo:
? sc [6,1,4,8] = [[],[6],[1],[1,4],[4],
[1,4,8],[4,8],[1,8],[6,8],[8]]-}
sc lista = combinacion lista
where combinacion [] = []
combinacion (x:xs) = combinacion xs
++ (combina x $ combinacion xs)
combina elemento [] = [[elemento]]
combina elemento (x:xs) | x == [] = combina elemento xs
| elemento <=head x=(elemento:x):(combina elemento xs )
| otherwise=combina elemento xs
--Solucion con foldr
sc2:: [Int] -> [[Int]]
sc2 lista = foldr f [] lista
where f elemento xs = xs ++ (combina elemento xs)
{-Ejercicio 3.17
Escriba una funcion dividir, que, dados una lista no decreciente xs y un elemento x, devuelva una tupla de dos
listas (ys,zs), con xs = ys ++ zs, donde todos los elementos de ys sean menores o iguales que x, y todos los
elementos de zs sean mayores que x.
Escriba una funcion insertar, que, dados una lista no decreciente ys y un elemento y, devuelva una lista no
decreciente igual a ys mas el elemento y insertado en el lugar correspondiente.
dividir :: a -> [a] -> ([a],[a])-}
x <=: (h:t) = null t || h>x
dividir2 x xs= (takeWhile (x>=) xs,dropWhile (x>=) xs)
dividir3 :: (Ord a)=>a -> [a] -> ([a],[a])
dividir3 x xs=until ((x <=:).snd) (\(ys,(h:t))->(ys++[h],t)) ([],xs)
dividir4 x xs=span (x>=) xs
insertar x xs=let (ys,zs)=dividir2 x xs
in ys++(x:zs)
{-Ejercicio 3.18
Escriba una funcion unico, que, dada una lista devuelva una lista que contenga exactamente los elementos que se
encuentran solamente una vez en la lista dada. Por ejemplo:
"Cuales son las letras unicas en esta frase?"
"oicf?" -}
strt="Cuales son las letras unicas en esta frase?"
unico str=let low=map toLower str;nuby=nub low
in nuby\\(low\\nuby)
unico1 []=[]
unico1 ls=let low=map toLower ls
f acc []=acc
f acc (h:t)
|elem h t =f acc flt
|otherwise=f (acc++[h]) flt
where flt=filter (/= h) t
in f [] low
unico2=concat.(filter ((== 1).length)).group.sort.(map toLower)
{-Ejercicio 3.19
a. Escriba una funcion segcrec (segmentos crecientes), que dada una lista, devuelva una lista de listas que cumpla
con las siguientes condiciones:
• la concatenacion de los elementos en el resultado devuelve la lista original
• todos los elementos del resultado son listas no decrecientes y tampoco son vacias
• por cada segmento no decreciente ys de la lista dada, existe un elemento en el resultado del cual ys es
un segmento
La definicion de segcrec debe ser en base a foldl o foldr.
b. De tambien una definicion recursiva de segcrec.
Ejemplo:
? segcrec [1,2,3,4,2,3,5,6,4,8,3,2]
[[1,2,3,4],[2,3,5,6],[4,8],[3],[2]]-}
lst 3.19=[1,2,3,4,2,3,5,6,4,8,3,2]
segrec []=[]
segrec (h:t)=let f lst x | x>=(last.last$lst)=(init lst)++[last lst++[x]]
| otherwise=lst++[[x]]
in foldl f [[h]] t
segrec1 lst=let f x []=[[x]]
f x (h':t') | x<(head h') = (x:h'):t'
| otherwise=[x]:(h':t')
in foldr f [] $ lst
segrec2 []=[]
segrec2 [h]=[[h]]
segrec2 (h:h':t)|h>h'=[h]:next
|otherwise=(h:(head next)):(tail next)
where next=segrec2 $ h':t
{-Ejercicio 3.20
Escriba una funcion recursiva esSubLista, que, dadas dos listas, devuelva True si la segunda lista es una sublista
de la primera, y False si no. Decimos que ys es una sublista de la lista xs si existe una lista creciente de numeros
positivos is, con ys = [xs!!i|i<-is]. Ejemplos:
? esSubLista "muchisimo" "uso"
True
? esSubLista [1,4,2,5,7] [4,7]
True
? esSubLista [1,4,2,5,7] [2,1]
False
-- Useful tips -}
esSubLista xs []=True
esSubLista [] ys=False
esSubLista (h:t) (h':t')
| (h:t)==(h':t')=True
| h==h'=esSubLista (h:t) t'
| otherwise = esSubLista t (h':t')
{-Ejercicio 3.21
Escriba una funcion partir, que, dados un predicado p y una lista xs, devuelva una tupla de listas (ys,zs) en tal
forma que ys contenga todos los elementos de xs, que cumplan con la condicion p, y en zs el resto de los elementos
de xs. Por ejemplo:
? partir digit "a1!,bc4"
("14","a!,bc")
Es posible escribir una definicion simple y correcta:
partir p xs = (filter p xs, filter (not.p) xs)
Pero, en este ejercicio queremos practicar el uso de las funciones fold. Entonces, debes dar una definicion en base
a foldr o foldl.-}
-- Esta funcion es una generalizacion del ejercicio 3.17 y se pueden trasladar las soluciones
-- directamente poniendo el predicado como parametro
partir=span
partir1 pred xs= (takeWhile pred xs,dropWhile pred xs)
partir2 pred xs=until (pred.head.snd) (\(ys,(h:t))->(ys++[h],t)) ([],xs)
--Pero nos la piden con foldl y/o foldr
partirL pred xs=let nxt (xs,ys) x
| pred x=(xs,ys++[x])
| otherwise =(xs++[x],ys)
in foldl nxt ([],[]) xs
partirR pred xs=let nxt x (xs,ys) |pred x=(xs,x:ys)
|otherwise=(x:xs,ys)
in foldr nxt ([],[]) xs
{-Ejercicio 3.22
Escriba las funciones sumar, multiplicar, restar y dividir para numeros complejos. Un numero complejo es de la
forma a + bi, con a, b numeros reales, y i un numero con la propiedad: i2 = -1. Para la funcion dividirCompl
puede ser util primero derivar una formula para 1/a+bi . Para esto, puedes calcular los valores de x e y en la ecuacion
(a + bi)*(x + yi) = (1 + 0i).-}
-- Mezcla de la libreria de Haskell complex y la funcion de division de Alberto
infix 6 :+
data (RealFloat a) => Complex a = !a :+ !a deriving (Eq,Read,Show)
instance (RealFloat a) => Num (Complex a) where
(x:+y) + (x':+y') = (x+x') :+ (y+y')
(x:+y) - (x':+y') = (x-x') :+ (y-y')
(x:+y) * (x':+y') = (x*x'-y*y') :+ (x*y'+y*x')
abs z = undefined
signum z = undefined
fromInteger n = fromInteger n :+ 0
dividirCompl:: Complex Float-> Complex Float-> Complex Float
dividirCompl (x:+y) (x':+y') = (x:+y) * ((x'/div):+(-y'/div))
where div = x'*x' + y'*y'
{-Ejercicio 3.23
En sistemas de numeracion en base k (con k un numero entero y k > 1), un numero puede ser representado por una
lista de numeros, todos menores que k y mayores o iguales a cero.
En el sistema de numeracion en base 10 (k = 10), la lista [9,8,4] representa el numero 984 (9*100+8*10+4*1).
En el sistema de numeracion en base tres (k = 3), la lista [2,0,1] representa el numero 19 (2*9+0*3+1*1).
Escriba una funcion listaAnumero, que, dados un numero k y una lista ms de numeros m (0 <= m < k), devuelva el
numero representado por la lista en el sistema de numeracion en base 10.
Defina la funcion en base a foldl.-}
listaNum k xs= let f (acc,0) x =(acc+x,0)
f (acc,i) x =(acc+(x*k^i),i-1)
in fst $ foldl f (0,(length xs)-1) xs
{-Ejercicio 3.24
Podemos cambiar la representacion de numeros en un sistema de numeracion en base k que esta descrita en el
ejercicio 3.23 por una representacion en que esta el numero al reves. Entonces, en este caso, el numero 984 en el
sistema de numeracion es representado por la lista [4,8,9].
Escriba una funcion listaAnumeroR que haga lo mismo que la funcion listaAnumero, pero ahora con la representacion al reves.
Defina la funcion en base a foldr.-}
listaNumR k xs=let f x (acc,0)=(acc+x,0)
f x (acc,i)=(acc+(x*k^i),i-1)
in fst $ foldr f (0,(length xs)-1) xs
-- Solucion mas elegante de Xavier Garcia Buils
porKmas k m n = m*k+n
listaAnumero k = foldl (porKmas k) 0
listaAnumeroR k = foldr (flip $ porKmas k) 0
{-Ejercicio 3.25
Escriba una funcion multiplicar, que, dados un numero positivo menor que 10 m y una lista de numeros ns, que
representa un numero n como esta descrito en el ejercicio 3.24, devuelva una lista que represente la multiplicacion
n*m, tambien segun la representacion descrita en el ejercicio anterior. Puede suponer que trabajamos en un sistema
de numeracion en base 10. Ejemplos:
? multiplicar 3 [4,8,9]
[2,5,9,2]
? multiplicar 5 [9,9,9,1,4,6]
[5,9,9,9,0,2,3]
Una solucion podria ser: cambiar el numero representado en la lista por un numero entero y despues multiplicar.
Esta solucion no se permite, porque, en este caso, no se pueden multiplicar numeros que sean muy grandes (la
maquina acepta solamente numeros enteros hasta cierto limite). Por eso, debe aplicar otro sistema de multiplicar,
por ejemplo el sistema que consiste en multiplicar numero por numero y guardar cada vez el resto. En este caso,
trabaja con un par de valores: los numeros del resultado ya calculados y el resto de la ultima multiplicacion. Use
foldr o foldl.-}
x % y = (div x y,mod x y)
mult x xs=let m (acc,r) y=
let (d,r')=(y*x+r)%10
in (r':acc,d)
in uncurry (flip (:)) $ foldl m ([],0) xs
multR2 x xs=let m y (acc,r)=
let (d,r')=(y*x+r)%10
in (r':acc,d)
in uncurry (flip(:)) $ foldr m ([],0) xs
{-Como podemos ver de varios ejercicios hay un patron comun en varios de ellos: fold con una tupla que
consiste un acumulador y tarnformar una lista en otra (map) COmo para casi cada patron hay un funcion de
orden superior para evitar repeticiones en este caso es mapAccumR y mapAccumL. Podemos reescribir la ultima
de mis soluciones:-}
multR x xs=uncurry (:) $ mapAccumR (\r y->(y*x+r)%10) 0 xs
{-Ejercicio 3.26
Escriba una funcion multip que haga lo mismo que la funcion multiplicar descrita en el ejercicio 3.25, pero ahora
multiplique dos listas en la representacion especial, y no (como en el ejercicio anterior), un numero entero menor
que 10 con una lista en la representacion especial. Por ejemplo:
? multip [1,3] [4,8,9]
[6,3,9,3]
Es util usar la funcion multip y escribir una funcion sumar mas. La funcion sumar debe sumar dos numeros
representados en listas como en los anteriores ejercicios.-}
mas= (uncurry (:)).
(mapAccumR (\r x->(x+r)%10) 0).
(map sum).transpose
multp xs ys=let r0 i=replicate i 0
acc i n=(i+1,r0(length xs-i)++(multR n ys)++r0 i)
in mapAccumR acc 0 xs --mal
infix 8 $>
--($>) :: a-> [(a->b)] -> [b]
fs $> x = map ($ x) fs
-- Monads
tellMe :: Int -> Writer String Int
tellMe x= do
let y ="2" -- <- getLine
tell $ "You have written:" ++ y
let r=x+(read y :: Int)
tell $ "The result is:" ++ show r
return r
arbol' n = [if c == n + q
then '\n'
else if c <= n - q
then ' '
else '*' |
q <- [1 .. n], c <- [1 .. n + q]]
arbol'' n = unlines [r (n-x) ' ' ++ r (2*x-1) '*' |x<-[1..n]]
where r=replicate
-- Genera los INFINITOS árboles de Navidad
bosque=a["*"] where a x=x:a (map (' ':) x ++ [replicate (1+2*length x) '*'])
-- Para tomar un árbol concreto, basta referenciarlo por su índice
arbol=(!!)bosque
-- Para imprimirlo
talar=putStrLn.unlines.arbol
bosque'=iterate(\x->map(' ':)x++[replicate(1+2*length x) '*'])["*"]
--bosque=a["*"]where a x=x:a(map(' ':)x++[take(1+2*length x)$repeat '*'])
minSubsetSum'=head.sortBy(comparing length).filter(\s->s/=[] && sum s==0).subsequences
vigenere m c=zipWith f m $ cycle c
where f x y=chr $ (g x + g y - 130) `mod` 26 + 65;g=ord.toUpper
f x=x+1
g x y=(f x) + (f y)