Rust - Types génériques

Les génériques sont une facilité pour écrire du code pour plusieurs contextes avec différents types. Dans Rust, les génériques font référence au paramétrage des types de données et des traits. Generics permet d'écrire du code plus concis et plus propre en réduisant la duplication de code et en assurant la sécurité de type. Le concept de génériques peut être appliqué aux méthodes, fonctions, structures, énumérations, collections et traits.

le <T> syntaxconnu sous le nom de paramètre de type, est utilisé pour déclarer une construction générique. T représente n'importe quel type de données.

Illustration: Collection générique

L'exemple suivant déclare un vecteur qui ne peut stocker que des entiers.

fn main(){
   let mut vector_integer: Vec<i32> = vec![20,30];
   vector_integer.push(40);
   println!("{:?}",vector_integer);
}

Production

[20, 30, 40]

Considérez l'extrait suivant -

fn main() {
   let mut vector_integer: Vec<i32> = vec![20,30];
   vector_integer.push(40);
   vector_integer.push("hello"); 
   //error[E0308]: mismatched types
   println!("{:?}",vector_integer);
}

L'exemple ci-dessus montre qu'un vecteur de type entier ne peut stocker que des valeurs entières. Donc, si nous essayons de pousser une valeur de chaîne dans la collection, le compilateur retournera une erreur. Les génériques rendent les collections plus sûres.

Illustration: Structure générique

Le paramètre type représente un type, que le compilateur remplira plus tard.

struct Data<T> {
   value:T,
}
fn main() {
   //generic type of i32
   let t:Data<i32> = Data{value:350};
   println!("value is :{} ",t.value);
   //generic type of String
   let t2:Data<String> = Data{value:"Tom".to_string()};
   println!("value is :{} ",t2.value);
}

L'exemple ci-dessus déclare une structure générique nommée Data . Le type <T> indique un type de données. La fonction main () crée deux instances - une instance entière et une instance de chaîne, de la structure.

Production

value is :350
value is :Tom

Traits

Les traits peuvent être utilisés pour implémenter un ensemble standard de comportements (méthodes) sur plusieurs structures. Les traits sont commeinterfacesdans la programmation orientée objet. La syntaxe du trait est comme indiqué ci-dessous -

Déclarer un trait

trait some_trait {
   //abstract or method which is empty
   fn method1(&self);
   // this is already implemented , this is free
   fn method2(&self){
      //some contents of method2
   }
}

Les traits peuvent contenir des méthodes concrètes (méthodes avec corps) ou des méthodes abstraites (méthodes sans corps). Utilisez une méthode concrète si la définition de la méthode sera partagée par toutes les structures mettant en œuvre le trait. Cependant, une structure peut choisir de remplacer une fonction définie par le trait.

Utilisez des méthodes abstraites si la définition de méthode varie pour les structures d'implémentation.

Syntaxe - Implémenter un trait

impl some_trait for structure_name {
   // implement method1() there..
   fn method1(&self ){
   }
}

Les exemples suivants définissent un trait imprimable avec une méthode print () , qui est implémentée par le livre de structure .

fn main(){
   //create an instance of the structure
   let b1 = Book {
      id:1001,
      name:"Rust in Action"
   };
   b1.print();
}
//declare a structure
struct Book {
   name:&'static str,
   id:u32
}
//declare a trait
trait Printable {
   fn print(&self);
}
//implement the trait
impl Printable for Book {
   fn print(&self){
      println!("Printing book with id:{} and name {}",self.id,self.name)
   }
}

Production

Printing book with id:1001 and name Rust in Action

Fonctions génériques

L'exemple définit une fonction générique qui affiche un paramètre qui lui est passé. Le paramètre peut être de n'importe quel type. Le type du paramètre doit implémenter le trait Display afin que sa valeur puisse être imprimée par println! macro.

use std::fmt::Display;

fn main(){
   print_pro(10 as u8);
   print_pro(20 as u16);
   print_pro("Hello TutorialsPoint");
}

fn print_pro<T:Display>(t:T){
   println!("Inside print_pro generic function:");
   println!("{}",t);
}

Production

Inside print_pro generic function:
10
Inside print_pro generic function:
20
Inside print_pro generic function:
Hello TutorialsPoint