72 votes

Architecture d'application / composition en F #

J'ai été faire SOLIDE en C# pour un joli niveau extrême, ces derniers temps, et à un certain moment de réaliser que je suis essentiellement ne pas faire grand chose d'autre que de la composition de fonctions aujourd'hui. Et après j'ai récemment commencé à regarder les F# encore une fois, j'ai pensé que ce serait probablement la plus appropriée le choix de la langue pour beaucoup de ce que je fais en ce moment, donc je voudrais l'essayer et le port d'un monde réel projet C# F# comme une preuve de concept. Je pense que je pourrais enlever le code (dans un très non-idiomatiques de la mode), mais je ne peux pas imaginer ce qu'une architecture ressemblerait qui me permet de travailler dans la même souplesse qu'en C#.

Ce que je veux dire par là, c'est que j'ai beaucoup de petites classes et les interfaces que je compose à l'aide d'un conteneur IoC, et j'ai aussi utiliser des formes comme Décorateur et Composite beaucoup. Il en résulte une (à mon avis) très flexibles et évolutives architecture d'ensemble qui me permet de facilement remplacer ou étendre les fonctionnalités à tout moment de la demande. Selon la taille de l'changements nécessaires, c'est, pourrais-je seulement besoin d'écrire une nouvelle mise en œuvre d'une interface, remplacer le Cio à l'enregistrement et à faire. Même si le changement est plus grand, je peux remplacer une partie de l'objet graphique tandis que le reste de l'application signifie tout simplement qu'elle a fait avant.

Maintenant, avec F#, je n'ai pas de classes et d'interfaces (je sais que je peux, mais je pense que c'est à côté de la question, quand je veux faire effectivement la programmation fonctionnelle), je n'ai pas de constructeur d'injection, et je n'ai pas du Cio conteneurs. Je sais que je peux faire quelque chose comme un Décorateur motif à l'aide des fonctions d'ordre supérieur, mais ce n'est pas tout à fait semblent me donner le même type de flexibilité et de facilité de maintenance des classes avec le constructeur d'injection.

Tenir compte de ces C# types:

public class Dings
{
    public string Lol { get; set; }

    public string Rofl { get; set; }
}

public interface IGetStuff
{
    IEnumerable<Dings> For(Guid id);
}

public class AsdFilteringGetStuff : IGetStuff
{
    private readonly IGetStuff _innerGetStuff;

    public AsdFilteringGetStuff(IGetStuff innerGetStuff)
    {
        this._innerGetStuff = innerGetStuff;
    }

    public IEnumerable<Dings> For(Guid id)
    {
        return this._innerGetStuff.For(id).Where(d => d.Lol == "asd");
    }
}

public class GeneratingGetStuff : IGetStuff
{
    public IEnumerable<Dings> For(Guid id)
    {
        IEnumerable<Dings> dingse;

        // somehow knows how to create correct dingse for the ID

        return dingse;
    }
}

Je vais dire à mon conteneur IoC pour résoudre AsdFilteringGetStuff pour IGetStuff et GeneratingGetStuff de sa propre dépendance à cette interface. Maintenant si j'ai besoin d'un filtre différent ou enlever le filtre au total, j'ai peut-être besoin la mise en œuvre de l' IGetStuff , puis il suffit de changer le Cio de l'enregistrement. Tant que l'interface reste la même, je n'ai pas besoin de toucher à des trucs à l'intérieur de l'application. OCP et LSP, activé par le DIP.

Maintenant, je fais quoi en F#?

type Dings (lol, rofl) =
    member x.Lol = lol
    member x.Rofl = rofl

let GenerateDingse id =
    // create list

let AsdFilteredDingse id =
    GenerateDingse id |> List.filter (fun x -> x.Lol = "asd")

J'aime la façon dont beaucoup moins de code, c'est, mais j'ai perdent de leur souplesse. Oui, je peux appeler AsdFilteredDingse ou GenerateDingse dans le même lieu, parce que les types sont les mêmes - mais comment puis-je décider sur un sans codage en dur elle sur le site d'appel? Aussi, bien que ces deux fonctions sont interchangeables, maintenant, je ne peut pas remplacer le générateur de fonction à l'intérieur d' AsdFilteredDingse sans modification de cette fonction. Ce n'est pas très agréable.

Prochaine tentative:

let GenerateDingse id =
    // create list

let AsdFilteredDingse (generator : System.Guid -> Dings list) id =
    generator id |> List.filter (fun x -> x.Lol = "asd")

Maintenant, j'ai la composabilité en faisant AsdFilteredDingse un ordre supérieur de la fonction, mais les deux fonctions ne sont pas interchangeables plus. Sur la deuxième pensée, ils probablement ne devrait pas être de toute façon.

Ce que je pourrais faire d'autre? J'ai pu imiter la "composition de la racine" concept de mon C# SOLIDES dans le dernier fichier du projet F#. La plupart des fichiers sont juste des collections de fonctions, alors j'ai une sorte de "registre", qui remplace le conteneur IoC, et enfin il y a une fonction que j'appelle pour exécuter la demande et qui utilise les fonctions de la "greffe". Dans le "registre", je sais que j'ai besoin d'une fonction de type (Guid> des Coups de liste), que j'appellerai GetDingseForId. C'est celui que j'appelle, jamais l'individu fonctions définies précédemment.

Pour le décorateur, la définition serait

let GetDingseForId id = AsdFilteredDingse GenerateDingse

Pour supprimer le filtre, je changerais que pour

let GetDingseForId id = GenerateDingse

Le revers de la médaille(?) cela, c'est que toutes les fonctions qui utilisent d'autres fonctions devraient raisonnablement être des fonctions d'ordre supérieur, et mon "registre" aurait de la carte toutes les fonctions que j'utilise, parce que les fonctions réelles définies précédemment peuvent pas appeler toutes les fonctions définies plus tard et, en particulier, ceux de la "greffe". Je pourrais également exécuter dans la circulaire des problèmes de dépendance avec le "registre" des mappages.

Tout cela a un sens? Comment avez-vous vraiment construire une application F# pour être facile à entretenir et évolutives (pour ne pas mentionner testable)?

58voto

Mark Seemann Points 102767

C'est facile une fois que vous vous rendez compte que l'Orienté Objet Constructeur d'Injection correspond de très près à la Fonctionnelle Partielle de la Fonction de Demande.

Tout d'abord, j'écrirais Dings comme un type d'enregistrement:

type Dings = { Lol : string; Rofl : string }

En F#, IGetStuff interface peut être réduit à une seule fonction avec la signature

Guid -> seq<Dings>

Un client à l'aide de cette fonction serait de prendre comme paramètre:

let Client getStuff =
    getStuff(Guid("055E7FF1-2919-4246-876E-1DA71980BE9C")) |> Seq.toList

La signature de l' Client de la fonction est:

(Guid -> #seq<'b>) -> 'b list

Comme vous pouvez le voir, il prend une fonction de la cible de la signature entrée et retourne une liste.

Générateur

Le générateur de fonction est facile à écrire:

let GenerateDingse id =
    seq {
        yield { Lol = "Ha!"; Rofl = "Ha ha ha!" }
        yield { Lol = "Ho!"; Rofl = "Ho ho ho!" }
        yield { Lol = "asd"; Rofl = "ASD" } }

L' GenerateDingse fonction a cette signature:

'a -> seq<Dings>

C'est en fait plus générique que l' Guid -> seq<Dings>, mais ce n'est pas un problème. Si vous ne voulez composer l' Client avec GenerateDingse, vous pouvez simplement l'utiliser comme ceci:

let result = Client GenerateDingse

Qui permettrait le retour de tous les trois - Ding des valeurs de GenerateDingse.

Décorateur

L'original Décorateur est un peu plus difficile, mais pas beaucoup. En général, au lieu de l'ajout de la Décorées (intérieure) type comme un argument du constructeur, il vous suffit de l'ajouter en tant que la valeur d'un paramètre à une fonction:

let AdsFilteredDingse id s = s |> Seq.filter (fun d -> d.Lol = "asd")

Cette fonction a cette signature:

'a -> seq<Dings> -> seq<Dings>

Ce n'est pas tout à fait ce que nous voulons, mais il est facile de le composer avec d' GenerateDingse:

let composed id = GenerateDingse id |> AdsFilteredDingse id

L' composed fonction a la signature

'a -> seq<Dings>

Tout ce que nous recherchons!

Vous pouvez maintenant utiliser Client avec composed comme ceci:

let result = Client composed

qui sera de retour que [{Lol = "asd"; Rofl = "ASD";}].

Vous ne pas avoir à définir l' composed fonction première; vous pouvez aussi composer sur place:

let result = Client (fun id -> GenerateDingse id |> AdsFilteredDingse id)

Cela revient aussi [{Lol = "asd"; Rofl = "ASD";}].

Alternative Décorateur

L'exemple précédent fonctionne bien, mais n'est pas vraiment Décorer une fonction similaire. Voici une autre:

let AdsFilteredDingse id f = f id |> Seq.filter (fun d -> d.Lol = "asd")

Cette fonction a pour signature:

'a -> ('a -> #seq<Dings>) -> seq<Dings>

Comme vous pouvez le voir, l' f argument est une autre fonction avec la même signature, de sorte qu'il ressemble de plus près à le Décorateur modèle. Vous pouvez le composer comme ceci:

let composed id = GenerateDingse |> AdsFilteredDingse id

Encore une fois, vous pouvez utiliser Client avec composed comme ceci:

let result = Client composed

ou en ligne, comme ceci:

let result = Client (fun id -> GenerateDingse |> AdsFilteredDingse id)

Pour plus d'exemples et principes pour la composition des applications entières avec F#, voir mon cours en ligne sur l'architecture Fonctionnelle avec F#.

Pour en savoir plus sur Orientée Objet Principes et la façon dont ils correspondent à la Programmation Fonctionnelle, voir mon billet de blog sur les principes SOLIDES et comment ils s'appliquent à FP.

Prograide.com

Prograide est une communauté de développeurs qui cherche à élargir la connaissance de la programmation au-delà de l'anglais.
Pour cela nous avons les plus grands doutes résolus en français et vous pouvez aussi poser vos propres questions ou résoudre celles des autres.

Powered by:

X