Tema 3: Características de la programación funcional
Transcripción
Tema 3: Características de la programación funcional
Tema 3: Características de la programación funcional Sesión 5: El paradigma funcional (1) martes 22 de febrero de 2011 Referencias • Capítulo 1.1.5 SICP: [[http://mitpress.mit.edu/sicp/full-text/book/book-ZH-10.html#%_sec_1.1.5][The Substitution Model for Procedure Application]] • Capítulo 10 PLP: Functional Languages martes 22 de febrero de 2011 Hoy veremos • Historia del paradigma funcional • Características del paradigma funcional puro • Modelo de computación basado en sustitución martes 22 de febrero de 2011 Orígenes históricos • Raíces teóricas, antes de aparecer los ordenadores • 1930s: Alan Turing, Alonzo Church, Stephen Kleene, Emil Post, etc. • Turing: Máquina de Turing, programación imperativa • Kleene y Post: métodos abstractos, sustituciones algebraicas • Church: Cálculo lambda, programación funcional • Modelo de cálculo lambda basado en la definición de funciones y la aplicación de estas funciones a argumentos martes 22 de febrero de 2011 Conceptos de la programación funcional En un sentido estricto, la programación funcional define un programa como una función matemática que convierte unas entradas en unas salidas, sin estado interno ni efectos laterales • Lenguajes funcionales puros: Miranda, Haskell, pH, Sisal, ML... • (1958, John MacCarthy). LISP es el primer lenguaje de alto nivel basado en el paradigma funcional • LISP no es programación funcional pura, tienen algunas instrucciones imperativas que permiten estado local y efectos laterales • LISP lenguaje revolucionario, introduce nuevos conceptos: funciones como objetos primitivos, funciones orden superior, polimorfismo, listas, recursión, símbolos, homogeneidad de datos y programas, bucle “read-eval-print” martes 22 de febrero de 2011 Características de la programación funcional • Programación declarativa: • no hay asignación ni cambio de estado • no hay referencias: identificadores asociados a valores • no hay efectos laterales • Recursión • Funciones como tipos de datos primitivos martes 22 de febrero de 2011 Programación declarativa vs imperativa • Programación imperativa: pasos de ejecución y estado de variables int x = x + 1; int y = y + 3; (define (cuadrado x) (* x x)) • Ejemplo de estado local (no existe en programación declarativa): int function contador () { static int c = 0; c++; return c; } contador(): 1 contador(): 2 contador(): 3 martes 22 de febrero de 2011 Programación declarativa vs imperativa • Programación declarativa: Dentro del ámbito de declaración de las variables x1 … xn todas las ocurrencias de una expresión e que contiene únicamente las variables x1 … xn tienen el mismo valor. • Consecuencia: optimización. Si una expresión e aparece en varios lugares dentro de un mismo ámbito, sólo es necesario evaluarla una vez. (define (f x) ...) (+ (f 2) (f 2)) (define (f x) ...) (define y (f 2)) (+ y y) Una función llamada con los mismos argumentos siempre devuelve el mismo resultado martes 22 de febrero de 2011 Modelo de computación de sustitución • Un modelo computacional es un formalismo (conjunto de reglas) que definen el funcionamiento de un programa. • En los lenguajes funcionales basados en la evaluación de expresiones, el modelo computacional define cuál va a ser el resultado de evaluar una determinada expresión. • El modelo de sustitución se basa en una versión simplificada de la regla de reducción del cálculo lambda. martes 22 de febrero de 2011 Modelo de computación de sustitución • Reglas para evaluar una expresión e de Scheme: • Si e es un valor primitivo, devolver ese mismo valor. • Si e es una variable, devolver su valor asociado. • Si e es una expresión del tipo (f arg1 … argn), donde f el nombre de un procedimiento primitivo (+, -, …), evaluar arg1 … argn y aplicar el procedimiento al resultado. • Si e es una expresión del tipo (f arg1 … argn), donde f es el nombre de un procedimiento compuesto (definido con un define), sustituir f por su cuerpo, reemplazando cada parámetro formal del procedimiento por el correspondiente argumento evaluado. Evaluar la expresión resultante. martes 22 de febrero de 2011 Ejemplo modelo de sustitución (define (double x) (+ x x)) (define (square y) (* y y)) (define (f z) (+ square (double z)) 1)) • Evaluamos (f (+ 2 1)) con orden aplicativo martes 22 de febrero de 2011 Orden normal vs orden aplicativo • Orden aplicativo (Scheme): se evalúan primero los argumentos, luego se sustituye • Orden normal: se realizan todas las sustituciones hasta que se tiene una larga expresión formada por expresiones primitivas; se evalúa entonces • Evaluamos (f (+ 2 1)) con orden normal En programación funcional el orden normal y el orden aplicativo siempre devolverán el mismo resultado martes 22 de febrero de 2011 Orden normal vs orden aplicativo • El orden importa si no tenemos programación funcional pura: (define (zero x) (- x x)) (zero (random 10)) martes 22 de febrero de 2011 Simulación en Scheme (load "simply.scm") (load "order.scm") (def (doble x) (+ x x)) (def (cuadrado x) (* x x)) (def (f z) (+ (cuadrado (doble z)) 1)) (applic (f (+ 2 1))) (normal (f (+ 2 1))) martes 22 de febrero de 2011 Ejercicios • Dada la función: (define (double x) (+ x x)) • Evaluar la expresión (double (* 3 4)) en orden aplicativo y en orden normal. ¿Cuál es más eficiente? • Dada la función: (define (switch (cond ((< x ((= x ((> x x a b c) 0) a) 0) b) 0) c)))) • Evaluar la expresión (switch -1 (+ 1 2) (+ 2 3) (+ 3 4)) en orden aplicativo y en orden normal. ¿Cuál es más eficiente? martes 22 de febrero de 2011