Un langage fortement typé
Rust est un langage typé statiquement, cela veut dire que le type des variables doit être connu au moment de la compilation.
Pour certains langages, cela veut dire que le programmeur doit spécifier de quel type est chaque variable (comme Java, C ou C++). Cela impose une lourde charge au programmeur, l'obligeant à répéter le type d'une variable plusieurs fois, ce qui peut potentiellement entraver la lisibilité et la refactorisation dans des projets plus conséquents.
Tandis que d'autres langages utilisent une forme d'inférence de type, ils sont capable de déduire le type d'une variable et le programmeur ne devra la spécifier que si le compilateur n'est pas capable de l'inférer pour nous (comme Haskell, Scala ou Kotlin). Certains langages à typage statique permettent l'inférence de type de programme entier. Bien que pratique lors du développement initial, cela réduit la capacité du compilateur à fournir des informations d'erreur utiles lorsque les types ne correspondent plus.
Rust apprends de ces deux styles et exige que les éléments de premier niveau tels que les arguments de fonction et les constantes aient des types explicites, tout en permettant l'inférence de type à l'intérieur du corps de la fonction. Il fait de son mieux penser plus loin programmeur tout en encourageant la maintenabilité à long terme
Dans cet exemple, le compilateur de Rust infère le type de twice
, 2
, et 1
car le paramètre value
et le type de retour sont déclaré comme entiers signé sur 32 bits
fn simple_math(value: i32) -> i32 {
let twice = value * 2;
twice - 1
}
Un langage typé dynamiquement, par contre, doit connaître le type des variables au moment de l'exécution. Cela veut dire que le programmeur peut écrire un peu plus vite car il n'a pas à spécifier le type à chaque fois. Python, JavaScript ou Ruby sont des langages typés dynamiquement.
Le principal avantage de Rust ici est que toute vérification de type peut être faite par le compilateur, ce qui implique que beaucoup de bugs peuvent être détectés durant le développement.
Cela ne veut pas dire que tous les systèmes de type statique sont équivalent.
De nombreux langages à typage statique ont une grande astérisque à côté d'eux: ils permettent le concept de NULL
.
Le concept de NULL
permet à toute valeur d'être ce qu'elle dit ou rien, créant effectivement un deuxième type possible pour chaque type.
Comme Haskell et quelques autre langages de programmation moderne, Rust encode cette possibilité en utilisant un type optionnel, et le compilateur vous demande de gérer le cas None
.
Cela empêche les occurrences de la redoutée erreur d'exécution TypeError: Cannot read property 'foo' of null
(ou l'équivalent du langage), au lieu de la promouvoir comme une erreur de compilation que vous pouvez résoudre avant qu'un utilisateur ne le voie.
Voici en exemple une fonction pour saluer quelqu'un en fonction de si l'on connaît son nom ou non.
fn greet_user(name: Option<String>) {
match name {
Some(name) => println!("Bonjour, {}", name),
None => println!("Bienvenue à vous, étranger!"),
}
}
Si l'on oublie le cas None
dans le match
ou aurait essayé d'utiliser name
comme si il s'agissait d'une valeur String
toujours présente, le compilateur se plaindra.
#![allow(unused_variables)] fn main() { fn greet_user(name: Option<String>) { match name { Some(name) => println!("Bonjour, {}", name), } } }
Note: n'hésitez pas à tester le code lorsque vous voyez un bouton play dans le bloc de code. Dans ce cas, cela vous permettra de voir l'erreur renvoyé par le compilateur.