Types flexibles
Besoin
Lors de la création de certaines fonctions génériques, il faut spécifier qu'un paramètre de type est un sous-type d'un certain autre type.
→ Illustration grâce à un exemple :
open System.Collections.Generic
// V1
let add item (collection: ICollection<_>) =
collection.Add item
collection
let a = List([1..3]) // List<int>
let b = a |> add 4 // ICollection<int> != List<int> ❗Solutions :
V2 : indiquer une contrainte de type
V3 : indiquer un type flexible
(* V1 ❌ *) let add item (collection: ICollection<_>) = …
(* V2a 😖 *) let add<'t, 'u when 'u :> ICollection<'t>> (item: 't) (collection: 'u) : 'u = …
(* V2b 😕 *) let add (item: 't) (collection: 'u when 'u :> ICollection<'t>) : 'u = …
(* V3 ✅ *) let add item (collection: #ICollection<_>) = …⚖️ Bilan :
V2a : syntaxe similaire au C♯ → verbeux et pas très lisible ! 😖
V2b : version améliorée en F♯ → + lisible mais encore un peu verbeux ! 😕
V3 : syntaxe proche de V1 → concision « dans l'esprit F♯ » ✅
Autres usages
Faciliter l'usage de la fonction sans avoir besoin d'un upcast
Avec un type flexible :
Dans l'exemple ci-dessous, items est inféré avec la bonne contrainte :
💡 Quid de faciliter la lecture du code avec un type flexible ?
⚠️ Astuce précédente ne marche pas toujours !
x et y doivent satisfaire 2 conditions
'a : comparison≃ les types dexetyimplémententIComparable→(x: #IComparable) (y: #IComparable)?x:'aety:'a→xetyont le même type → Non exprimable sous forme de type flexible ! 😞
Résumé
Type flexible :
Utilisé dans la déclaration de certaine fonction générique
Indique qu'un paramètre de type est un sous-type d'un type spécifié
Sucre syntaxique au format
#super-typeÉquivalent de
'T when 'T :> super-type
Autres usages :
Faciliter l'usage de la fonction sans avoir besoin d'un upcast
Faciliter la lecture du code ?
Mis à jour
Ce contenu vous a-t-il été utile ?