Una procedura o funzione può contenere al suo interno una chiamata a un'altra procedura. Compreso, la subroutine può chiamare se stessa. In questo caso, al computer non importa. Lui, come sempre, esegue costantemente i comandi che ha incontrato dall'alto verso il basso.

Se ricordi la matematica, allora lì puoi incontrare il principio dell'induzione matematica. È come segue: qualche affermazione è vera per ogni naturale n if
    1) è valido per n = 1;
    2) dalla validità dell'affermazione per ogni naturale arbitrario n = k ne consegue che è vera per n = k+1.

In programmazione, questa tecnica è chiamata ricorsione.

La ricorsione è un modo per definire un insieme di oggetti in termini dell'insieme stesso, basato su dati casi base semplici.

Ricorsivo è una procedura (funzione) che richiama se stessa direttamente o tramite altre procedure e funzioni.
Esempio di procedura ricorsiva:

void Rec(int a)
{
  if (a>0) { Rec(a-1); }
  Console.WriteLine(a);
}

Schematicamente, il lavoro di ricorsione può essere rappresentato come un diagramma di flusso.

 
Rec() la procedura viene eseguita con parametro 3 Poi, all'interno della procedura con parametro 3, viene chiamata la procedura con parametro 2, e così via, fino a quando viene chiamata la procedura con parametro 0. work. Poi il controllo viene ritrasferito alla procedura con il parametro 1, anch'essa termina il suo lavoro stampando il numero 1, e così via. prima della procedura con parametro 3. 

Tutte le procedure chiamate vengono archiviate in memoria finché non completano il loro lavoro. Il numero di procedure concorrenti è chiamato profondità di ricorsione.

Abbiamo scoperto che la ricorsione è l'esecuzione ripetuta di comandi contenuti in una subroutine. E questo, a sua volta, è simile al lavoro del ciclo. Esistono linguaggi di programmazione in cui il costrutto loop è del tutto assente, ad esempio Prolog. 
Proviamo a simulare il lavoro del ciclo for
Il ciclo for contiene una variabile contapassi. In una subroutine ricorsiva, tale variabile può essere passata come parametro.
// proceduraLoopImitation() con due parametri
// primo parametro – contapassi, secondo parametro – numero totale di passi
static void LoopImitation(int i, int n)
{
  Console.WriteLine("Ciao N" + i); // istruzione da ripetere per qualsiasi valore i 
  if (i < n) // finché il contatore del ciclo non è uguale a n,
  {
    LoopImitation(i+1, n); // chiamando un nuovo procedura di istanza, con parametro i+1 (vai al valore i successivo)
  }
} 

Ricorsione e iterazione
Per capire la ricorsione, devi capire la ricorsione...
 
Iterazione nella programmazione: una fasedi un processo ciclico di elaborazione dei dati. 
Spesso gli algoritmi iterativi nella fase corrente (iterazione) utilizzano il risultato della stessa operazione o azione calcolata nelle fasi precedenti.  Un esempio di tali calcoli è il calcolo delle relazioni di ricorrenza. 
Un semplice esempio di valore ricorsivo è il fattoriale: \(N!=1 \cdot 2 \cdot 3 \cdot \ ... \ \cdot N\).
Il calcolo del valore ad ogni passo (iterazione) è \(N=N \cdot i\) .  Quando calcoliamo il valore di \(N\), prendiamo il valore già memorizzato \(N\).

Il fattoriale di un numero può anche essere descritto usando la formula ricorrente:
\(\begin{equation*} n!= \begin{cases} 1 &\text{n <= 1,}\\ (n-1)! \cdot n &\text{n > 1.} \end{casi} \end{equazione*}\)

Potresti notare che questa descrizione non è altro che una funzione ricorsiva.
Qui la prima riga (\(n <= 1\)) è il caso base (condizione di terminazione della ricorsione) e la seconda riga è la transizione al passo successivo.  
 
Funzione fattoriale ricorsiva Algoritmo iterativo
static int Factorial(int n){
se (n > 1)
return n * Fattoriale(n - 1);
altrimenti restituisce 1;
}
int x = 1;
for (int i = 2; i <= n; i++)
  x = x * io;
Console.WriteLine("%d", x);

Dovrebbe essere chiaro che le chiamate di funzione comportano un sovraccarico aggiuntivo, quindi un calcolo fattoriale non ricorsivo sarà leggermente più veloce. 

Conclusione:
dove puoi scrivere un programma con un semplice algoritmo iterativo, senza ricorsione, allora devi scrivere senza ricorsione. Tuttavia, esiste un'ampia classe di problemi in cui il processo computazionale è implementato solo mediante ricorsione.
D'altra parte, gli algoritmi ricorsivi sono spesso più comprensibili.

Traduzione ricorsiva di un numero da un sistema numerico a un altro

In in alcune situazioni nelle procedure, puoi usare la parola return  senza un argomento - cioè, in effetti, la procedura continua a non restituire nulla. Questo può essere utile durante la ricorsione, quando  ;return  viene utilizzato per terminare la discesa nei casi base dei valori dei parametri ricorsivi. Ad esempio, una procedura che converte un numero da decimale a binario potrebbe essere simile a questa: static void stampaDue(int n) {     se (n == 0) ritorno;   stampaDue(n / 2);   if (n % 2 == 0) Console.Write(0);   else Console.Write(1); }

Attività
Nell'alfabeto della lingua della tribù "Tumba-Yumba"; quattro lettere: "K", "L", "M" e "N". Devi visualizzare tutte le parole costituite da n lettere che possono essere costruite dalle lettere di questo alfabeto.

Il problema è un normale problema di forza bruta che può essere ridotto a un problema minore.
Sostituiremo in sequenza le lettere per la parola.
La prima posizione di una parola può essere una delle 4 lettere dell'alfabeto (K. L, M, N).
Mettiamo prima la lettera K. Quindi, per ottenere tutte le varianti con la prima lettera K, devi enumerare tutte le possibili combinazioni di lettere nelle rimanenti posizioni n - 1 e così via. (vedi foto).
Quindi, il problema si riduce a risolvere quattro problemi di lunghezza n - 1.
 
Itera su n caratteri in modo ricorsivo
w[0]='K'; // itera sugli ultimi caratteri L-1 w[0]='L'; // itera sugli ultimi caratteri L-1 w[0]='M'; // itera sugli ultimi caratteri L-1 w[0]='N'; // itera sugli ultimi caratteri L-1 w - una stringa di caratteri che memorizza la parola di lavoro.
Così, abbiamo ottenuto la ricorsività. Possiamo organizzare la soluzione del problema sotto forma di una procedura ricorsiva. 
Resta da determinare quando finirà la ricorsione? Quando tutti i caratteri sono impostati, cioè, il numero di caratteri impostati è n. In questo caso è necessario visualizzare sullo schermo la parola risultante ed uscire dalla procedura.

Il programma C# avrà questo aspetto.
// w - parametro modificabile (risultato stringa) // Alla procedura TumbaWords viene passato l'alfabeto come stringa di caratteri, // la parola parola e il numero di caratteri già impostati (all'inizio – 0) static void TumbaWords( string A, ref string w, int N ) { if (N == w.Length) // w.Length - il numero di caratteri nella stringa {   // se tutti i caratteri sono già stati impostati sulla parola,       // quindi è necessario emettere una stringa e terminare la procedura Console.WriteLine(w); ritorno; } for ( int i = 0; i < w.Length; i ++ ) // se la condizione precedente è falsa (ovvero, non tutti i caratteri sono spaziati, {     // se la condizione di cui sopra è falsa (ovvero, non tutti i caratteri sono spaziati,     // quindi nel ciclo passiamo attraverso tutti i caratteri dell'alfabeto e   // metti alternativamente il carattere nel primo spazio libero w += LA[i]; TumbaWords(A, ref w, N+1); w = w.Rimuovi(w.Lunghezza - 1); // e poi rimuovi l'ultimo carattere aggiunto,   // per creare una nuova parola con lo stesso prefisso } } vuoto statico Main() { int n = Convert.ToInt32(Console.ReadLine()); parola stringa = ""; TumbaWords("KLMN", ref word, 0); }
NOTA che w è un parametro mutabile (stringa di risultato)!