Advertisement
fcamuso

Pillole video 7

Apr 2nd, 2025
381
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
C++ 8.12 KB | None | 0 0
  1. #include <iostream>
  2. #include <chrono>
  3. #include <vector>
  4.  
  5.  
  6. using namespace std;
  7.  
  8. struct Nodo {
  9.   string dato = "";
  10.  
  11.   Nodo *next = nullptr;
  12. };
  13.  
  14.  
  15.  
  16. void ins_testa(Nodo *&il, Nodo *nuovo)
  17. {
  18.     nuovo->next = il;
  19.     il = nuovo;
  20. }
  21.  
  22.  
  23.  
  24. void ins_coda(Nodo *&il, Nodo *nuovo)
  25. {
  26.   if (il == nullptr) //lista vuota ?
  27.   {
  28.     nuovo->next = il;
  29.     il = nuovo;
  30.   }
  31.   else
  32.   {
  33.     Nodo *ultimo = il;
  34.  
  35.     //otteniamo un puntatore all'ultimo nodo attuale
  36.     while(ultimo->next != nullptr) ultimo = ultimo->next;
  37.  
  38.     ultimo->next = nuovo;
  39.     nuovo->next = nullptr;
  40.   }
  41. }
  42.  
  43. void stampa_dalla_coda(Nodo *p, int profondita, int &massima)
  44. {
  45.   massima = max(profondita, massima);
  46.  
  47.   if (p!=nullptr)
  48.     stampa_dalla_coda(p->next, profondita+1, massima);
  49. }
  50.  
  51.  
  52. //Equazione di ricorrenza temporale: T(n) = 2*T(n/2) + O(n)
  53. //Equazione di ricorrenza spaziale: T(n) = T(n/2) + O(1)
  54. //Complessità temporale: O(nlog(n))
  55. //Complessità spaziale: O(log(n))
  56. void stampaInvertita(Nodo *testa,int  dim){
  57.  
  58.     //Casi base
  59.     //Lista vuota
  60.     if (testa == nullptr)
  61.         return;
  62.     //Lista con un solo elemento
  63.     if (testa->next == nullptr)
  64.         ;//cout << testa->dato << endl;
  65.         //Passo ricorsivo
  66.     else{
  67.         Nodo* attuale = testa;
  68.         //Suddivido in due metà la lista
  69.         for (int i = 0; i < dim/2 - 1; ++i) {
  70.             attuale = attuale->next;
  71.         }
  72.         //Mi fermo all'ultimo elemento della prima metà
  73.         //ed imposto il puntatore al successivo a nullptr, così da avere due liste fisicamente separate
  74.         Nodo* temp = attuale;
  75.         attuale = attuale->next;
  76.         temp->next = nullptr;
  77.  
  78.         //Ora richiedo la stampa della seconda metà seguita dalla prima; l'ordine delle chiamate è fondamentale
  79.         stampaInvertita(attuale, dim/2);
  80.         stampaInvertita(testa, dim - dim/2);
  81.  
  82.         //Dopo la stampa non resta che ricongiungere le due liste precedentemente divise
  83.         temp->next = attuale;
  84.     }
  85. }
  86.  
  87. //credits Alex-qk5ei (youtube)
  88. int stampaInvertitaMisuraStack(Nodo *testa, int dim){
  89.  
  90.     //Casi base
  91.     //Lista vuota
  92.     if (testa == nullptr)
  93.         return 1;
  94.     //Lista con un solo elemento
  95.     if (testa->next == nullptr) {
  96.         ;//cout << testa->dato << endl;
  97.         return 1;
  98.     }
  99.     //Passo ricorsivo
  100.     else{
  101.         Nodo* attuale = testa;
  102.  
  103.         //Suddivido in due metà la lista
  104.         for (int i = 0; i < (dim>>1) - 1; ++i) {
  105.             attuale = attuale->next;
  106.         }
  107.  
  108.         //Mi fermo all'ultimo elemento della prima metà
  109.         //ed imposto il puntatore al successivo a nullptr, così da avere due liste fisicamente separate
  110.         Nodo* temp = attuale;
  111.         attuale = attuale->next;
  112.         temp->next = nullptr;
  113.  
  114.         /*La profondità massima dello stack P(n) può essere definita ricorsivamente come: P(n) = max(P(n/2), P(n-n/2)) + 1
  115.         L'addendo 1 equivale alla presenza sullo stack del record di attivazione corrente
  116.         Inoltre, considerando che n-n/2 >= n/2 in ogni caso, max(P(n/2), P(n-n/2)) = P(n-n/2) e, conseguentemente
  117.         P(n) = P(n-n/2) + 1
  118.         L'istruzione sottostante rappresenta dunque la profondità massima che andrà restituita al metodo chiamante
  119.         */
  120.         //Ora richiedo la stampa della seconda metà seguita dalla prima; l'ordine delle chiamate è fondamentale
  121.         int profondita = stampaInvertitaMisuraStack(attuale, dim - (dim>>1)) + 1;
  122.         stampaInvertitaMisuraStack(testa, dim>>1);
  123.         //Dopo la stampa non resta che ricongiungere le due liste precedentemente divise
  124.         temp->next = attuale;
  125.  
  126.         return profondita;
  127.     }
  128. }
  129.  
  130. int fattoriale(int n)
  131. {
  132.    if (n == 0)
  133.     return 1;
  134.    else
  135.      return n * fattoriale(n - 1);
  136. }
  137.  
  138.  
  139. int fattoriale_tail(int n, int accumulatore=1)
  140. {
  141.    if (n == 0)
  142.     return accumulatore;
  143.    else
  144.     return fattoriale_tail(n - 1, n * accumulatore);
  145. }
  146.  
  147.  
  148. enum Comando {START, STOP};
  149. auto Cronometro(Comando comando = Comando::START)
  150. {
  151.   static std::chrono::time_point<std::chrono::system_clock> inizio;
  152.   static std::chrono::time_point<std::chrono::system_clock> fine;
  153.  
  154.   if (comando == Comando::START)
  155.   {
  156.     inizio = chrono::high_resolution_clock::now();
  157.     fine = inizio;
  158.   }
  159.   else
  160.     fine = chrono::high_resolution_clock::now();
  161.  
  162.  
  163.   return chrono::duration_cast<std::chrono::milliseconds>(fine - inizio).count();
  164. }
  165.  
  166.  
  167.  
  168.  
  169.  
  170.  
  171.  
  172.  
  173. void stampaInvertitaVector(Nodo *il)
  174. {
  175.   vector<Nodo*> v;
  176.  
  177.   while(il!=nullptr)
  178.   {
  179.     v.push_back(il);
  180.     il = il-> next;
  181.   }
  182.  
  183.   for (int i=v.size()-1; i>=0; i--);
  184.     //cout << v[i] -> dato << " ";
  185. }
  186.  
  187.  
  188.  
  189.  
  190.  
  191.  
  192.  
  193.  
  194. void InvertiLista(Nodo *&il)
  195. {
  196.  
  197.   Nodo * precedente = nullptr;
  198.   Nodo * corrente=il;
  199.   Nodo * successivo = nullptr;
  200.  
  201.   while(corrente!=nullptr)
  202.   {
  203.     successivo = corrente->next;
  204.     corrente->next =  precedente;
  205.  
  206.     precedente = corrente;
  207.     corrente = successivo;
  208.   }
  209.  
  210.   il=precedente;
  211. }
  212.  
  213.  
  214.  
  215.  
  216.  
  217. void stampa(Nodo *p)
  218. {
  219.   while (p!=nullptr)
  220.   {
  221.     //cout << p->dato << " ";
  222.  
  223.     //al prossimo nodo
  224.     p = p->next;
  225.   }
  226. }
  227.  
  228. // Struttura per rappresentare una "mossa"
  229. struct HanoiMove {
  230.     int n;          // Numero di dischi
  231.     char from;      // Piolo di partenza (A, B, C)
  232.     char to;        // Piolo di destinazione
  233.     char aux;       // Piolo ausiliario
  234. };
  235.  
  236. // Versione iterativa senza STL (stack manuale)
  237. void hanoi_iterative_no_stl(int n, char from = 'A', char to = 'C', char aux = 'B') {
  238.     const int MAX_STACK_SIZE = 1000; // Dimensione massima dello stack
  239.     HanoiMove stack[MAX_STACK_SIZE];
  240.     int top = -1; // Inizializza lo stack vuoto
  241.  
  242.     // Simula la prima chiamata ricorsiva
  243.     stack[++top] = {n, from, to, aux};
  244.  
  245.     while (top >= 0) { // Finché lo stack non è vuoto
  246.         HanoiMove current = stack[top--]; // Pop
  247.  
  248.         if (current.n == 1) {
  249.             // Caso base: sposta il disco
  250.             cout << "Muovi disco 1 da " << current.from << " a " << current.to << endl;
  251.         } else {
  252.             // Simula le chiamate ricorsive in ordine inverso (LIFO)
  253.             // 1. hanoi(n-1, aux, to, from)
  254.             stack[++top] = {current.n - 1, current.aux, current.to, current.from};
  255.             // 2. Sposta il disco n da from a to (caso base fittizio)
  256.             stack[++top] = {1, current.from, current.to, current.aux};
  257.             // 3. hanoi(n-1, from, aux, to)
  258.             stack[++top] = {current.n - 1, current.from, current.aux, current.to};
  259.         }
  260.     }
  261. }
  262.  
  263.  
  264. void hanoi(int n, char from = 'A', char to = 'C', char aux = 'B') {
  265.     if (n == 1) {
  266.         cout << "Muovi disco 1 da " << from << " a " << to << endl;
  267.         return;
  268.     }
  269.     hanoi(n - 1, from, aux, to);  // Sposta n-1 dischi da 'from' a 'aux'
  270.     cout << "Muovi disco " << n << " da " << from << " a " << to << endl;
  271.     hanoi(n - 1, aux, to, from);  // Sposta n-1 dischi da 'aux' a 'to'
  272. }
  273.  
  274. int main()
  275. {
  276.     //cout << fattoriale_tail(60001) << endl;
  277.     //hanoi_iterative_no_stl(3);
  278.  
  279.     string dati_prova[] = {"abaco", "amo", "bicicletta", "cane", "canguro", "mare", "sale", "zebra"};
  280.  
  281.     Nodo *il = nullptr;
  282.  
  283.     int quante_run=10000;
  284.     int num_ele = 20000;
  285.  
  286.     for (int i=0; i<num_ele; i++)
  287.       ins_testa(il, new Nodo{dati_prova[i%8]});
  288.  
  289.     //stampaInvertitaVector(il);
  290.  
  291. //    stampa(il);
  292. //    InvertiLista(il);
  293. //    stampa(il);
  294. //    InvertiLista(il);
  295. //    stampa(il);
  296.  
  297.  
  298.  
  299.     //CON RICORSIONE E DIVIDE ET IMPERA
  300.     Cronometro(Comando::START);
  301.     for (int run=0; run<quante_run; run++)
  302.           stampaInvertita(il,num_ele);
  303.     cout << "Ricorsione con divide/impera: " << Cronometro(Comando::STOP) << endl;
  304.  
  305.     //CON VECTOR DI APPOGGIO
  306.     Cronometro(Comando::START);
  307.     for (int run=0; run<quante_run; run++)
  308.           stampaInvertitaVector(il);
  309.     cout << "Iterativa con vector di puntatori: " << Cronometro(Comando::STOP) << endl;
  310.  
  311.  
  312.     //CON INVERSIONE PUNTATORI
  313.     Cronometro(Comando::START);
  314.     for (int run=0; run<quante_run; run++)
  315.     {
  316.       InvertiLista(il);
  317.       stampa(il);
  318.       InvertiLista(il);
  319.     }
  320.     cout << "Iterativa con inversione fisica lista: " << Cronometro(Comando::STOP) << endl;
  321.  
  322.  
  323.  
  324.     return 0;
  325. }
  326.  
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement