Warning: "continue" targeting switch is equivalent to "break". Did you mean to use "continue 2"? in /membri/cosimofanelli/wp-content/plugins/qtranslate-x/qtranslate_frontend.php on line 497

Warning: Parameter 2 to qtranxf_postsFilter() expected to be a reference, value given in /membri/cosimofanelli/wp-includes/class-wp-hook.php on line 308
C - ottimizzazioni e piccoli accorgimenti - CosimoF

C – ottimizzazioni e piccoli accorgimenti

My project's page

C – ottimizzazioni e piccoli accorgimenti

10 marzo 2017 Programmazione 0

Salve a tutti, ultimamente, lavorando ad un programma in C, mi è capitato di avere a che fare con qualche piccolo accorgimento riguardante l’ottimizzazione. Ho trovato poche informazioni in italiano quindi ho studiato dalle documentazioni di GCC e da vari articoli che parlano degli “internals” del linguaggio, decidendo di condividere questi, anche se piccoli, accorgimenti con voi: potrebbero tornarvi utili 😀

Cercherò di essere il più chiaro possibile ed evitare fraintendimenti. Per comprendere questo articolo sono richieste delle conoscenze base relative ad array e puntatori.

Bene, dopo questa noiosa introduzione passiamo al dunque.
Nell’ottimizzazione di un programma, una delle prime cose da considerare è l’uso di array e indici. Per accedere ad un elemento di un array usando gli indici il compilatore dovrà eseguire una serie di calcoli, soprattutto moltiplicazioni. Il mio consiglio se si vuole ottimizzare un programma che fa largo uso di array è quello di vedere queste strutture per quello che realmente sono: un puntatore al primo elemento al quale applicare l’aritmetica dei puntatori.

A primo impatto questo tipo di approccio potrebbe sembrare ostico, ma è abbastanza semplice, basta solo ragionarci un po’ e una volta entrati nell’ottica diventa come contare normalmente. Usando l’aritmetica dei puntatori, l’incremento di un puntatore si riduce ad una banale somma che richiede molte meno risorse computazionali.
Snippets come quelli dell’esempio sottostante sarebbe meglio evitarli se il nostro obbiettivo è fare un programma molto più veloce ed ottimizzato.

void somma_elementi(int* primo, int* secondo, int* somma, int elementi)
{
 for (int i = 0; i < elementi; i++)
     somma[i] = primo[i] + secondo[i];
}

Un’alternativa del tutto equivalente a questo codice (anche se migliorabile) potrebbe essere questa:

void somma_elementi(int* primo, int* secondo, int* somma, int elementi)
{
 for (int i = 0; i < elementi; i++)
     *(somma + i) = *(primo + i) + *(secondo + i);
}

Possiamo notare come questo codice potrebbe essere riscritto in maniera leggermente differente e più chiara utilizzando l’operatore di post-incremento ottenendo uno snippet di questo tipo:

void somma_elementi(int* primo, int* secondo, int* somma, int elementi)
{
 for (int i = 0; i < elementi; i++)
     *(somma++) = *(primo++) + *(secondo++);
}

Riflettendoci un attimo possiamo notare che le operazioni di post-incremento sono leggermente più lente di quelle di pre-incremento: questo perchè il compilatore è obbligato a conservare una copia del vecchio valore, restituirlo non ancora incrementato, eseguire il codice ed effettuare l’incremento. Se si tratta di poche iterazioni tutto questo non ha un peso rilevante, ma vi posso assicurare che su un numero eleveto di iterazioni la differenza in termini di prestazioni e complessità computazionale si nota. Per questo sono arrivato ad una soluzione che usi solo l’aritmetica dei puntatori e il pre-incremento.
Questo dovrebbe essere il risultato finale:

void somma_elementi(int* primo, int* secondo, int* somma, int elementi)
{
 --primo;
 --secondo;
 --somma;
 for (int i = 0; i < elementi; i++)
     *(++somma) = *(++primo) + *(++secondo);
}

Nello snippet qui sopra ho decrementato i puntatori per poi incrementarli nel ciclo for senza perdere informazioni. Arrivato a questo punto ero contento del mio programma perchè ero riuscito a ottimizzarlo e a compattare il codice. Però non è finita qui: c’è un grave errore nel mio programma, riuscite a notarlo da soli?

Ok, ve lo dico nel caso non lo aveste notato. Parlando con un amico e rileggendo un attimo il manuale dell’ANSI C, nel paragrafo 6 (circa array e puntatori) ho notato che quello che avevo fatto non andava per niente bene: nonostante questo programma funzioni sulla quasi totalità delle CPU è presente un “undefined behavior”, un comportamento indefinito. Decrementare il puntatore al primo elemento di un array non fa altro che farmi uscire dalla memoria dedicata all’array. Su alcuni compilatori o CPU questo non fa altro che bloccare il programma e lanciare una eccezione che potrebbe essere quella di Out of Memory o quella di Segmentation Fault. Per questo mi sono subito precipitato a correggere questo obbrobrio: avere codice non sicuro mi fa stare male. In questo caso la correzione era piuttosto banale e scontata, ve la mostro:

void somma_elementi_ottimizzata(int* primo, int* secondo, int* somma, int elementi)
{
 for (int i = 0; i < elementi; i++)
 {
     *somma = *primo + *secondo;
     ++primo;
     ++secondo;
     ++somma;
 }
}

Perfetto, adesso ho l’anima in pace e il programma funziona correttamente senza un codice “unsafe”. Arrivati a questo punto possiamo provare ad analizzare brevemente questo pezzo di codice e relazionarlo ai precedenti. Come possiamo vedere, rispetto ai primi snippets non uso ne indici ne operatori di post-incremento, uso semplicemente l’aritmetica dei puntatori e gli operatori di pre-incremento.

Facendo invece un confronto con lo snippet precedente notiamo chiaramente come evitando il decremento all’inizio possiamo ridurre i cicli di clock necessari risparmiando tempo e soprattutto senza riscontrare problemi di “undefined behavior”.

Spero di essere stato abbastanza chiaro, per qualsiasi dubbio o segnalazione di errori scrivete pure nei commenti. A breve pubblicherò altre guide, tra cui una riguardante l’aritmetica dei puntatori.

Lascia un commento

Il tuo indirizzo email non sarà pubblicato. I campi obbligatori sono contrassegnati *