Aller au contenu
Wikipédia l'encyclopédie libre

Curryfication

Un article de Wikipédia, l'encyclopédie libre.

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

[modifier | modifier le code ]

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 :

 curry (add (x,y)) → addcur x → lambda (y → x + y)

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.

Exemples dans les langages de programmation

[modifier | modifier le code ]

En Haskell, voici une fonction non-curryfiée :

uncurried_add(x,y)=x+y

et la même après curryfication :

curried_addxy=x+y

ou bien :

curried_addx=\y->x+y

où la barre oblique \ introduit une λ-expression (lambda-expression) et sert à définir des fonctions anonymes. Et voici une application partielle :

main=do
print(uncurried_add(3,5))-- 8
print((curried_add3)5)-- 8
print(curried_add35)-- 8
letadd_three=curried_add3
print(add_three5)-- 8
print(add_three12)-- 15
Remarques

La curryfication peut se faire à la main ou bien par un programme. Pour les fonctions à deux arguments, voici ces programmes en Haskell :

main=do
letuncurried=uncurrycurried_add
print(uncurried(3,5))-- 8

letcurried=curryuncurried_add
print((curried3)5)-- 8
print(curried35)-- 8
letadd_three=curried3
print(add_three5)-- 8
print(add_three12)-- 15

Même fonction en Python :

defuncurried_add(x, y):
 return x + y
defcurried_add(x):
 return lambda y: x + y
print(uncurried_add(3, 5)) # 8
print(curried_add(3)(5)) # 8
add_three = curried_add(3)
print(add_three(5)) # 8
print(add_three(12)) # 15

Même fonction en Caml :

let uncurried_add(x, y) = x + y;; (* Type de la fonction : (entier * entier) -> entier *)
let curried_add x y = x + y;; (* Type de la fonction : entier -> entier -> entier *)
uncurried_add(3,4);; (* Retourne 7. Type de la valeur retournée : entier. *)
curried_add 3;; (* Retourne la fonction qui ajoute 3. Type de la valeur retournée : entier -> entier. *)
(curried_add 3) 4;; (* Crée la fonction qui ajoute 3 et l'applique à l'entier 4. Retourne 7. Type de la valeur retournée : entier. *)
let add_three = curried_add 3;; (* Définit la fonction add_three comme la fonction qui ajoute 3 à son argument. *)
add_three 5;; (* Applique add_three à 5. Retourne 8. *)
add_three 12;; (* Applique add_three à 12. Retourne 15. *)
defuncurried_add(x,y)
returnx+y
end
defcurried_add(x)
returnlambda{|y|x+y}
end
putsuncurried_add(3,5)# 8
putscurried_add(3).call(5)# 8
add_three=curried_add(3)
putsadd_three.call(5)# 8
putsadd_three.call(12)# 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.

uncurried_add=lambda{|x,y|x+y}
putsuncurried_add.call(3,5)# 8
putsuncurried_add.curry[3][5]# 8
add_three=uncurried_add.curry[3]
putsadd_three.call(5)# 8
putsadd_three.call(12)# 15
(defineuncurried-add(lambda(xy)
(+xy)))
(definecurried-add(lambda(x)
(lambda(y)
(+xy))))
(display(uncurried-add35))(newline); 8
(display((curried-add3)5))(newline); 8
(defineadd-three(curried-add3))
(display(add-three5))(newline); 8
(display(add-three12))(newline); 15
defuncurried_add(x:Int,y:Int)=x+y// => Int
// Après currying
defcurried_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 partielles
defadd_three0=curried_add0(3)// retourne une fonction de type: (Int) => Int
curried_add0(3)(5)// Retourne 8
add_three0(5)// Retourne 8
defadd_three1=curried_add1(3)// retourne une fonction de type: (Int) => Int
curried_add1(3)(5)// Retourne 8
add_three1(5)// Retourne 8
defadd_three2=curried_add2(3)// retourne une fonction de type: (Int) => Int
curried_add2(3)(5)// Retourne 8
add_three2(5)// Retourne 8
procuncurried_add{xy}{
expr{$x+$y}
}
proccurried_addx{
listapply[listy[format{expr{%d+$y}}$x]]
}
putsstdout[uncurried_add35];# 8
putsstdout[{*}[curried_add3]5];# 8
setadd_three[curried_add3]
putsstdout[{*}$add_three5];# 8
putsstdout[{*}$add_three12];# 15

Un JAPH qui est un exemple de curryfication en Perl 6 :

subjaph(Str $lang) {say"just another $lang hacker";}
my&perl6Japh:=&japh.assuming("Perl6");
perl6Japh();
functionuncurried_add(x,y){
returnx+y;
}
functioncurried_add(x){
returnfunction(y){
returnx+y;
};
}
console.log(uncurried_add(3,5));// 8
console.log(curried_add(3)(5));// 8
constadd_three=curried_add(3);
console.log(add_three(5));// 8
console.log(add_three(12));// 15
constuncurriedAdd=(x,y)=>x+y;
constcurriedAdd=x=>y=>x+y;
uncurriedAdd(2,5);// 7
curriedAdd(2)(5);// 7
(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++

[modifier | modifier le code ]

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 :

#include<iostream>
intuncurried_add(intx,inty){
returnx+y;
}
autocurried_add(intx){
return[x](inty){
returnx+y;
};
}
intmain(){
std::cout<<uncurried_add(3,5)<<std::endl;// 8
std::cout<<curried_add(3)(5)<<std::endl;// 8
autoadd_three=curried_add(3);
std::cout<<add_three(5)<<std::endl;// 8
std::cout<<add_three(12)<<std::endl;// 15
}

À 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).

Notes et références

[modifier | modifier le code ]
  1. (en) John C. Reynolds, « Definitional Interpreters for Higher-Order Programming Languages », Higher-Order and Symbolic Computation , vol. 11, no 4,‎ , p. 374 (DOI 10.1023/A:1010027404223 ) :

    « 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.) »

Sur les autres projets Wikimedia :

Articles connexes

[modifier | modifier le code ]
v · m
Création
Structure
Comportement
Fonctionnel
Patron d'architecture
Autres patrons


AltStyle によって変換されたページ (->オリジナル) /