En informatique, plus précisément en programmation fonctionnelle, la curryfication est la transformation d'une fonction à plusieurs arguments en une fonction à un argument qui retourne une fonction sur le reste des arguments. L'opération inverse est possible et s'appelle la décurryfication.
Le terme vient du nom du mathématicien américain Haskell Curry, bien que cette opération ait été introduite pour la première fois par Moses Schönfinkel[1].
Définition
Considérons une fonction add qui prend deux arguments (x et y) et en renvoie la somme. En la curryfiant, on obtient une fonction addcur qui prend un argument et renvoie une fonction qui prend le deuxième argument. En pseudo-langage :
La curryfication permet l'application partielle : si on appelle la fonction curryfiée avec l'argument 3, on récupère une fonction qui ajoute 3 à son argument.
letuncurried_add(x,y)=x+y;;(* Type de la fonction : (entier * entier) -> entier *)letcurried_addxy=x+y;;(* Type de la fonction : entier -> entier -> entier *)uncurried_add(3,4);;(* Retourne 7. Type de la valeur retournée : entier. *)curried_add3;;(* Retourne la fonction qui ajoute 3. Type de la valeur retournée : entier -> entier. *)(curried_add3)4;;(* Crée la fonction qui ajoute 3 et l'applique à l'entier 4. Retourne 7. Type de la valeur retournée : entier. *)letadd_three=curried_add3;;(* Définit la fonction add_three comme la fonction qui ajoute 3 à son argument. *)add_three5;;(* Applique add_three à 5. Retourne 8. *)add_three12;;(* Applique add_three à 12. Retourne 15. *)
Ruby fournit l'expression lambda avec la méthode curry. La méthode prend en argument un argument optionnel : l'arity, qui permet des processus plus fins, qui ne seront pas discutés ici.
defuncurried_add(x:Int,y:Int)=x+y// => Int// Après curryingdefcurried_add0(x:Int)(y:Int)=x+y// => Int// Ce qui équivaut (si on décompose) àdefcurried_add1(x:Int)=(y:Int)=>x+y// => (Int) => Int // Ou encore à:defcurried_add2=(x:Int)=>(y:Int)=>x+y// => (Int) => (Int) => Int // applications partiellesdefadd_three0=curried_add0(3)// retourne une fonction de type: (Int) => Intcurried_add0(3)(5)// Retourne 8add_three0(5)// Retourne 8defadd_three1=curried_add1(3)// retourne une fonction de type: (Int) => Intcurried_add1(3)(5)// Retourne 8add_three1(5)// Retourne 8defadd_three2=curried_add2(3)// retourne une fonction de type: (Int) => Intcurried_add2(3)(5)// Retourne 8add_three2(5)// Retourne 8
(defn uncurried-add[xy](+ xy))(defn curried-add[x](fn [y](+ xy)));; curried-add renvoie une fonction anonyme(println (uncurried-add35));; 8(println ((curried-add3)5));; 8(def add-three(curried-add3))(println (add-three5));; 8(println (add-three12));; 15
Modern C++
Le C++11 a introduit les lambdas dans le langage, puis le C++14 a introduit la possibilité de définir un type de retour auto pour faciliter l'écriture de certaines fonctions.
Ces deux améliorations permettent d'implémenter la curryfication :
À noter que depuis le C++17, la mention du type int en argument de l'expression lambda peut également être remplacée par auto, généralisant l'implémentation (au prix de faire de la méthode curried_add un template implicite).
« In the last line we have used a trick called Currying (after the logician H. Curry) to solve the problem of introducing a binary operation into a language where all functions must accept a single argument. (The referee comments that although "Currying" is tastier, "Schönfinkeling” might be more accurate.) »