Você está na página 1de 32

Programação Funcional

1 Aula  Apresentação
a

Pedro Vasconcelos
DCC/FCUP

2010

Pedro Vasconcelos DCC/FCUP Programação Funcional 1a Aula  Apresentação


Objectivos

Introdução à programação funcional usando a linguagem


Haskell
No nal do curso, o(a) aluno(a) deverá ser capaz de:
1 Denir funções em Haskell usando equações com padrões e
guardas
2 Codicar algoritmos recursivos elementares sobre listas e
árvores
3 Denir novos tipos algébricos para representar dados
4 Decompor problemas de programação usando funções de
ordem superior e lazy evaluation
5 Provar propriedades elementares de programas usando teoria
equacional e indução

Pedro Vasconcelos DCC/FCUP Programação Funcional 1a Aula  Apresentação


Funcionamento
Aulas teóricas 2 × 1 h por semana
Aulas teórico-práticas 1 h por semana
Aulas práticas 2 h por semana
Docentes Pedro Vasconcelos (T & TP), Miguel Filgueiras, Luis
Lopes, Luis Torgo
Página da disciplina com slides das aulas e folhas de exercícios
http://www.dcc.fc.up.pt/~pbv/aulas/pf
Avaliação um teste a meio do semestre (25% da nota)
exame nal (75% da nota)
é necessária frequência de 3/4 das aulas práticas1

1 Excepto trabalhadores estudantes.


Pedro Vasconcelos DCC/FCUP Programação Funcional 1a Aula  Apresentação
Bibliograa
Na biblioteca. . .
1 Programming in Haskell, Graham Hutton. Cambridge
University Press, 2007.
2 Introduction to Functional Programming, Richard Bird &
Philip Wadler, Prentice-Hall International, 1988.

Na web. . .
1 Learn you a Haskell for great good!,
http://learnyouahaskell.com/
2 Real World Haskell,
http://book.realworldhaskell.org/

Pedro Vasconcelos DCC/FCUP Programação Funcional 1a Aula  Apresentação


Programação funcional: o que é?

É um paradigma de programação
Preconiza a computação como a aplicação de funções
matemáticas a valores imutáveis (dados)
Por contraposição: no paradigma imperativo a computação é
vista como a execução de instruções que mudam variáveis
(estado)
Podemos programar num estilo (quase) funcional em muitas
linguagens. . .
Linguagens funcionais suportam melhor o paradigma funcional
Exemplos: Scheme, ML, Erlang, Haskell, F#

Pedro Vasconcelos DCC/FCUP Programação Funcional 1a Aula  Apresentação


Exemplo: somar os naturais de 1 a 10 em Java

total = 0;
for (i=1; i<=10; ++i)
total = total + i;

O programa é uma sequência de instruções


O resultado é obtido por mutação das variáveis i e total

Pedro Vasconcelos DCC/FCUP Programação Funcional 1a Aula  Apresentação


Execução passo-a-passo
passo instrução i total
1 total=0 ? 0
2 i=1 1 0
3 total=total+i 1 1
4 ++i 2 1
5 total=total+i 2 3
6 ++i 3 3
7 total=total+i 3 6
8 ++i 4 6
9 total=total+i 4 10
.. .. .. ..
. . . .

21 total=total+i 10 55
22 ++i 11 55

Pedro Vasconcelos DCC/FCUP Programação Funcional 1a Aula  Apresentação


Exemplo: somar os naturais de 1 a 10 em Haskell

sum [1..10]

O resultado é expresso como a aplicação da função sum à lista de


naturais entre 1 e 10.

Pedro Vasconcelos DCC/FCUP Programação Funcional 1a Aula  Apresentação


Redução

Redução da expressão original até obter um resultado que não pode


ser mais simplicado.

sum [1..10] =
= sum [1, 2, 3, 4, 5, 6, 7, 8, 9, 10] =
= 1 + 2 + 3 + 4 + 5 + 6 + 7 + 8 + 9 + 10 =
= 55

Pedro Vasconcelos DCC/FCUP Programação Funcional 1a Aula  Apresentação


Vantagens da programação funcional
Mais alto nível
programas mais concisos
mais próximos duma especicação matemática
Maior modularidade
polimorsmo, ordem superior, lazy evaluation
Maior facilidade de vericação da correcção dos programas
demonstrações de correcção
testes automáticos
Maior facilidade em execução concorrente ou em paralelo
a ordem de execução não afecta os resultados
transparência referencial: o valor duma expressão depende
apenas dos nomes visíveis

Pedro Vasconcelos DCC/FCUP Programação Funcional 1a Aula  Apresentação


Desvantagens da programação funcional

Maior distância do hardware:


compiladores/interpretadores mais complexos
difícil prever os custos de execução (tempo/espaço)
Alguns algoritmos são mais ecientes quando implementados
de forma imperativa

Pedro Vasconcelos DCC/FCUP Programação Funcional 1a Aula  Apresentação


Um pouco de história...
1930s Alonzo Church desenvolve o cálculo-λ, um formalismo
matemático para a denição de funções
1950s Inspirado no cálculo-λ, John McCarthy desenvolve o
LISP, uma das primeiras linguagens de programação
1970s Robin Milner desenvolve o ML, a primeira linguagem
funcional moderna, com polimorsmo e inferência de
tipos
1970s1980s David Turner desenvolve várias linguagens que
empregam lazy evaluation, culminando na linguagem
comercial Miranda
1987 Um comité académico internacional inicia o
desenvolvimento do Haskell, uma linguagem funcional
lazy padronizada e aberta
2003 Publicação do Haskell 98, uma denição padronizada
da linguagem
Pedro Vasconcelos DCC/FCUP Programação Funcional 1a Aula  Apresentação
Haskell
Uma linguagem puramente funcional de uso genérico
Nomeada em homenagem ao matemático americano
Haskell B. Curry (19001982)
Pensada tanto para ensino como para o desenvolvimento de
aplicações reais
Resultado de mais de vinte anos de investigação por uma
comunidade de base académica muito activa
Implementações abertas e livremente disponíveis
Sítio ocial: http://www.haskell.org

Pedro Vasconcelos DCC/FCUP Programação Funcional 1a Aula  Apresentação


Hugs

Um interpretador interactivo de Haskell


Suporta Haskell 98 e bastantes extensões
Ideal para aprendizagem e desenvolvimento de pequenos
programas
Disponível em www.haskell.org/hugs

Pedro Vasconcelos DCC/FCUP Programação Funcional 1a Aula  Apresentação


Glasgow Haskell Compiler (GHC)

Um compilador de Haskell que gera código nativo


Suporta Haskell 98 e bastantes extensões
Optimização de código, interfaces a outras linguagens,
prolers, grande conjunto de bibliotecas, etc.
Inclui também o interpretador ghci (alternativa ao Hugs)
Disponível em www.haskell.org/ghc

Pedro Vasconcelos DCC/FCUP Programação Funcional 1a Aula  Apresentação


Primeiros passos

Linux: executar o comando hugs ou ghci


Windows: executar a aplicação WinHugs

$ ghci
GHCi, version 6.8.3: http://www.haskell.org/ghc/
Loading package base ... linking ... done.
Prelude>

Pedro Vasconcelos DCC/FCUP Programação Funcional 1a Aula  Apresentação


Uso do interpretador
O símbolo > indica que o interpretador espera uma expressão
para calcular.
O interpretador calcula o valor resultante e imprime-o

> 2+3*5
17

> (2+3)*5
25

> sqrt (3^2 + 4^2)


5.0

Pedro Vasconcelos DCC/FCUP Programação Funcional 1a Aula  Apresentação


Alguns operadores e funções aritméticas
+ adição
- subtração
* multiplicação
/ divisão fraccionária
 potência (expoente inteiro)

div quociente (divisão inteira)


mod resto (divisão inteira)
sqrt raiz quadrada

== igualdade
/= diferença
< > <= >= relações de ordem

Pedro Vasconcelos DCC/FCUP Programação Funcional 1a Aula  Apresentação


Algumas convenções sintáticas I
Os argumentos de funções são separados por espaços
A aplicação tem maior precendência do que os operadores

Haskell Matemática
f x f (x )
f (g x) f (g (x ))
f (g x) (h x) f (g (x ), h(x ))
f x y + 1 f (x , y ) + 1
f x (y+1) f (x , y + 1)

sqrt x + 1 x +1

sqrt (x + 1) x +1

Pedro Vasconcelos DCC/FCUP Programação Funcional 1a Aula  Apresentação


Algumas convenções sintáticas II
Qualquer operador ⊕ pode ser tratado como uma função
binária escrevendo-o entre parêntesis curvos (⊕)
Reciprocamente: qualquer função binária f pode ser usada
como operador escrevendo-a entre contra-aspas `f `

(+) x y = x+y
(*) y 2 = y*2

x`mod`2 = mod x 2
f x `div` n = div (f x) n

Pedro Vasconcelos DCC/FCUP Programação Funcional 1a Aula  Apresentação


O prelúdio-padrão (standard Prelude)

O módulo Prelude contém um grande conjunto de funções


pré-denidas:
os operadores e funções aritméticas;
funções genéricas sobre listas
. . . e muitas outras.

O prelúdio-padrão pode ser usado em qualquer programa e é


carregado por omissão pelo interpretador.

Pedro Vasconcelos DCC/FCUP Programação Funcional 1a Aula  Apresentação


Algumas funções do prelúdio I
> head [1,2,3,4] obter o 1o elemento
1

> tail [1,2,3,4] remover o 1o elemento


[2,3,4]

> length [1,2,3,4,5] comprimento


5

> take 3 [1,2,3,4,5] obter um prexo


[1,2,3]

> drop 3 [1,2,3,4,5] remover um prexo


[4,5]

Pedro Vasconcelos DCC/FCUP Programação Funcional 1a Aula  Apresentação


Algumas funções do prelúdio II
> [1,2,3] ++ [4,5] concatenar
[1,2,3,4,5]

> reverse [1,2,3,4,5] inverter


[5,4,3,2,1]

> [1,2,3,4,5] !! 3 indexação


4

> sum [1,2,3,4,5] soma dos elementos


15

> product [1,2,3,4,5] produto dos elementos


120

Pedro Vasconcelos DCC/FCUP Programação Funcional 1a Aula  Apresentação


Denir novas funções

Vamos denir novas funções num cheiro de texto


Usamos um editor de texto externo (e.g. Emacs)
O nome do cheiro deve terminar em .hs (Haskell script)2

2 Alternativa: .lhs (literate Haskell script)


Pedro Vasconcelos DCC/FCUP Programação Funcional 1a Aula  Apresentação
Criar um cheiro de denições
Usando o editor, criamos um novo cheiro test.hs:
dobro x = x + x
quadruplo x = dobro (dobro x)

Usamos o comando :load para carregar estas denições no GHCi.

$ ghci
...
> :load teste.hs
[1 of 1] Compiling Main ( teste.hs, interpreted )
Ok, modules loaded: Main.

Pedro Vasconcelos DCC/FCUP Programação Funcional 1a Aula  Apresentação


Exemplos de uso

> dobro 2
4

> quadruplo 2
8

> take (quadruplo 2) [1..10]


[1,2,3,4,5,6,7,8]

Pedro Vasconcelos DCC/FCUP Programação Funcional 1a Aula  Apresentação


Modicar o cheiro
Acrescentamos duas novas denições ao cheiro:

factorial n = product [1..n]


media x y = (x+y)/2

No interpretador usamos :reload para carregar as modicações.

> :reload
...
> factorial 10
3628800
> media 2 3
2.5

Pedro Vasconcelos DCC/FCUP Programação Funcional 1a Aula  Apresentação


Comandos úteis do interpretador

:load ch carregar um cheiro


:reload re-carregar modicações
:edit editar o cheiro actual
:set editor prog denir o editor
:type expr mostrar o tipo duma expressão
:help obter ajuda
:quit terminar a sessão

Podem ser abreviados, e.g. :l em vez de :load.

Pedro Vasconcelos DCC/FCUP Programação Funcional 1a Aula  Apresentação


Identicadores
Os nomes de funções e argumentos devem começar por letras
mínusculas e podem incluir letras, dígitos, sublinhados e apóstrofes:

fun1 x_2 y' fooBar

As seguintes palavras reservadas não podem ser usadas como


identicadores:

case class data default deriving do else


if import in infix infixl infixr instance
let module newtype of then type where

Pedro Vasconcelos DCC/FCUP Programação Funcional 1a Aula  Apresentação


Denições locais
Podemos fazer denições locais usando where.

a = b+c
where b = 1
c = 2
d = a*2

A indentação indica o âmbito das declarações; também podemos


usar agrupamento explícito.

a = b+c
where {b = 1;
c = 2}
d = a*2

Pedro Vasconcelos DCC/FCUP Programação Funcional 1a Aula  Apresentação


Indentação

Todas as denições num mesmo âmbito devem começar na mesma


coluna.
a = 1 a = 1 a = 1

b = 2 b = 2 b = 2

c = 3 c = 3 c = 3
ERRADO ERRADO OK
A ordem das denições não é relevante.

Pedro Vasconcelos DCC/FCUP Programação Funcional 1a Aula  Apresentação


Comentários
Simples: começam por -- até ao nal da linha
Embricados: delimitados por {- e -}

-- factorial de um número inteiro


factorial n = product [1..n]

-- média de dois valores


media x y = (x+y)/2

{- ** as definições seguintes estão comentadas **


dobro x = x+x
quadrado x = x*x
-}

Pedro Vasconcelos DCC/FCUP Programação Funcional 1a Aula  Apresentação

Você também pode gostar