Quello che mi sembra è che ti venga chiesto di restituire l'ennesima fibonacci n., Dove n è il parametro passato. È possibile utilizzare vari metodi per rispondere a questa domanda, mentre tutti questi variano in termini di complessità temporale e complessità del codice.
Metodo 1 (Utilizzare la ricorsione) Un metodo semplice che è una relazione di verifica matematica dell'implementazione diretta recusriva sopra riportata.
int fib(int n)
{
if (n <= 1)
return n;
return fib(n-1) + fib(n-2);
}
Tempo Complessità: T (n) = T (n-1) + T (n-2), che è esponenziale. Possiamo osservare che questa implementazione fa un sacco di lavoro ripetuto (vedere il seguente albero di ricorsione). Quindi questa è una cattiva implementazione per il n ° numero di Fibonacci.
fib(5)
/ \
fib(4) fib(3)
/ \ / \
fib(3) fib(2) fib(2) fib(1)
/ \ / \ / \
fib (2) fib (1) fib (1) fib (0) fib (1) fib (0) /\ fib (1) fib (0) Spazio Extra: O (n) se consideriamo la dimensione dello stack di chiamata di fuinction, altrimenti O (1).
Metodo 2 (Usa programmazione dinamica) È possibile evitare il lavoro ripetuto eseguito con il metodo 1 memorizzando i numeri di Fibonacci calcolati finora.
int fib(int n)
{
/* Declare an array to store fibonacci numbers. */
int f[n+1];
int i;
/* 0th and 1st number of the series are 0 and 1*/
f[0] = 0;
f[1] = 1;
for (i = 2; i <= n; i++)
{
/* Add the previous 2 numbers in the series
and store it */
f[i] = f[i-1] + f[i-2];
}
return f[n];
}
Tempo Complessità: O (n) Spazio Extra: O (n)
Metodo 3 (Spazio Otimized Metodo 2) possiamo ottimizzare lo spazio utilizzato nel metodo 2 memorizzando i due numeri precedenti solo perché questo è tutto ciò che serve per ottenere il prossimo numero di Fibannaci in serie.
int fib(int n)
{
int a = 0, b = 1, c, i;
if(n == 0)
return a;
for (i = 2; i <= n; i++)
{
c = a + b;
a = b;
b = c;
}
return b;
}
Tempo Complessità: O (n) Spazio Extra: O (1)
Metodo 4 (Usando il potere della MATRX {{1,1}, {0,1}}) Questa un altro O (n) che si basa sul fatto che se moltiplichiamo n volte la matrice M = {{1,1}, {0,1}} a se stessa (in altre parole calcoliamo il potere (M, n)), allora ottieni il numero (n + 1) di Fibonacci come elemento a riga e colonna (0, 0) nella matrice risultante.
La rappresentazione matriciale dà la seguente espressione chiuso per i numeri di Fibonacci:
/* Helper function that multiplies 2 matricies F and M of size 2*2, and
puts the multiplication result back to F[][] */
void multiply(int F[2][2], int M[2][2]);
/* Helper function that calculates F[][] raise to the power n and puts the
result in F[][]
Note that this function is desinged only for fib() and won't work as general
power function */
void power(int F[2][2], int n);
int fib(int n)
{
int F[2][2] = {{1,1},{1,0}};
if(n == 0)
return 0;
power(F, n-1);
return F[0][0];
}
void multiply(int F[2][2], int M[2][2])
{
int x = F[0][0]*M[0][0] + F[0][1]*M[1][0];
int y = F[0][0]*M[0][1] + F[0][1]*M[1][1];
int z = F[1][0]*M[0][0] + F[1][1]*M[1][0];
int w = F[1][0]*M[0][1] + F[1][1]*M[1][1];
F[0][0] = x;
F[0][1] = y;
F[1][0] = z;
F[1][1] = w;
}
void power(int F[2][2], int n)
{
int i;
int M[2][2] = {{1,1},{1,0}};
// n - 1 times multiply the matrix to {{1,0},{0,1}}
for (i = 2; i <= n; i++)
multiply(F, M);
}
Tempo Complessità: O (n) spazio aggiuntivo: O (1)
Metodo 5 (ottimizzato metodo 4) Il metodo 4 può essere ottimizzato per lavorare nella complessità del tempo O (Logn). Possiamo fare la moltiplicazione ricorsivo per ottenere il potere (M, N) nel metodo prevous (Simile alla ottimizzazione fatto in questo post)
void multiply(int F[2][2], int M[2][2]);
void power(int F[2][2], int n);
/* function that returns nth Fibonacci number */
int fib(int n)
{
int F[2][2] = {{1,1},{1,0}};
if(n == 0)
return 0;
power(F, n-1);
return F[0][0];
}
/* Optimized version of power() in method 4 */
void power(int F[2][2], int n)
{
if(n == 0 || n == 1)
return;
int M[2][2] = {{1,1},{1,0}};
power(F, n/2);
multiply(F, F);
if(n%2 != 0)
multiply(F, M);
}
void multiply(int F[2][2], int M[2][2])
{
int x = F[0][0]*M[0][0] + F[0][1]*M[1][0];
int y = F[0][0]*M[0][1] + F[0][1]*M[1][1];
int z = F[1][0]*M[0][0] + F[1][1]*M[1][0];
int w = F[1][0]*M[0][1] + F[1][1]*M[1][1];
F[0][0] = x;
F[0][1] = y;
F[1][0] = z;
F[1][1] = w;
}
Tempo Complessità: O (log) Spazio Extra: O (log) se consideriamo la funzione stack size delle chiamate, altrimenti O (1).
Driver Program: int main() { int n = 9; printf ("% d", fib (9)); getchar(); return 0; }
Riferimenti: http://en.wikipedia.org/wiki/Fibonacci_number http://www.ics.uci.edu/~eppstein/161/960109.html
Ah .... questo lo ha reso molto più chiaro. Grazie. – KingKongFrog
Hai letto la mia risposta? –
supponiamo che l'indice _i_ sia dato. a) esegui attraverso la serie di fib: 1 1 (non dice quanto lontano o cosa sia, vero?) b) return _i_ – greybeard