Types contravariants dans Go 1.18 Génériques
Dans Go 1.18, les types génériques permettent de définir des fonctions avec des types qui varient en fonction des arguments de type. La contravariance, un type spécifique de variance, permet aux fonctions d'accepter un plus large éventail de types d'entrée tout en conservant une relation bien définie entre les types d'entrée et de sortie.
Fonction Pipe
La fonction Pipe fournie vise à composer des fonctions qui transforment des valeurs. Il s'attend à ce que le type de sortie de la fonction de gauche corresponde au type d'entrée de la fonction de droite. Cependant, dans l'exemple suivant, la compilation échoue :
func Pipe[A, T1, T2 any](left func(A) T1, right func(T1) T2) func(A) T2 { return func(a A) T2 { return right(left(a)) } } func OpenFile(name string) *os.File { ... } func ReadAll(rdr io.Reader) []byte { ... } var OpenRead = Pipe(OpenFile, ReadAll)
Le compilateur se plaint car T1 est considéré comme *os.File même s'il est compatible avec io.Reader.
Y a-t-il une solution ?
Malheureusement, il n'y a aucun moyen dans les génériques Go 1.18 de modifier la signature de Pipe pour activer le comportement souhaité. Go ne prend pas en charge les types de résultats covariants, ce qui signifie que le type de sortie d'une fonction ne peut pas varier en fonction du type d'entrée.
Est-ce un bug ?
Le comportement dans Go 1.18 n'est pas un bug. C'est intentionnel, comme expliqué dans la FAQ pour les génériques Go.
Solution de contournement
Pour résoudre cette limitation, une version modifiée de la fonction Pipe peut être utilisée, mais cela sacrifie la sécurité des types au moment de la compilation :
func Pipe[A, T1, T2, T3 any](left func(A) T1, right func(T2) T3) func(A) T3 { return func(a A) T3 { return right(any(left(a)).(T2)) } }
Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!