Basée sur l'utilisation des fonctions, à la condition de respecter certains principes.
La prog fonctionnelle n'admet pas le changement d'états et la mutation des données (contrairement à la prog impérative).
Les langages fonctionnels sont ceux vouent un culte à ces principes ou sont basés sur eux :
Lisp (1958), Scheme (1975), Common Lisp (1984), Haskell (1987), OCaml (1996), Scala (2003), PureScript (2013)…
Les langages de programmation impératifs acceptant le passage de fonctions en paramètres peuvent être utilisés dans le cadre d'une approche fonctionnelle :
ECMAScript, Java, C#, PHP, Perl, Python, Ruby, Kotlin…
La programmation fonctionnelle s'affranchit de façon radicale des effets secondaires (ou effets de bord) en interdisant les opérations d'affectation (immutabilité).
Le paradigme fonctionnel n'utilise pas de machine à états pour décrire un programme, mais un emboîtement de fonctions qui agissent comme des “boîtes noires” que l'on peut imbriquer les unes dans les autres.
Chaque boîte possédant plusieurs paramètres en entrée mais une seule sortie, elle ne peut sortir qu'une seule valeur possible pour chaque n-uplet de valeurs présentées en entrée. De cette façon, les fonctions n'introduisent pas d'effets de bord.
Exemples d'effets de bord (side effects) :
Tous les exemples ci-dessous sont traités en javascript (ECMAScript).
Si vous ne connaissez pas Javascript : Learn JS in 5 minutes
Une fonction est dite pure à la double condition :
let pi = 3.14 ; const calculateArea = (radius) => radius * radius * pi; calculateArea(10);// returns 314.0, ou autre chose si pi est modifié
let pi = 3.14; const calculateArea = (radius, c) => radius * radius * c; calculateArea(10, pi);// returns 314.0
const PI = 3.14; const calculateArea = (radius) => radius * radius * PI; calculateArea(10);// returns 314.0
Exemples de fonctions impures :
const charactersCounter = (text) => `Character count: ${text.length}`; function analyzeFile(filename) { let fileContent = open(filename); return charactersCounter(fileContent); }
function getRandomArbitrary(min, max) { return Math.random() * (max - min) + min; }
Exemple de fonction impure :
let counter = 1; function increaseCounter(value) { counter = value + 1; } increaseCounter(counter); console.log(counter);// 2
Rendue pure :
let counter = 1; const increaseCounter = (value) => value + 1; increaseCounter(counter);// 2 console.log(counter);// 1
Avantages :
Les données doivent être immutables, c'est-à-dire que leur valeur ne peut changer après initialisation.
let values = [1, 2, 3, 4, 5]; let sumOfValues = 0; for (let i = 0; i < values.length; i++) { sumOfValues += values[i]; } sumOfValues // 15
let list = [1, 2, 3, 4, 5]; let accumulator = 0; function sum(list, accumulator) { if (list.length == 0) { return accumulator; } return sum(list.slice(1), accumulator + list[0]); } sum(list, accumulator); // 15 list; // [1, 2, 3, 4, 5] accumulator; // 0
Referential transparency
La transparence référentielle est obtenue grâce aux fonctions pures, et elle consiste à permettre le remplacement d'un appel d'une fonction pure, par la valeur qu'elle retourne,sans perturber le fonctionnement du programme.
Exemple :
Soit la fonction carré suivante :
const square = (n) => n * n;
Pour la même valeur en entrée, cette fonction pure retournera toujours la même valeur :
square(2);// 4 square(2);// 4 square(2);// 4 // ...
Avec le paramètre 2
, la fonction square
retourne toujours 4
. Il est donc possible deremplacer square(2)
par 4
: Notre fonction est donc référentiellement transparente.
pure functions + immutable data = referential transparency
Grâce à ce concept, il est possible de memoïzer (https://en.wikipedia.org/wiki/Memoization) la fonction, pour accroître les performances, en évitant son appel.
Fonctions objet de première classe ou Functions as first-class entities :
Une fonction est un type de données (callback) et peut-être passée en paramètre d'une fonction ou être retournée par une fonction.
Les fonctions qui en manipulent d'autres sont qualifiées d'ordre supérieur.
Exemple :
Considérons les fonctions suivantes :
const doubleSum = (a, b) => (a + b) * 2; const doubleSubtraction = (a, b) => (a - b) * 2;
Nous pourrions considérer que nous avons 3 fonctions :
const sum = (a, b) => a + b; const subtraction = (a, b) => a - b; const doubleOperator = (f, a, b) => f(a, b) * 2; doubleOperator(sum, 3, 1);// 8 doubleOperator(subtraction, 3, 1);// 4
doubleOperator est une fonction d'ordre supérieur.
Structure permettant de manipuler des langages fonctionnels purs dans des traits impératifs.
Exemples :