Algoritmos Recursivos

Fazer download em pdf ou txt
Fazer download em pdf ou txt
Você está na página 1de 7

Algoritmos Recursivos

Um objeto é denominado recursivo quando sua definição é parcialmente feita em termos dele
mesmo. A recursividade (ou recursão) é encontrada principalmente na matemática, mas está
presente em algumas situações do cotidiano. Por exemplo, quando um objeto é colocado entre dois
espelhos planos paralelos e frente a frente surge uma imagem recursiva, porque a imagem do objeto
refletida num espelho passa a ser o objeto a ser refletido no outro espelho e, assim, sucessivamente.
Em programação, a recursividade é um mecanismo útil e poderoso que permite a uma
função chamar a si mesma direta ou indiretamente, ou seja, uma função é dita recursiva se ela
contém pelo menos uma chamada explícita ou implícita a si própria.
A idéia básica de um algoritmo recursivo consiste em diminuir sucessivamente o problema
em um problema menor ou mais simples, até que o tamanho ou a simplicidade do problema reduzido
permita resolvê-lo de forma direta, sem recorrer a si mesmo. Quando isso ocorre, diz que o algoritmo
atingiu uma condição de parada, a qual deve estar presente em pelo menos um local dentro
algoritmo. Sem esta condição o algoritmo não pára de chamar a si mesmo, até estourar a
capacidade da pilha de execução, o que geralmente causa efeitos colaterais e até mesmo o término
indesejável do programa.
Para todo algoritmo recursivo existe um outro correspondente iterativo (não recursivo), que
executa a mesma tarefa. Implementar um algoritmo recursivo (partindo de uma definição recursiva do
problema) em uma linguagem de programação de alto nível como Pascal e C é simples e quase
imediato, pois o seu código é praticamente transcrito para a sintaxe da linguagem. Por essa razão,
em geral, os algoritmos recursivos possuem código mais claro (legível) e mais compacto do que os
correspondentes iterativos. Além disso, muitas vezes, é evidente a natureza recursiva do problema a
ser resolvido, como é o caso de problemas envolvendo árvores — estruturas de dados naturalmente
recursivas. Entretanto, também há desvantagens: i) algoritmos recursivos quase sempre consomem
mais recursos (especialmente memória, devido uso intensivo da pilha) do computador, logo tendem a
apresentar um desempenho inferior aos iterativos; e ii) algoritmos recursivos são mais difíceis de
serem depurados, especialmente quando existem muitas chamadas em espera na pilha de
execução.

Exemplo 1: Função Fatorial (!)


Esta função é um dos exemplos clássicos de recursividade e, por isso, de citação quase
obrigatória. Eis sua definição recursiva:

1 se n = 0
n! =
n ⋅ (n − 1)! se n > 0

Observe no Quadro 01 a facilidade em transformar a definição da função para Portugol e C.

UFGD/FCET – Prof. Wellington Lima dos Santos – Algoritmos e Estruturas de Dados II


Quadro 01 - Implementações recursivas, em Portugol e C, da função fatorial
função Fat(n : natural) : natural unsigned int Fat(unsigned int n)
início {
se n = 0 então if (n == 0)
retorne 1 return 1;
senão else
retorne n * Fat(n - 1) return n * Fat(n – 1);
fim }

pela definição, valor de 4! é calculado como:


4!=4*3!=4*(3*2!)=4*(3*(2*1!))=4*(3*(2*(1*0!)))=4*(3*(2*(1*1)))=24

Note que função é chamada recursivamente com argumento decrescente até chegar ao caso trivial
(0!), cujo valor é 1. Este caso trivial (condição de parada) encerra a seqüência de chamadas
recursivas. A seqüência de chamadas é melhor ilustrada abaixo:

4! = 4 * 3!
3! = 3 * 2! O contexto de cada chamada é
2! = 2 * 1! empilhado
1! = 1 * 0!
0! = 1 Condição de parada
1! = 1
2! = 2 O contexto de cada chamada é
3! = 6 desempilhado
4! = 24

Como n ! = n × ( n − 1) × ( n − 2) × × 3 × 2 × 1 , é muito simples implementar um algoritmo iterativo da


função fatorial. No Quadro 02, são apresentados dois algoritmos iterativos que se equivalem.

Quadro 02 - Implementações iterativas, em Portugol e C, da função fatorial


função Fat_NR(n : natural) : natural unsigned int Fat_NR(unsigned int n)
declare {
i, x : inteiro unsigned int i, x;
início x = 1;
x ← 1 for (i = 2; i <= n; i++)
para i ← 2 até n faça x = x * i;
x ← x * i return x;
retorne x }
fim

Exemplo 2: Seqüência de Fibonacci


A seqüência [0, 1, 1, 2, 3, 5, 8, 13, 21, ...] é conhecida como seqüência ou série de Fibonacci
e tem aplicações teóricas (principalmente em teoria dos números) e práticas, alguns padrões na
natureza parecem segui-la. Pode ser obtida através da definição recursiva:

0 se n = 0
Fib(n) = 1 se n = 1 ,
Fib(n − 1) + Fib(n − 2) se n > 1

UFGD/FCET – Prof. Wellington Lima dos Santos – Algoritmos e Estruturas de Dados II


a qual pode ser implementada como:

função Fib(n : natural) : natural


início
se n = 0 ou n = 1 então
retorne n
senão
retorne Fib(n-2) + Fib(n-1)
fim

Note que, para n > 1, cada chamada causa 2 novas chamadas de Fib, isto é, o número total
de chamadas cresce exponencialmente. Observe no diagrama de execução abaixo que para Fib(5),
são feitas 14 chamadas da função. Além disso, Fib(0) e Fib(1) são chamadas 3 vezes; Fib(1) é
chamada 5 vezes. Para Fib(25) são feitas 242784 chamadas recursivas!

F(5)

F(3) F(4)

F(1) F(2) F(2) F(3)

F(0) F(1) F(0) F(1) F(1) F(2)

F(0) F(1)

Figura 01 - Diagrama de execução de Fib(5), para ilustrar que o número de chamadas cresce
exponencialmente com o argumento de Fib.

No caso da seqüência de Fibonacci é relativamente simples implementar um algoritmo iterativo com


complexidade O(n), que tire proveito dos valores já calculados. Eis uma possibilidade:

função Fib_NR(n : natural) : natural


declare
i, F1, F2, F : natural
início
se n = 0 ou n = 1 então
retorne 1
senão
início
F1 ← 0
F2 ← 1
para i ← 1 até n - 1 faça
início
F ← F1 + F2
F1 ← F2
F2 ← F
fim
retorne F
fim
fim

UFGD/FCET – Prof. Wellington Lima dos Santos – Algoritmos e Estruturas de Dados II


Exemplo 3: Problema da Torre de Hanói
O problema ou quebra-cabeça conhecido como torre de Hanói foi publicado em 1883 pelo
matemático francês Edouard Lucas, também conhecido por seus estudos com a série Fibonacci.
Considerando que a torre A tenha n discos empilhados segundo diâmetros decrescentes, o problema
consiste em mover os n discos para torre C, com o mínimo de movimentos, usando a torre B como
auxiliar. As duas regras são simples: somente o disco superior de uma das torres pode movido e um
disco maior nunca pode ficar sobre outro menor.

6
5
4
3
2
1

Torre A Torre B Torre C


Figura 02 - Torres de Hanói.

Solução: Para entender melhor o problema, é interessante analisarmos os casos sucessivamente


mais complexos, começando pelo mais simples:
1º) A tem 1 disco – movemos A[1]→C[ ], isto é, movemos o disco 1 de A para C.
2º) A tem 2 discos – movemos A[2]→B[ ], A[1]→C[ ] e B[2]→C[1].
3º) A tem 3 discos – movemos A[2,3]→B[ ], A[1]→C[ ] e B[2,3]→C[1]. Mas pelas regras não podemos
mover A[2,3] (sub-torre A composta pelos discos 2 e 3) diretamente para B[ ], nem B[2, 3]→C[1].
Sem problemas! Vimos no 2º caso como mover uma torre com 2 discos de uma origem (A) para
uma torre destino (C), usando uma torre auxiliar (B). Para mover a sub-torre A[2,3] para B[ ],
usamos a torre C como auxiliar.
4º) A tem 4 discos – movemos A[2,3,4]→B[ ], A[1]→C[ ], B[2,3,4]→C[1]. Vimos no 3º caso como
mover três discos de uma origem para um destino.

nº) A tem n discos – movemos: A[2,3,4,...,n-1,n]→B[ ], A[1]→C[ ] e B[2,3,4,...,n-1,n]→C[ ].

Para solucionarmos o problema com n discos, precisamos da solução do problema com (n-1) discos,
que requer a solução do problema com (n-2) discos, e assim por diante, até chegarmos ao problema
com 2 discos que depende da solução trivial: mover apenas 1 disco de uma origem para um destino.
A solução para este problema é um algoritmo surpreendentemente simples:

UFGD/FCET – Prof. Wellington Lima dos Santos – Algoritmos e Estruturas de Dados II


procedimento MoveTorre(N : natural; Orig, Dest, Aux : caracter)
início
se N = 1 então
Escreva(“Movimento: ”, Orig, “ -> ”, Dest)
senão
início
MoveTorre(N - 1, Orig, Aux, Dest)
Escreva(“Movimento: ”, Orig, “ -> ”, Dest)
MoveTorre(N - 1, Aux, Dest, Orig)
fim
fim

Para resolvermos o problema com 3 discos basta “chamarmos” MoveTorre(3, ‘A’, ‘B’, ‘C’) para
obtermos a seguinte saída:

Movimento: A -> C
Movimento: A -> B
Movimento: C -> B
Movimento: A -> C
Movimento: B -> A
Movimento: B -> C
Movimento: A -> C

A B C A B C A B C A B C
1 2 3

A B C A B C A B C A B C
4 5 6 7

Figura 03 - Seqüência de movimentos para solucionar o problema da Torre de Hanói com 3 discos.

Observe que o algoritmo MoveTorre faz duas chamadas a si mesmo, portanto o número de
movimentos cresce exponencialmente com o número de discos. Mais precisamente, a solução do
problema com n discos requer 2n-1 movimentos. Assim, se uma pessoa se dispusesse a resolver o
problema com 25 discos, com dedicação exclusiva de 8 horas por dia e realizando um movimento por
segundo, esta “pobre criatura” gastaria 3354431 segundos, ou seja, mais de 3 anos para solucionar o
problema. Esse tempo dobraria a cada disco acrescentado!

RESUMO
Conceitos
- Um algoritmo é dito recursivo quando ele faz uma chamada a si mesmo, diretamente (podemos
ver o algoritmo sendo chamado dentro dele mesmo) ou indireta (o algoritmo chama um outro
algoritmo, que por sua vez invoca uma chamada ao primeiro).

UFGD/FCET – Prof. Wellington Lima dos Santos – Algoritmos e Estruturas de Dados II


- Um algoritmo recursivo deve ter pelo menos uma condição de parada, para que não seja
invocado indefinidamente. Esta condição de parada corresponde a instâncias suficiente
pequenas ou simples do problema original, que podem ser resolvidas diretamente.
- Para todo algoritmo recursivo existe pelo menos um algoritmo iterativo correspondente e vice-
versa. Todavia, muitas vezes pode ser difícil encontrar essa correspondência.

Vantagens
Os algoritmos recursivos normalmente são mais compactos, mais legíveis e mais fáceis de
serem compreendidos. Algoritmos para resolver problemas de natureza recursiva são fáceis de
serem implementados em linguagens de programação de alto nível.

Desvantagens
Por usarem intensivamente a pilha de execução, o que requer alocações e desalocações de
memória, os algoritmos recursivos tendem a ser mais lentos que os equivalentes iterativos, mas,
em muitos casos, pode valer a pena sacrificar a eficiência em benefício da clareza. Algoritmos
recursivos são mais difíceis de serem depurados durante a fase de desenvolvimento.

Aplicações
- Nem sempre a natureza recursiva do problema garante que um algoritmo recursivo seja a
melhor opção para resolvê-lo. O algoritmo recursivo para obter a seqüência de Fibonacci é um
ótimo exemplo disso.
- Algoritmos recursivos são aplicados em diversas situações como em: i) problemas envolvendo
manipulações de árvores; ii) analisadores léxicos recursivos de compiladores; e iii) problemas
que envolvem tentativa e erro ("Backtracking").

Exercícios
1) Considere a função Comb(n, k), que representa o número de grupos distintos com k pessoas que
podem ser formados a partir de n pessoas. Por exemplo, Comb(4, 3) = 4, pois com 4 pessoas (A,
B, C, D), é possível formar 4 diferentes grupos: ABC, ABD, ACD e BCD. Sabe-se que:

n se k = 1
Comb ( n, k ) = 1 se k = n
Comb(n − 1, k − 1) + Comb(n − 1, k ) se 1 < k < n
Implemente em portugol uma função recursiva para Comb (n, k) e mostre o diagrama de execução
para chamada Comb (5, 3). Sabendo-se ainda Comb (n, k) = n! / (k! (n-k)!), implemente uma função
não recursiva de Comb (n, k).

2) Implemente recursivamente uma função Max que retorne o maior valor armazenado em um vetor
V, contendo n números inteiros.

UFGD/FCET – Prof. Wellington Lima dos Santos – Algoritmos e Estruturas de Dados II


3) Dada a implementação em portugol da função abaixo:
função F(N : inteiro) : inteiro
início
se N < 4 então
F ← 3 N
senão
F ← 2 * F(N - 4) + 5
fim
Quais são os valores de F(3) e de F(7)?

4) O cálculo da raiz quadrada de um número real N pode ser feito através do seguinte algoritmo:

x se | x 2 − N |≤ ε
RaizQ ( N, x ,ε ) = x2 + N
RaizQ ( N, ,ε) caso contrário
2x
em que x é uma aproximação inicial do valor N e ε é um erro admissível. Implemente o
algoritmo em Portugol e mostre o diagrama de execução para a chamada RaizQ(13, 3.2, 0.001).

5) Dada a definição da função de Ackerman:

n +1 se m = 0
A(m, n ) = A(n − 1,1) se m > 0 e n = 0
A(m − 1, A( m, n − 1)) se m > 0 e n > 0
válida para valores inteiros não negativos de m e n, implemente uma versão recursiva do
algoritmo e faça o diagrama de execução de A(1, 2).

6) A função f(x, n) = xn, em que x é um número real e n um número inteiro, pode ser calculada
eficientemente como nos exemplos abaixo:
x0 = 1; x1 = x; x2 = x2; x3 = x x2; x4 = (x2)2 x5 = x (x2)2; x6 = (x x2)2; x11 = x ((x (x2)2)2; x-2 = 1/x2 etc.
Elabore a definição recursiva de f(x, n) e implemente um algoritmo recursivo para f(x,n).

7) A recursividade pode ser utilizada para gerar todas as possíveis permutações de um conjunto de
símbolos. Por exemplo, existem seis permutações no conjunto de símbolos A, B e C: ABC, ACB,
BAC, BCA, CBA e CAB. O conjunto de permutações de N símbolos é gerado tomando-se cada
símbolo por vez e prefixando-o a todas as permutações que resultam dos (N - 1) símbolos
restantes. Conseqüentemente, permutações num conjunto de símbolos podem ser especificadas
em termos de permutações num conjunto menor de símbolos. Formule um algoritmo recursivo
para este problema.

8) Considere uma partida de futebol entre duas equipes A x B, cujo placar final é m x n, em que m e n
são os números de gols marcados por A e B, respectivamente. Implemente um algoritmo
recursivo que imprima todas as possíveis sucessões de gols marcados. Por exemplo, para um
resultado de 3 x 1 as possíveis sucessões de gols são "A A A B”, "A A B A”, "A B A A" e "B A A A".

UFGD/FCET – Prof. Wellington Lima dos Santos – Algoritmos e Estruturas de Dados II

Você também pode gostar