Condividi tramite


Differenze tra programmazione funzionale e programmazione imperativa

In questo argomento vengono presentate le differenze tra la programmazione funzionale e la più tradizionale programmazione imperativa (procedurale).

Differenze tra programmazione funzionale eprogrammazione imperativa

Il paradigma della programmazione funzionale è stato creato espressamente per supportare un approccio funzionale puro alla soluzione dei problemi. La programmazione funzionale è un tipo di programmazione dichiarativa. Al contrario, la maggior parte dei linguaggi più diffusi, compresi i linguaggi di programmazione orientata a oggetti quali C#, Visual Basic, C++ e Java, è progettata principalmente per supportare la programmazione imperativa (procedurale).

Con un approccio imperativo lo sviluppatore scrive codice in cui vengono descritti in dettaglio i passaggi esatti che devono essere eseguiti dal computer per raggiungere l'obiettivo. Questo tipo di programmazione viene a volte definito algoritmico. Al contrario, un approccio funzionale implica la composizione del problema come set di funzioni da eseguire. È necessario definire con attenzione l'input e l'output di ogni funzione. Nella tabella seguente sono descritte alcune delle differenze generali tra questi due approcci.

Caratteristica

Approccio imperativo

Approccio funzionale

Obiettivo del programmatore

Come eseguire attività (algoritmi) e come tenere traccia delle modifiche di stato.

Quali informazioni si desiderano e quali trasformazioni sono richieste.

Modifiche di stato

Importante.

Non esistente.

Ordine di esecuzione

Importante.

Importanza limitata.

Controllo del flusso primario

Cicli, condizionali e chiamate di funzioni (metodi).

Chiamate di funzioni, inclusa la ricorsione.

Unità di modifica primaria

Istanze di strutture o classi.

Funzioni come oggetti e raccolte dati di prima classe.

Anche se la maggior parte dei linguaggi è progettata per supportare un paradigma di programmazione specifico, i linguaggi generici sono in genere sufficientemente flessibili da supportare più paradigmi. Ad esempio, è possibile usare la maggior parte dei linguaggi che contengono puntatori a funzioni per supportare efficacemente la programmazione funzionale. Inoltre, in C# 3.0 e Visual Basic 9.0 sono state aggiunte estensioni del linguaggio esplicite per supportare la programmazione funzionale, tra cui espressioni lambda e inferenza dei tipi. La tecnologia LINQ è un tipo di programmazione funzionale e dichiarativa.

Programmazione funzionale tramite XSLT

Gli sviluppatori XSLT hanno in genere una certa familiarità con l'approccio funzionale. Il modo più efficace per sviluppare un foglio di stile XSLT consiste nel considerare ogni modello come una trasformazione isolata e componibile. All'ordine di esecuzione non viene data alcuna importanza. XSLT non consente effetti collaterali, con l'eccezione che i meccanismi di escape per l'esecuzione di codice procedurale può introdurre effetti collaterali che pregiudicano la purezza funzionale. Tuttavia, pur essendo uno strumento efficace, alcune caratteristiche di XSLT non sono ottimali. Ad esempio, l'espressione di costrutti di programmazione in XML rende il codice relativamente dettagliato e pertanto difficile da mantenere. Inoltre, l'affidamento eccessivo alla ricorsione per il controllo del flusso può generare codice difficile da leggere. Per altre informazioni su XSLT, vedere Trasformazioni XSLT.

Tuttavia, XSLT ha dimostrato il vantaggio dell'utilizzo di un approccio funzionale puro per trasformare XML da una forma a un'altra. La programmazione funzionale con LINQ to XML è simile a XSLT per molti aspetti. Tuttavia, i costrutti di programmazione introdotti da LINQ t XML e C# 3.0 e Visual Basic 9.0 consentono di scrivere trasformazioni funzionali pure che risultano più facili da leggere e mantenere rispetto a XSLT.

Vantaggi delle funzioni pure

Il motivo principale per implementare le trasformazioni funzionali come funzioni pure è che le funzioni pure sono componibili, ossia sono autonome e senza stato. Queste caratteristiche offrono numerosi vantaggi, tra cui quelli riportati di seguito:

  • Maggiore facilità di lettura e manutenibilità, in quanto ogni funzione è progettata per realizzare un'attività specifica con gli argomenti assegnati. La funzione non si basa su alcuno stato esterno.

  • Sviluppo iterativo più semplice, in quanto è più agevole eseguire il refactoring del codice ed è più facile implementare le modifiche di progettazione. Si supponga ad esempio di scrivere una trasformazione complicata e poi di realizzare che una parte di codice viene ripetuta diverse volte al suo interno. Se si esegue il refactoring tramite un metodo puro, è possibile chiamare il metodo puro quando è necessario senza preoccuparsi degli effetti collaterali.

  • Procedure più semplici di test e debug, in quanto le funzioni pure possono essere più facilmente testate in isolamento ed è possibile scrivere codice di test che chiama la funzione pura con valori tipici, casi limite validi e casi limite non validi.

Transizione per gli sviluppatori OOP

Nella tradizionale programmazione orientata a oggetti (OOP, Object-Oriented Programming), gli sviluppatori sono in genere abituati a programmare nello stile imperativo/procedurale. Per passare allo sviluppo in uno stile funzionale puro, devono modificare il loro modo di ragionare e l'approccio adottato per lo sviluppo.

Per risolvere i problemi, gli sviluppatori progettano gerarchie di classi, si dedicano al corretto incapsulamento e ragionano in termini di contratti tra classi. Il comportamento e lo stato dei tipi di oggetto sono di fondamentale importanza e per rispondere a tale esigenza vengono fornite funzionalità del linguaggio quali classi, interfacce, ereditarietà e polimorfismo.

Al contrario, l'approccio della programmazione funzionale ai problemi si traduce nella valutazione di trasformazioni funzionali pure di raccolte dati. Nella programmazione funzionale vengono evitati i dati di stato e modificabili e viene data invece una maggiore enfasi all'applicazione di funzioni.

Fortunatamente, in C# e Visual Basic non è necessario passare completamente alla programmazione funzionale, perché sono supportati sia l'approccio imperativo che quello funzionale. Gli sviluppatori possono scegliere l'approccio più appropriato per un determinato scenario. In effetti, nei programmi vengono spesso combinati entrambi gli approcci.

Vedere anche

Concetti

Introduzione alle trasformazioni funzionali pure

Refactoring in funzioni pure

Altre risorse

Trasformazioni XSLT