Torna indietro   Hardware Upgrade Forum > Software > Programmazione

Recensione HUAWEI Mate X7: un foldable ottimo, ma restano i soliti problemi
Recensione HUAWEI Mate X7: un foldable ottimo, ma restano i soliti problemi
Mate X7 rinnova la sfida nel segmento dei pieghevoli premium puntando su un design ancora più sottile e resistente, unito al ritorno dei processori proprietari della serie Kirin. L'assenza dei servizi Google e del 5G pesa ancora sull'esperienza utente, ma il comparto fotografico e la qualità costruttiva cercano di compensare queste mancanze strutturali con soluzioni ingegneristiche di altissimo livello
Nioh 3: souls-like punitivo e Action RPG
Nioh 3: souls-like punitivo e Action RPG
Nioh 3 aggiorna la formula Team NINJA con aree esplorabili più grandi, due stili di combattimento intercambiabili al volo (Samurai e Ninja) e un sistema di progressione pieno di attività, basi nemiche e sfide legate al Crogiolo. La recensione entra nel dettaglio su combattimento, build, progressione e requisiti PC
Test in super anteprima di Navimow i220 LiDAR: il robot tagliaerba per tutti
Test in super anteprima di Navimow i220 LiDAR: il robot tagliaerba per tutti
La facilità di installazione e la completa automazione di tutte le fasi di utilizzo, rendono questo prodotto l'ideale per molti clienti. Ecco com'è andata la nostra prova in anteprima
Tutti gli articoli Tutte le news

Vai al Forum
Rispondi
 
Strumenti
Old 01-08-2010, 12:16   #1
anonimizzato
 
Messaggi: n/a
[Generico] chiarimento su funzione ricorsiva

Ciao a tutti,

avrei bisogno di un chiarimento, passo passo, per capire l'esatto funzionamento di una funzione ricorsiva.

Premesso che l'algoritmo e l'utilità mi è chiara, non riesco a focalizzare come venga costruito il risultato finale di tale funzione.

Mi spiego: prendiamo la classica serie di Fibonacci, risolta qui sotto con una funzione ricorsiva in Ruby.

Codice:
def fattoriale(x)
  risultato = 0
  if (x == 0)
    risultato = 1
  else
    risultato = x * fattoriale(x - 1)
  end
  risultato
end

puts fattoriale(4)
Quello che non riesco a stamparmi dentru u cevvelletto (Ras della fossa cit.) è come viene gestito ad ogni passaggio il valore nella variabile risultato dato che l'espressione

Codice:
risultato = x * fattoriale(x - 1)
non restituisce subito un valore a risultato (giusto?) ma richiama a sua volta la funzione.

Grazie in anticipo.
  Rispondi citando il messaggio o parte di esso
Old 01-08-2010, 12:23   #2
anonimizzato
 
Messaggi: n/a
Forse così: 4 x ( 3 x ( 2 x ( 1 x 1) ) ) ?


  Rispondi citando il messaggio o parte di esso
Old 01-08-2010, 12:45   #3
WarDuck
Senior Member
 
L'Avatar di WarDuck
 
Iscritto dal: May 2001
Messaggi: 12939
Il trucco è seguire passo passo le chiamate della funzione (prova ad eseguire la funzione "a mano" con carta e penna).

Supponendo che tu conosca bene il meccanismo delle chiamate a funzione (in generale), quello che devi notare è che ad ogni passo (se x > 0) viene chiamata la funzione con il parametro x decrementato di 1.

Questo significa che ad un certo punto dell'esecuzione arrivi al passo base, ovvero x = 0 (di fatto lì si interrompe la catena ricorsiva, perché la funzione ritorna 1).
WarDuck è offline   Rispondi citando il messaggio o parte di esso
Old 01-08-2010, 12:59   #4
anonimizzato
 
Messaggi: n/a
Quote:
Originariamente inviato da WarDuck Guarda i messaggi
Questo significa che ad un certo punto dell'esecuzione arrivi al passo base, ovvero x = 0 (di fatto lì si interrompe la catena ricorsiva, perché la funzione ritorna 1).

Si si questo l'avevo capito.

Ho provato a mettere un puts davanti a

Codice:
risultato = x * fattoriale(x - 1)
e se eseguo ad esempio fattoriale(4)

mi ritrovo in output:

Codice:
1
2
6
24
non riesco però a capire come venga memorizzato il valore finale di risultato.

Ad ogni chiamata infatti viene re-inizializzato a 0
  Rispondi citando il messaggio o parte di esso
Old 01-08-2010, 13:00   #5
anonimizzato
 
Messaggi: n/a
Quote:
Originariamente inviato da WarDuck Guarda i messaggi
Supponendo che tu conosca bene il meccanismo delle chiamate a funzione (in generale)
In che senso?
  Rispondi citando il messaggio o parte di esso
Old 01-08-2010, 13:03   #6
nuovoUtente86
Senior Member
 
Iscritto dal: Mar 2007
Messaggi: 7863
Quote:
Originariamente inviato da Sgurbat Guarda i messaggi
In che senso?
funzionamento dello stack
nuovoUtente86 è offline   Rispondi citando il messaggio o parte di esso
Old 01-08-2010, 13:06   #7
anonimizzato
 
Messaggi: n/a
Quote:
Originariamente inviato da nuovoUtente86 Guarda i messaggi
funzionamento dello stack
Conosco stack, queue ecc. ma solo superficialmente. Non ho fatto l'UNI.

Ho trovato questo:
Quote:
Una funzione ricorsiva "salva" il suo stato nel momento in cui richiama se' stessa, solitamente nello stack. Ogni volta che la ricorsione viene invocata, tutte le variabili presenti vengono inserite nello stack ed una nuova serie di variabili viene creata (sempre dallo stack).
http://www.soft-land.org/articoli/recurse
  Rispondi citando il messaggio o parte di esso
Old 01-08-2010, 13:24   #8
tuccio`
Senior Member
 
Iscritto dal: Apr 2010
Città: Frosinone
Messaggi: 416
lo stack è una parte della memoria riservata alla memoria statica e alle chiamate a funzione, ogni volta che una funzione viene chiamata, viene allocata la memoria di cui necessita nello stack, e quando la chiamata finisce viene liberata (si chiama stack proprio perché funziona con politica lifo)

ad esempio, ogni volta che chiami la funzione fattoriale, viene allocata una nuova variabile risultato (quindi l'indirizzo su cui viene scritto il risultato per n sarà diverso dall'indirizzo su cui lo scrive per n-1)

tutto quello che devi pensare quando scrivi una funzione ricorsiva è:
posso risolvere un problema servendomi della soluzione a un problema "più piccolo" dello stesso tipo, in modo da potermi fermare prima o poi perché conosco la soluzione di un caso base? il funzionamento delle stack, l'albero delle chiamate ti possono aiutare a renderti conto che la cosa funziona
tuccio` è offline   Rispondi citando il messaggio o parte di esso
Old 01-08-2010, 13:28   #9
anonimizzato
 
Messaggi: n/a
Quote:
Originariamente inviato da tuccio` Guarda i messaggi
lo stack è una parte della memoria riservata alla memoria statica e alle chiamate a funzione, ogni volta che una funzione viene chiamata, viene allocata la memoria di cui necessita nello stack, e quando la chiamata finisce viene liberata (si chiama stack proprio perché funziona con politica lifo)

ad esempio, ogni volta che chiami la funzione fattoriale, viene allocata una nuova variabile risultato (quindi l'indirizzo su cui viene scritto il risultato per n sarà diverso dall'indirizzo su cui lo scrive per n-1)

tutto quello che devi pensare quando scrivi una funzione ricorsiva è:
posso risolvere un problema servendomi della soluzione a un problema "più piccolo" dello stesso tipo, in modo da potermi fermare prima o poi perché conosco la soluzione di un caso base? il funzionamento delle stack, l'albero delle chiamate ti possono aiutare a renderti conto che la cosa funziona
Ok.

Mi potete confermare (o smentire) che, di fatto, il calcolo del risultato finale avviene partendo dalla chiamata più interna della ricorsione per, via via, tornare alla prima chiamata?

Cioè che in sostanza il calcolo avviene in una forma simile:

Codice:
(1x2x3x4)
e non

Codice:
(4x3x2x1)
come, magari, potrebbe sembrare inizialmente dato che il valore dell'argomento che passiamo alla funzione. quando chiamata nel nostro codice. è quello più alto (ad esempio 4) per:

Codice:
fattoriale(4)
Grazie ancora.

Ultima modifica di anonimizzato : 01-08-2010 alle 13:34.
  Rispondi citando il messaggio o parte di esso
Old 01-08-2010, 13:39   #10
tuccio`
Senior Member
 
Iscritto dal: Apr 2010
Città: Frosinone
Messaggi: 416
se vuoi sapere se è così nella funzione che hai scritto, la risposta è sì

non puoi "linearizzare" così però

diciamo che è più come dire 4! = 4 * 3! e poi prendere un pezzo di carta su cui calcolare 3! = 3 * 2!, finché non arrivi a 0! fattoriale che sai essere 1

Ultima modifica di tuccio` : 01-08-2010 alle 13:42.
tuccio` è offline   Rispondi citando il messaggio o parte di esso
Old 01-08-2010, 13:47   #11
anonimizzato
 
Messaggi: n/a
Quote:
Originariamente inviato da tuccio` Guarda i messaggi
se vuoi sapere se è così nella funzione che hai scritto, la risposta è sì
Perfetto, anzi per essere più preciso è forse così?

Codice:
(((1x1)x2)x3)x4
  Rispondi citando il messaggio o parte di esso
Old 01-08-2010, 13:51   #12
anonimizzato
 
Messaggi: n/a
Quote:
Originariamente inviato da tuccio` Guarda i messaggi
non puoi "linearizzare" così però

diciamo che è più come dire 4! = 4 * 3! e poi prendere un pezzo di carta su cui calcolare 3! = 3 * 2!, finché non arrivi a 0! fattoriale che sai essere 1
Ok, ora forse mi è più chiaro.

Anche scritta in questo modo infatti si intuisce la ricorsione.

Codice:
4 * 3! 
3! = 3 x 2! 
2! = 2 x 1!
  Rispondi citando il messaggio o parte di esso
Old 01-08-2010, 13:55   #13
nuovoUtente86
Senior Member
 
Iscritto dal: Mar 2007
Messaggi: 7863
Devi avere ben chiaro in mente come vengono allocati i record di attivazione delle varie chiamate a funzioni e le successive deallocazioni. Ha poco senso semplificare, come ti è stato detto, la cosa ad una visione lineare.

Ultima modifica di nuovoUtente86 : 01-08-2010 alle 14:01.
nuovoUtente86 è offline   Rispondi citando il messaggio o parte di esso
Old 01-08-2010, 13:58   #14
anonimizzato
 
Messaggi: n/a
Quote:
Originariamente inviato da nuovoUtente86 Guarda i messaggi
Devi avere ben chiaro in mente come vengono allocati i record di attivazione delle varie chiamate a funzioni e le successive deallocazioni. Ha poco senso semplificare, come ti è stato detto, la cosa ad una visione lineare della cosa.
Comprendo.

Nel caso hai qualche riferimento online dove mi possa studiare meglio la cosa?

Grazie ancora.
  Rispondi citando il messaggio o parte di esso
Old 01-08-2010, 14:00   #15
nuovoUtente86
Senior Member
 
Iscritto dal: Mar 2007
Messaggi: 7863
Quote:
prendiamo la classica serie di Fibonacci
giusto per complettezza, la serie di Fibonacci (Fn= Fn-1 + Fn-2 ) e la funzione fattoriale non sono la stessa cosa.
nuovoUtente86 è offline   Rispondi citando il messaggio o parte di esso
Old 01-08-2010, 14:12   #16
Albi89
Senior Member
 
Iscritto dal: May 2004
Città: Napoli
Messaggi: 773
Quote:
Originariamente inviato da nuovoUtente86 Guarda i messaggi
giusto per complettezza, la serie di Fibonacci (Fn= Fn-1 + Fn-2 ) e la funzione fattoriale non sono la stessa cosa.
Che poi a stretto rigore sarebbe una successione
__________________
If builders built buildings the way programmers wrote programs, then the first woodpecker that came along would destroy civilization.
--Gerald Weinberg
Albi89 è offline   Rispondi citando il messaggio o parte di esso
Old 01-08-2010, 15:50   #17
anonimizzato
 
Messaggi: n/a
Quote:
Originariamente inviato da nuovoUtente86 Guarda i messaggi
giusto per complettezza, la serie di Fibonacci (Fn= Fn-1 + Fn-2 ) e la funzione fattoriale non sono la stessa cosa.
Hai assolutamente ragione.

Ho usato l'esempio della serie di Fibonacci giusto come esempio per parlare delle funzioni ricorsive.
  Rispondi citando il messaggio o parte di esso
Old 01-08-2010, 18:52   #18
cionci
Senior Member
 
L'Avatar di cionci
 
Iscritto dal: Apr 2000
Città: Vicino a Montecatini(Pistoia) Moto:Kawasaki Ninja ZX-9R Scudetti: 29
Messaggi: 53971
Quote:
Originariamente inviato da nuovoUtente86 Guarda i messaggi
Devi avere ben chiaro in mente come vengono allocati i record di attivazione delle varie chiamate a funzioni e le successive deallocazioni. Ha poco senso semplificare, come ti è stato detto, la cosa ad una visione lineare.
Secondo me no. A cosa gli serve il record di attivazione ?
Gli basta sapere ad ogni istante i valori delle variabili locali e dei parametri attuali.

Sgurbat: ribadisco, come ti era già stato consigliato, di seguire passo su passo su carta l'intera esecuzione con valori semplici
Prendendo l'esempio sopra, per scriverlo su carta scrivi il nome della funzione con i parametri attuali, seguita dalle operazioni eseguite e per ogni ricorsione indenta.
Codice:
fattoriale(4):
risultato = 4 * fattoriale(3)
   fattoriale(3):
   risultato = 3 * fattoriale(2)
     fattoriale(2):
     risultato = 2 * fattoriale(1)
       fattoriale(1):
       risultato = 1 * fattoriale(0)
         fattoriale(0):
         risultato = 1
       risultato = 1
     risultato = 2
   risultato = 6    
risultato = 24
Puoi anche associare il funzionamento della ricorsione ad una dimostrazione per induzione.
Ad esempio:

Passo 0: - fattoriale(0) = 1
Passo 1: - fattoriale(1) = 1 * fattoriale(0)
Passo n: - fattoriale(n) = n * fattoriale(n-1)

In questo caso c'è un solo passo 0, che corrisponde ad una condizione di arresto. Nel caso ci fossero diverse condizioni di arresto, dovrai esplorare un passo 0 ed un passo 1 per ogni condizione di arresto. Il passo 1 corrisponde ad una chiamata della funzione che genera una ulteriore singola chiamata ricorsiva, la quale mette in atto la condizione di arresto da testare.
Questo modo di pensare ti tornerà molto utile anche per scrivere le funzioni ricorsive, non solo per capirle.

Ultima modifica di cionci : 01-08-2010 alle 19:15.
cionci è offline   Rispondi citando il messaggio o parte di esso
Old 02-08-2010, 12:34   #19
nuovoUtente86
Senior Member
 
Iscritto dal: Mar 2007
Messaggi: 7863
Quote:
Secondo me no. A cosa gli serve il record di attivazione ?
Gli basta sapere ad ogni istante i valori delle variabili locali e dei parametri attuali.
perchè dalla sua prima richiesta sembra che i dubbi non siano tanto sulla natura algoritmica del problema, quanto proprio sulla gestione delle chiamate. Probabilmente capire bene cosa avviene a basso livello, può fugare le incertezze. Ragionando ad un livello più astratto, probabilmente, nell' immediato e per funzioni semplici si ha una comprensione più veloce ma anche meno approfondita e stabile.
nuovoUtente86 è offline   Rispondi citando il messaggio o parte di esso
Old 02-08-2010, 13:06   #20
cionci
Senior Member
 
L'Avatar di cionci
 
Iscritto dal: Apr 2000
Città: Vicino a Montecatini(Pistoia) Moto:Kawasaki Ninja ZX-9R Scudetti: 29
Messaggi: 53971
Secondo me ha poco senso conoscere il record di attivazione per comprendere come funzionano le chiamate ricorsive, anche perché il formato del record di attivazione non è standard e dipende dal compilatore e dalla macchina fisica/virtuale per cui si va a compilare/interpretare il linguaggio.
Giusto per fare un esempio comune: il formato del record di attivazione varia ad esempio fra IA32 e x86-64... Ad esempio in x86-64 non esiste un vero e proprio record di attivazione, ma per chiamate fino a, mi sembra, 8 parametri si fa uso dei registri general purpose per passare i parametri. Quindi anche semplicemente parlare di record di attivazione e stack in questo caso perde di significato.
Poi se si va a parlare di linguaggi interpretati/compilati JIT come nel caso sopra (è Ruby se non erro) le cose si fanno ancora più complicate...

Ultima modifica di cionci : 02-08-2010 alle 13:09.
cionci è offline   Rispondi citando il messaggio o parte di esso
 Rispondi


Recensione HUAWEI Mate X7: un foldable ottimo, ma restano i soliti problemi Recensione HUAWEI Mate X7: un foldable ottimo, m...
Nioh 3: souls-like punitivo e Action RPG Nioh 3: souls-like punitivo e Action RPG
Test in super anteprima di Navimow i220 LiDAR: il robot tagliaerba per tutti Test in super anteprima di Navimow i220 LiDAR: i...
Dark Perk Ergo e Sym provati tra wireless, software via browser e peso ridotto Dark Perk Ergo e Sym provati tra wireless, softw...
DJI RS 5: stabilizzazione e tracking intelligente per ogni videomaker DJI RS 5: stabilizzazione e tracking intelligent...
Al centro della Via Lattea ci potrebbe e...
Elon Musk ora guarda alla Luna: SpaceX p...
La Cina ha lanciato nuovamente lo spazio...
Blue Origin potrebbe realizzare il lande...
Artemis II: il prossimo Wet Dress Rehear...
Il nuovo HONOR 600 sta arrivando e avr&a...
La crisi delle memorie non coinvolger&ag...
Windows domina su Steam, ma molti utenti...
Per non incorrere in nuovi aumenti delle...
Cubi Z AI 8M visto da vicino, un mini-PC...
Datacenter nello Spazio, affascinante ma...
Social e minori, Butti apre al dibattito...
Tutte le offerte Amazon del weekend, sol...
Amazon spinge sull'usato garantito: 10% ...
TikTok rischia una maxi-multa in Europa:...
Chromium
GPU-Z
OCCT
LibreOffice Portable
Opera One Portable
Opera One 106
CCleaner Portable
CCleaner Standard
Cpu-Z
Driver NVIDIA GeForce 546.65 WHQL
SmartFTP
Trillian
Google Chrome Portable
Google Chrome 120
VirtualBox
Tutti gli articoli Tutte le news Tutti i download

Strumenti

Regole
Non Puoi aprire nuove discussioni
Non Puoi rispondere ai messaggi
Non Puoi allegare file
Non Puoi modificare i tuoi messaggi

Il codice vB è On
Le Faccine sono On
Il codice [IMG] è On
Il codice HTML è Off
Vai al Forum


Tutti gli orari sono GMT +1. Ora sono le: 00:11.


Powered by vBulletin® Version 3.6.4
Copyright ©2000 - 2026, Jelsoft Enterprises Ltd.
Served by www3v