Refactorisation dans des fonctions pures
L'un des aspects importants des transformations fonctionnelles pures consiste à apprendre à refactoriser le code à l'aide de fonctions pures.
Notes
La nomenclature courante dans la programmation fonctionnelle consiste à refactoriser les programmes à l'aide de fonctions pures.En Visual Basic et C++, cela correspond à l'utilisation des fonctions dans les langages respectifs.Toutefois, en C#, les fonctions portent le nom de méthodes.Pour les besoins de cette discussion, une fonction pure est implémentée en tant que méthode en C#.
Comme mentionné précédemment dans cette section, une fonction pure présente deux caractéristiques utiles :
Elle n'a aucun effet secondaire. La fonction ne change aucune variable et aucune donnée de tout type en dehors de la fonction.
Elle est cohérente. Étant donné le même ensemble de données d'entrée, elle retournera toujours la même valeur de sortie.
L'une des manières de basculer vers la programmation fonctionnelle consiste à refactoriser le code existant afin d'éliminer les effets secondaires indésirables et les dépendances externes. De cette manière, vous pouvez créer des versions avec fonctions pures du code existant.
Cette rubrique explique ce qu'est et ce que n'est pas une fonction pure. Le didacticiel Manipulation d'informations dans un document WordprocessingML montre comment manipuler un document WordprocessingML et contient deux exemples illustrant comment effectuer une refactorisation à l'aide d'une fonction pure.
Les exemples suivants comparent deux fonctions non pures et une fonction pure.
Dans le code suivant, la fonction HypenatedConcat n'est pas pure car elle modifie le membre de données aMember dans la classe :
public class Program
{
private static string aMember = "StringOne";
public static void HypenatedConcat(string appendStr)
{
aMember += '-' + appendStr;
}
public static void Main()
{
HypenatedConcat("StringTwo");
Console.WriteLine(aMember);
}
}
Module Module1
Dim aMember As String = "StringOne"
Public Sub HypenatedConcat(ByVal appendStr As String)
aMember = aMember & "-" & appendStr
End Sub
Sub Main()
HypenatedConcat("StringTwo")
Console.WriteLine(aMember)
End Sub
End Module
Ce code génère la sortie suivante :
StringOne-StringTwo
Notez qu'il importe peu que les données modifiées aient un accès public ou private, ou qu'il s'agisse d'un membre static (shared) ou d'un membre d'instance. Une fonction pure ne change aucune donnée en dehors de la fonction.
En outre, la version suivante de cette même fonction n'est pas pure car elle modifie le contenu de son paramètre, sb.
public class Program
{
public static void HypenatedConcat(StringBuilder sb, String appendStr)
{
sb.Append('-' + appendStr);
}
public static void Main()
{
StringBuilder sb1 = new StringBuilder("StringOne");
HypenatedConcat(sb1, "StringTwo");
Console.WriteLine(sb1);
}
}
Module Module1
Public Sub HypenatedConcat(ByVal sb As StringBuilder, ByVal appendStr As String)
sb.Append("-" & appendStr)
End Sub
Sub Main()
Dim sb1 As StringBuilder = New StringBuilder("StringOne")
HypenatedConcat(sb1, "StringTwo")
Console.WriteLine(sb1)
End Sub
End Module
Cette version du programme génère la même sortie que la première version, car la fonction HypenatedConcat a modifié la valeur (l'état) de son premier paramètre en appelant la fonction membre Append. Notez que cette altération a lieu en dépit du fait que HypenatedConcat utilise le passage de paramètre avec appel par valeur.
Important
Pour les types de référence, si vous passez un paramètre par valeur, une copie de la référence à un objet est passée.Cette copie est toujours associée aux mêmes données d'instance que la référence d'origine (jusqu'à ce que la variable de référence soit assignée à un nouvel objet).L'appel par référence n'est pas forcément nécessaire pour qu'une fonction modifie un paramètre.
Cette autre version du programme montre comment implémenter la fonction HypenatedConcat en tant que fonction pure.
class Program
{
public static string HyphenatedConcat(string s, string appendStr)
{
return (s + '-' + appendStr);
}
public static void Main(string[] args)
{
string s1 = "StringOne";
string s2 = HyphenatedConcat(s1, "StringTwo");
Console.WriteLine(s2);
}
}
Module Module1
Public Function HyphenatedConcat(ByVal s As String, ByVal appendStr As String) As String
Return (s & "-" & appendStr)
End Function
Sub Main()
Dim s1 As String = "StringOne"
Dim s2 As String = HyphenatedConcat(s1, "StringTwo")
Console.WriteLine(s2)
End Sub
End Module
Là encore, cette version génère la même ligne de sortie : StringOne-StringTwo. Notez que pour conserver la valeur concaténée, elle est stockée dans la variable intermédiaire s2.
L'une des approches qui peuvent se révéler très utiles consiste à écrire des fonctions localement impures (à savoir, qui déclarent et modifient des variables locales) mais globalement pures. Ces fonctions possèdent une grande partie des caractéristiques de composabilité requises, mais elles évitent certaines des tâches de programmation fonctionnelle plus compliquées, telles que l'utilisation de la récursivité lorsqu'une simple boucle génèrerait le même résultat.
L'une des caractéristiques importantes des opérateurs de requête standard est qu'ils sont implémentés en tant que fonctions pures.
Pour plus d'informations, consultez Vue d'ensemble des opérateurs de requête standard.
Introduction aux transformations fonctionnelles pures
Comparaison de la programmation fonctionnelle et de la programmation impérative