Programmazione ricorsiva.

Documenti analoghi
Programmazione ricorsiva: cenni

Programmazione ricorsiva: cenni

Variable-lenght Array (VLA)

LA RICORSIONE IN C. CdL Ingegneria Informatica n.o. Anno Accademico 2006/07 Fondamenti di Informatica I corso A Giacomo Piscitelli pag.

Programmazione ricorsiva: cenni

Il linguaggio C. Funzioni

Ricorsione in C. slides credit Prof. Paolo Romano

Corso di Laurea Ingegneria Informatica Fondamenti di Informatica

n n 1 n = > Il calcolo del fattoriale La funzione fattoriale, molto usata nel calcolo combinatorio, è così definita

RICORSIONE. Informatica B - A.A. 2013/2014

Ricorsione. Lucidi a cura di Gianpaolo Cugola, Carlo Ghezzi, Gian Pietro Picco Dipartimento di Elettronica e Informazione

Lezione 8 programmazione in Java. Anteprima. La ricorsione. Nicola Drago Dipartimento di Informatica Università di Verona

Funzioni e Ricorsione

Gestione della memoria

Programmazione ricorsiva

Corso di Fondamenti di Programmazione canale E-O. Un esempio. Funzioni ricorsive. La ricorsione

Capitolo 19. Ricorsione

Esercitazione 5. Procedure e Funzioni Il comando condizionale: switch

Esercitazione 6. Array

Ordinamenti ricorsivi

Pile Le pile: specifiche e realizzazioni attraverso rappresentazioni sequenziali e collegate. Pile e procedure ricorsive

Laboratorio di Informatica L-A 1

Corso di Laurea Ingegneria Informatica Fondamenti di Informatica 2

Algoritmi di ordinamento

Ricorsione. La ricorsione consiste nella possibilità di definire una funzione in termini di se stessa

Alcuni Soluzioni. Fornire la specifica di un algoritmo risolutivo utilizzando lo pseudocodice visto a lezione.

1) Somma dei primi n numeri naturali: 3) Ricerca di un elemento el in una sequenza di interi: FONDAMENTI DI INFORMATICA II Ricorsione 2

Algoritmi di ordinamento

Ricorsione. Moreno Marzolla Dipartimento di Informatica Scienza e Ingegneria (DISI) Università di Bologna

FUNZIONI RICORSIVE PROGRAMMAZIONE RICORSIVA: Esempi di problemi ricorsivi:

Fondamenti di Informatica

Ricorsione. Moreno Marzolla Dipartimento di Informatica Scienza e Ingegneria (DISI) Università di Bologna

n n 1 n = > Il calcolo del fattoriale La funzione fattoriale, molto usata nel calcolo combinatorio, è così definita

Esercitazione 7. Soluzione:

Fondamenti di Informatica 6. Algoritmi e pseudocodifica

Ricerca in una sequenza ordinata

Corso di Informatica

La Ricorsione. Carla Binucci e Walter Didimo

Ricorsione. Libro cap. 5 (da 5.14 in poi)

Laboratorio di Programmazione M-Z

La ricorsione. Induzione nel progetto e nella verifica di programmi ricorsivi. Le forme della ricorsione

RICORSIONE RICORSIONE

Laboratorio di Programmazione Appunti sulla lezione 5: Algoritmi di ordinamento (cont.) Alessandra Raffaetà

La principale modalità di calcolo è l applicazione di funzioni

LA RICORSIONE LA RICORSIONE LA RICORSIONE: ESEMPIO LA RICORSIONE: ESEMPIO LA RICORSIONE: ESEMPIO LA RICORSIONE: ESEMPIO

Fondamenti di Informatica T. Linguaggio C: Stack e Ricorsione

Tipi di dato strutturati: Array

CORSO DI PROGRAMMAZIONE

Programmazione a moduli in C

Corso di Fondamenti di Informatica. La ricorsione

Programmazione a Oggetti e JAVA. Prof. B.Buttarazzi A.A. 2012/2013

LA RICORSIONE. Una funzione matematica è definita ricorsivamente quando nella sua definizione compare un riferimento a se stessa

Programmazione II Università di Roma "La Sapienza" Appunti a cura della Prof.ssa FACHINI. Ricorsione per il "problem solving" Il problema del cambio.

Università degli Studi di Ferrara

Esercitazione 11. Liste semplici

Corso di Laurea Ingegneria Informatica Fondamenti di Informatica

Ricorsione. Moreno Marzolla Dipartimento di Informatica Scienza e Ingegneria (DISI) Università di Bologna

Un tipico esempio è la definizione del fattoriale n! di un numero n, la cui definizione è la seguente:

Ricorsione. Corso di Fondamenti di Informatica

Matlab: funzioni. Daniele Loiacono, Vittorio Zaccaria

Pensiero Algoritmico. Lezione 3 23 Novembre Ripasso. Anatomia di un programma. Anatomia di un programma. Ozalp Babaoglu Università di Bologna

La ricorsione. Ver Claudio Fornaro - Corso di programmazione in C

La programmazione nel linguaggio C

Prof. E. Occhiuto INFORMATICA 242AA a.a. 2010/11 pag. 1

Esercizio 2 (punti 7) Dato il seguente programma C: #include <stdio.h> int swap(int * nome, int length);

Operativamente, risolvere un problema con un approccio ricorsivo comporta

passaggio di vettori come parametri di funzioni/procedure. I Quando si passa un vettore come parametro ad una funzione, in

Fondamenti di Informatica - 1. Prof. B.Buttarazzi A.A. 2011/2012

ERRATA CORRIGE. void SvuotaBuffer(void); void SvuotaBuffer(void) { if(getchar()!=10) {svuotabuffer();} }

Ricorsione. Libro cap. 5 (da 5.14 in poi)

Tipi di dato strutturati: Array

Istruzioni iterative. Istruzioni iterative

Esercitazione 7. Procedure e Funzioni

Algoritmi di ordinamento: Array e ricorsione

RICORSIONE - schema ricorsivo (o induttivo) si esegue l'azione S, su un insieme di dati D, mediante eventuale esecuzione di

INFORMATICA DI BASE Linguaggio C Prof. Andrea Borghesan

Ricorsione. Unità 5. Domenico Daniele Bloisi. Corso di Fondamenti di Informatica Ingegneria delle Comunicazioni BCOR Ingegneria Elettronica BELR

a.a Codice corso: 21012

Dott. R. Gori, Dott. A. Rama INFORMATICA 242AA a.a. 2011/12 pag. 1. Le istruzioni iterative permettono di ripetere determinate azioni più volte:

Programmazione I - Laboratorio

ESERCIZIO: Analisi di un programma. Dato il seguente programma C:

Operazioni su file di caratteri

Esercitazioni di Tecniche di Programmazione. 6. Sesta esercitazione autoguidata: file binari e ricorsione

Esercizio 1 Liste: calcolo del numero di elementi ripetuti in una lista

liste ogni nodo ha un successore, tranne l ultimo della lista che ne ha zero; alberi binari ogni nodo ha zero, uno oppure due figli

Inserimento di un elemento in coda. quale va collegato quello nuovo. ultimo. *lista 8 3. aux. ultimo. *lista 8 3. aux

Informatica Generale Andrea Corradini Algoritmi: ordinamento per inserimento e ricorsione

Ricorsione. Emilio Di Giacomo e Walter Didimo

Laboratorio di Python

Esercizio 1 - cicli. Esercizio 1 cicli - Soluzione

a.a Codice corso: 21012

Linguaggio C: Array Valeria Cardellini

Le Funzioni e la Ricorsione

Introduzione al C. Lez. 2 Funzioni e Puntatori

Esercitazione 3. Oggi sono stati effettuati semplici esercizi in C utilizzando le seguenti istruzioni: if...else, while, printf, scanf

Transcript:

In quasi tutti i linguaggi di programmazione evoluti è ammessa la possibilità di definire funzioni/procedure ricorsive: durante l esecuzione di una funzione F è possibile chiamare la funzione F stessa. Ciò può avvenire direttamente: il corpo di F contiene una chiamata a F stessa. indirettamente: F contiene una chiamata a G che a sua volta contiene una chiamata a F. Questo può sembrare strano: se pensiamo che una funzione è destinata a risolvere un sottoproblema P, una definizione ricorsiva sembra indicare che per risolvere P dobbiamo... saper risolvere P! Dott. R. Gori INFORMATICA 242AA a.a. 2015/16 pag. 1

In realtà, la programmazione ricorsiva si basa sull osservazione che per molti problemi la soluzione per un caso generico può essere ricavata sulla base della soluzione di un altro caso, generalmente più semplice, dello stesso problema. La programmazione ricorsiva trova radici teoriche nel principio di induzione ben fondata che può essere visto come una generalizzazione del principio di induzione sui naturali La soluzione di un problema viene individuata supponendo di saperlo risolvere su casi più semplici. Bisogna poi essere in grado di risolvere direttamente il problema sui casi più semplici di qualunque altro. Dott. R. Gori INFORMATICA 242AA a.a. 2015/16 pag. 2

Esempio: Torre di Hanoi (leggenda Vietnamita). A B C pila di dischi di dimensione decrescente su un perno A vogliamo spostarla sul perno C, usando un perno di appoggio B vincoli: possiamo spostare un solo disco alla volta un disco più grande non può mai stare su un disco più piccolo secondo la leggenda: i monaci stanno spostando 64 dischi: quando avranno finito, ci sarà la fine del mondo Dott. R. Gori INFORMATICA 242AA a.a. 2015/16 pag. 3

Come individuare una soluzione per un numero N di dischi arbitrario? per N=1 la soluzione è immediata: spostiamo l unico disco da A a C se sappiamo risolvere il problema per N=1 lo sappiamo risolvere anche per N=2: come? A B C A B C A B C A B C Notiamo l utilizzo del perno ausiliario B Dott. R. Gori INFORMATICA 242AA a.a. 2015/16 pag. 4

Possiamo generalizzare il ragionamento? Se sappiamo risolvere il problema per N dischi, possiamo individuare una soluzione per lo stesso problema ma con N+1 dischi? N A B C A B C N A B C A B C Dott. R. Gori INFORMATICA 242AA a.a. 2015/16 pag. 5

Formalizziamo il ragionamento Indichiamo con hanoi(n, P1, P2, P3) il problema: spostare N dischi dal perno P1 al perno P2 utilizzando P3 come perno d appoggio. hanoi(n, P1, P2, P3) if (N=1) sposta da P1 a P2; else hanoi(n-1, P1, P3, P2); sposta da P1 a P2; hanoi(n-1, P3, P2, P1); Dott. R. Gori INFORMATICA 242AA a.a. 2015/16 pag. 6

Esempio: Soluzione di hanoi(3,a,c,b) hanoi(2,a,b,c) = hanoi(3,a,c,b) = sposta(a, C) hanoi(2,b,c,a) = hanoi(1,a,c,b) = sposta(a,b) hanoi(1,c,b,a) = hanoi(1,b,a,c) = sposta(b,c) hanoi(1,a,c,b) = sposta(a,c) sposta(c,b) sposta(b,a) sposta(a,c) Dott. R. Gori INFORMATICA 242AA a.a. 2015/16 pag. 7

Le funzioni ricorsive sono convenienti per implementare funzioni matematiche definite in modo induttivo. Esempio: Definizione induttiva di somma tra due interi non negativi: x se y=0 somma(x, y) = 1 + (somma(x, y 1)) se y > 0 La somma di x con 0 viene definita in modo immediato; la somma di x con il successore di y viene definita come il successore della somma tra x e y. Esempio: somma di 3 e 2: somma(3, 2) = 1 + (somma(3, 1)) = 1 + (1 + (somma(3, 0))) = 1 + (1 + (3)) = 1 + 4 = 5 Dott. R. Gori INFORMATICA 242AA a.a. 2015/16 pag. 8

Esempio: Funzione fattoriale. definizione iterativa: fatt(n) = n (n 1) (n 2) 2 1 definizione induttiva: 1 se n = 0 (caso base) fatt(n) = n fatt(n 1) se n > 0 (caso induttivo) È essenziale il fatto che, applicando ripetutamente il caso induttivo, ci riconduciamo prima o poi al caso base. fatt(3) = 3 fatt(2) = 3 (2 fatt(1)) = 3 (2 (1 fatt(0))) = 3 (2 (1 1)) = 3 (2 1) = 3 2 = 6 Dott. R. Gori INFORMATICA 242AA a.a. 2015/16 pag. 9

Il codice delle due diverse versioni definizione iterativa: int fatt(int n) int i,ris; ris=1; for (i=1;i<=n;i++) ris=ris*i; return ris; definizione ricorsiva: int fattric(int n) if (n == 0) return 1; else return n * fattric(n-1); Dott. R. Gori INFORMATICA 242AA a.a. 2015/16 pag. 10

Esempio: Programma che usa una funzione ricorsiva. #include <stdio.h> int fattric (int); main() int x, f; scanf("%d", &x); f = fattric(x); printf("fattoriale di %d: %d\n", x, f); int fattric(int n) int ris; if (n == 0) ris = 1; else ris = n * fattric(n-1); return ris; Dott. R. Gori INFORMATICA 242AA a.a. 2015/16 pag. 11

Evoluzione della pila (supponendo x=3). x 3 f? n 3 x 3 f? n 2 n 3 x 3 f? n 1 n 2 n 3 x 3 f? n 0 n 1 n 2 n 3 x 3 f? n 0 ris 1 n 1 n 2 n 3 x 3 f? n 1 ris 1 n 2 n 3 x 3 f? n 2 ris 2 n 3 x 3 f? n 3 ris 6 x 3 f? x 3 f 6 Dott. R. Gori INFORMATICA 242AA a.a. 2015/16 pag. 12

Esempio: Leggere una sequenza di caratteri terminata da \n e stamparla invertita. Ad esempio: casa = asac Problema: prima di poter iniziare a stampare dobbiamo aver letto e memorizzato tutta la sequenza: 1. usando una struttura dati opportuna ma dinamica (liste, le vedremo più avanti) 2. usando un procedimento ricorsivo. leggiamo un carattere della sequenza, c1, leggiamo e stampiamo ricorsivamente il resto della sequenza c2...cn e infine stampiamo c1; il caso base è rappresentato dalla lettura del carattere di fine sequenza. void invertinputric() char ch; ch = getchar(); if (ch!= \n ) invertinputric(); putchar(ch); else printf("sequenza invertita: "); Dott. R. Gori INFORMATICA 242AA a.a. 2015/16 pag. 13

main() printf("immetti una sequenza di caratteri\n"); invertinputric(); printf("\n"); Vediamo come evolve la pila per l input ABC\n ch A ch B ch A ch ch ch C B A ch \n ch C ch B ch A ch ch ch C B A ch ch B A ch A L output prodotto è il seguente Sequenza invertita: CBA Dott. R. Gori INFORMATICA 242AA a.a. 2015/16 pag. 14

Ricorsione multipla Si ha ricorsione multipla quando un attivazione di una funzione può causare più di una attivazione ricorsiva della stessa funzione (es. torre di Hanoi) Esempio: Definizione induttiva dei numeri di Fibonacci. F (0) = 0 F (1) = 1 F (n) = F (n 2) + F (n 1) se n > 1 F (0), F (1), F (2),... è detta sequenza dei numeri di Fibonacci: 0, 1, 1, 2, 3, 5, 8, 13, 21,... Dott. R. Gori INFORMATICA 242AA a.a. 2015/16 pag. 15

#include <stdio.h> int fibonacci (int); main() int n; printf("inserire un intero >= 0: "); scanf("%d", &n); printf("numero %d di Fibonacci: %d\n", n, fibonacci(n)); int fibonacci(int i) int ris; if (i == 0) ris = 0; else if (i == 1) ris = 1; else ris = fibonacci(i-1) + fibonacci(i-2); return ris; Dott. R. Gori INFORMATICA 242AA a.a. 2015/16 pag. 16

Esempi di funzioni ricorsive Tradurre in C la definizione induttiva già vista: x se y = 0 somma(x, y) = 1 + (somma(x, y 1)) se y > 0 int somma (int x, int y) int ris; if (y==0) ris = x; else ris = 1 + somma(x, y-1); return ris; Dott. R. Gori INFORMATICA 242AA a.a. 2015/16 pag. 17

Calcolo ricorsivo di x y (si assume y 0) x y 1 se y = 0 = x x y 1 altrimenti int exp (int x, int y) int ris; if (y==0) ris = 1; else ris = x * exp(x, y-1); return ris; Dott. R. Gori INFORMATICA 242AA a.a. 2015/16 pag. 18

Calcolare ricorsivamente la somma degli elementi di un array v compreso tra gli indici j e dim-1. Esprimiamo formalmente quanto richiesto: d 1 sumvet(v, j, dim 1) = v[i] i=j È evidente che: d 1 0 se dim = 0 v[i] = v[j] + d 1 v[i] se dim > 0 i=j i=j+1 La traduzione in C è immediata. Dott. R. Gori INFORMATICA 242AA a.a. 2015/16 pag. 19

int sumvet(int *v, int dim) if (dim==0) return 0; else return v[0] + sumvet(v+1,dim-1); int sumvet(int *v, int dim) int somma; if (dim==0) somma = 0; else somma = v[0] + sumvet(v+1,dim-1); return somma; Dott. R. Gori INFORMATICA 242AA a.a. 2015/16 pag. 20

Calcolare ricorsivamente il numero di occorrenze dell elemento x nell array v compreso tra gli indici j e dim-1. f (v, x, j, dim) = #i [j, dim 1] v[i] = x Anche in questo caso ragioniamo induttivamente: 0 se dim = 0 f (v, x, j, dim) = f (v, x, j + 1, dim) se dim > 0 v[j] x 1 + f (v, x, j + 1, dim) se dim > 0 v[j] = x Dott. R. Gori INFORMATICA 242AA a.a. 2015/16 pag. 21

int occorrenze (int *v, int occ; int dim) if (dim==0) occ= 0; else if (v[0]!=x) occ = occorrenze(v+1,x,dim-1); else occ = 1+occorrenze(v+1,x,dim-1); Dott. R. Gori INFORMATICA 242AA a.a. 2015/16 pag. 22

Scrivere una procedura ricorsiva che inverte la porzione di un array individuata dagli indici from e to. Vogliamo ottenere: Induttivamente:... 1 2 3 4 5... from to... 5 4 3 2 1...... 1 2 3 4 5... from to Dott. R. Gori INFORMATICA 242AA a.a. 2015/16 pag. 23

Scrivere una procedura ricorsiva che inverte la porzione di un array individuata dagli indici from e to. Vogliamo ottenere: Induttivamente:... 1 2 3 4 5... from to... 5 4 3 2 1...... 5 2 3 4 1... Questa situazione corrisponde alla chiamata ricorsiva su una porzione più piccola del vettore Dott. R. Gori INFORMATICA 242AA a.a. 2015/16 pag. 24

void swap(int *v, int i, int j) int temp; temp = v[i]; v[i] = v[j]; v[j] = temp; void invertiric (int *v, int from, int to) if (from < to) swap(v, from, to); invertiric(v, from+1, to-1); Si noti che la procedura non fa niente se la porzione individuata dal secondo e terzo parametro è vuota (from>to) o contiene un solo elemento (from=to) Dott. R. Gori INFORMATICA 242AA a.a. 2015/16 pag. 25