Rouille - Guide rapide

Rust est un langage de programmation au niveau des systèmes, développé par Graydon Hoare. Mozilla Labs a par la suite acquis le programme.

Langages de programmation des systèmes Application v / s

Les langages de programmation d'application tels que Java / C # sont utilisés pour créer des logiciels, qui fournissent des services directement à l'utilisateur. Ils nous aident à créer des applications métier telles que des feuilles de calcul, des traitements de texte, des applications Web ou des applications mobiles.

Les langages de programmation de systèmes tels que C / C ++ sont utilisés pour créer des logiciels et des plates-formes logicielles. Ils peuvent être utilisés pour construire des systèmes d'exploitation, des moteurs de jeux, des compilateurs, etc. Ces langages de programmation nécessitent un degré élevé d'interaction matérielle.

Les systèmes et les langages de programmation d'applications sont confrontés à deux problèmes majeurs -

  • Il est difficile d'écrire du code sécurisé.
  • Il est difficile d'écrire du code multi-thread.

Pourquoi Rust?

Rust se concentre sur trois objectifs -

  • Safety
  • Speed
  • Concurrency

Le langage a été conçu pour développer des logiciels hautement fiables et rapides de manière simple. Rust peut être utilisé pour écrire des programmes de haut niveau vers des programmes spécifiques au matériel.

Performance

Le langage de programmation Rust n'a pas de Garbage Collector (GC) par conception. Cela améliore les performances lors de l'exécution.

Sécurité de la mémoire au moment de la compilation

Les logiciels créés avec Rust sont à l'abri des problèmes de mémoire tels que les pointeurs suspendus, les dépassements de tampon et les fuites de mémoire.

Applications multi-threadées

Les règles de propriété et de sécurité de la mémoire de Rust fournissent la concurrence sans course de données.

Prise en charge de l'assemblage Web (WASM)

Web Assembly permet d'exécuter des algorithmes à haute intensité de calcul dans le navigateur, sur les appareils embarqués ou ailleurs. Il fonctionne à la vitesse du code natif. Rust peut être compilé dans Web Assembly pour une exécution rapide et fiable.

L'installation de Rust est facilitée grâce à rustup, un outil basé sur console pour gérer les versions de Rust et les outils associés.

Installation sous Windows

Apprenons à installer RUST sur Windows.

  • L'installation de Visual Studio 2013 ou version ultérieure avec les outils C ++ est obligatoire pour exécuter le programme Rust sous Windows. Tout d'abord, téléchargez Visual Studio à partir d'ici VS 2013 Express

  • Télécharger et installer rustup outil pour Windows. rustup-init.exeest disponible en téléchargement ici - Rust Lang

  • Double-cliquez rustup-init.exefichier. En cliquant, l'écran suivant apparaît.

  • Appuyez sur Entrée pour l'installation par défaut. Une fois l'installation terminée, l'écran suivant apparaît.

  • Depuis l'écran d'installation, il est clair que les fichiers liés à Rust sont stockés dans le dossier -

    C: \ Users \ {PC} \. Cargo \ bin

Le contenu du dossier est -

cargo-fmt.exe
cargo.exe
rls.exe
rust-gdb.exe
rust-lldb.exe
rustc.exe // this is the compiler for rust
rustdoc.exe
rustfmt.exe
rustup.exe
  • Cargoest le gestionnaire de paquets de Rust. Pour vérifier sicargo est installé, exécutez la commande suivante -

C:\Users\Admin>cargo -V
cargo 1.29.0 (524a578d7 2018-08-05)
  • Le compilateur de Rust est rustc. Pour vérifier la version du compilateur, exécutez la commande suivante -

C:\Users\Admin>cargo -V
cargo 1.29.0 (524a578d7 2018-08-05)

Installation sous Linux / Mac

À installer rustup sous Linux ou macOS, ouvrez un terminal et entrez la commande suivante.

$ curl https://sh.rustup.rs -sSf | sh

La commande télécharge un script et démarre l'installation du rustuptool, qui installe la dernière version stable de Rust. Vous pourriez être invité à entrer votre mot de passe. Si l'installation réussit, la ligne suivante apparaîtra -

Rust is installed now. Great!

Le script d'installation ajoute automatiquement Rust à votre système PATH après votre prochaine connexion. Pour commencer à utiliser Rust tout de suite au lieu de redémarrer votre terminal, exécutez la commande suivante dans votre shell pour ajouter manuellement Rust à votre système PATH -

$ source $HOME/.cargo/env

Vous pouvez également ajouter la ligne suivante à votre ~ / .bash_profile -

$ export PATH="$HOME/.cargo/bin:$PATH"

NOTE - Lorsque vous essayez de compiler un programme Rust et que vous obtenez des erreurs indiquant qu'un éditeur de liens n'a pas pu s'exécuter, cela signifie qu'un éditeur de liens n'est pas installé sur votre système et que vous devrez en installer un manuellement.

Utilisation du terrain de codage de points des didacticiels pour RUST

Une boucle de lecture-évaluation-impression (REPL) est un shell interactif facile à utiliser pour compiler et exécuter des programmes informatiques. Si vous souhaitez compiler et exécuter des programmes Rust en ligne dans le navigateur, utilisez Tutorialspoint Coding Ground .

Ce chapitre explique la syntaxe de base du langage Rust via un HelloWorld exemple.

  • Créer un HelloWorld-App dossier et accédez à ce dossier sur le terminal

C:\Users\Admin>mkdir HelloWorld-App
C:\Users\Admin>cd HelloWorld-App
C:\Users\Admin\HelloWorld-App>
  • Pour créer un fichier Rust, exécutez la commande suivante -

C:\Users\Admin\HelloWorld-App>notepad Hello.rs

Les fichiers programme Rust ont une extension .rs. La commande ci-dessus crée un fichier videHello.rset l'ouvre dans NOTEpad. Ajoutez le code ci-dessous à ce fichier -

fn
main(){
   println!("Rust says Hello to TutorialsPoint !!");
}

Le programme ci-dessus définit une fonction main fn main () . Le mot-clé fn est utilisé pour définir une fonction. Le main () est une fonction prédéfinie qui agit comme un point d'entrée dans le programme. println! est une macro prédéfinie dans Rust. Il est utilisé pour imprimer une chaîne (ici Hello) sur la console. Les appels de macro sont toujours marqués d'un point d'exclamation - ! .

  • Compilez le Hello.rs fichier utilisant rustc.

C:\Users\Admin\HelloWorld-App>rustc Hello.rs

Une fois la compilation réussie du programme, un fichier exécutable ( nom_fichier.exe ) est généré. Pour vérifier si le fichier .exe est généré, exécutez la commande suivante.

C:\Users\Admin\HelloWorld-App>dir
//lists the files in folder
Hello.exe
Hello.pdb
Hello.rs
  • Exécutez le fichier Hello.exe et vérifiez la sortie.

Qu'est-ce qu'une macro?

Rust fournit un système macro puissant qui permet la méta-programmation. Comme vous l'avez vu dans l'exemple précédent, les macros ressemblent à des fonctions, sauf que leur nom se termine par un bang (!), Mais au lieu de générer un appel de fonction, les macros sont développées en code source qui est compilé avec le reste du programme. Par conséquent, ils fournissent plus de fonctionnalités d'exécution à un programme contrairement aux fonctions. Les macros sont une version étendue des fonctions.

Utilisation de println! Macro - Syntaxe

println!(); // prints just a newline
println!("hello ");//prints hello
println!("format {} arguments", "some"); //prints format some arguments

Commentaires en rouille

Les commentaires sont un moyen d'améliorer la lisibilité d'un programme. Les commentaires peuvent être utilisés pour inclure des informations supplémentaires sur un programme comme l'auteur du code, des conseils sur une fonction / construction, etc. Le compilateur ignore les commentaires.

Rust prend en charge les types de commentaires suivants -

  • Commentaires sur une seule ligne (//) - Tout texte entre un // et la fin d'une ligne est traité comme un commentaire

  • Commentaires sur plusieurs lignes (/ * * /) - Ces commentaires peuvent s'étendre sur plusieurs lignes.

Exemple

//this is single line comment

/* This is a
   Multi-line comment
*/

Exécuter en ligne

Les programmes Rust peuvent être exécutés en ligne via Tutorialspoint Coding Ground . Écrivez le programme HelloWorld dans l'onglet Editeur et cliquez sur Exécuter pour afficher le résultat.

Le système de types représente les différents types de valeurs pris en charge par la langue. Le système de types vérifie la validité des valeurs fournies avant qu'elles ne soient stockées ou manipulées par le programme. Cela garantit que le code se comporte comme prévu. Le système de types permet en outre des indications de code plus riches et une documentation automatisée.

Rust est un langage typé statiquement. Chaque valeur de Rust est d'un certain type de données. Le compilateur peut déduire automatiquement le type de données de la variable en fonction de la valeur qui lui est assignée.

Déclarer une variable

Utilisez le let mot-clé pour déclarer une variable.

fn main() {
   let company_string = "TutorialsPoint";  // string type
   let rating_float = 4.5;                 // float type
   let is_growing_boolean = true;          // boolean type
   let icon_char = '♥';                    //unicode character type

   println!("company name is:{}",company_string);
   println!("company rating on 5 is:{}",rating_float);
   println!("company is growing :{}",is_growing_boolean);
   println!("company icon is:{}",icon_char);
}

Dans l'exemple ci-dessus, le type de données des variables sera déduit des valeurs qui leur sont attribuées. Par exemple, Rust attribuera le type de données chaîne à la variable company_string , le type de données float à rating_float , etc.

Le println! macro prend deux arguments -

  • Une syntaxe spéciale {} , qui est l'espace réservé
  • Le nom de la variable ou une constante

L'espace réservé sera remplacé par la valeur de la variable

La sortie de l'extrait de code ci-dessus sera -

company name is: TutorialsPoint
company rating on 5 is:4.5
company is growing: true
company icon is: ♥

Types scalaires

Un type scalaire représente une valeur unique. Par exemple, 10,3.14, «c». Rust a quatre types scalaires principaux.

  • Integer
  • Floating-point
  • Booleans
  • Characters

Nous en apprendrons davantage sur chaque type dans nos sections suivantes.

Entier

Un entier est un nombre sans composante fractionnaire. En termes simples, le type de données entier est utilisé pour représenter des nombres entiers.

Les entiers peuvent être classés comme signés et non signés. Les entiers signés peuvent stocker des valeurs négatives et positives. Les entiers non signés ne peuvent stocker que des valeurs positives. Une description détaillée si les types entiers sont donnés ci-dessous -

Sr.No. Taille Signé Non signé
1 8 bits i8 u8
2 16 bits i16 u16
3 32 bits i32 u32
4 64 bits i64 u64
5 128 bits i128 u128
6 Cambre isize utiliser

La taille d'un entier peut être arch . Cela signifie que la taille du type de données sera dérivée de l' architecture de la machine. Un entier dont la taille est arch sera de 32 bits sur une machine x86 et de 64 bits sur une machine x64. Un entier arch est principalement utilisé lors de l'indexation d'une sorte de collection.

Illustration

fn main() {
   let result = 10;    // i32 by default
   let age:u32 = 20;
   let sum:i32 = 5-15;
   let mark:isize = 10;
   let count:usize = 30;
   println!("result value is {}",result);
   println!("sum is {} and age is {}",sum,age);
   println!("mark is {} and count is {}",mark,count);
}

La sortie sera comme indiqué ci-dessous -

result value is 10
sum is -10 and age is 20
mark is 10 and count is 30

Le code ci-dessus renverra une erreur de compilation si vous remplacez la valeur de l' âge par une valeur à virgule flottante.

Plage d'entiers

Chaque variante signée peut stocker des nombres de - (2 ^ (n-1) à 2 ^ (n-1) -1 , où n est le nombre de bits utilisé par la variante. Par exemple, i8 peut stocker des nombres de - (2 ^ 7) à 2 ^ 7 -1 - ici nous avons remplacé n par 8.

Chaque variante non signée peut stocker des nombres de 0 à (2 ^ n) -1 . Par exemple, u8 peut stocker des nombres de 0 à 2 ^ 7 , ce qui est égal à 0 à 255.

Débordement d'entier

Un dépassement d'entier se produit lorsque la valeur affectée à une variable entière dépasse la plage définie par Rust pour le type de données. Comprenons cela avec un exemple -

fn main() {
   let age:u8 = 255;

   // 0 to 255 only allowed for u8
   let weight:u8 = 256;   //overflow value is 0
   let height:u8 = 257;   //overflow value is 1
   let score:u8 = 258;    //overflow value is 2

   println!("age is {} ",age);
   println!("weight is {}",weight);
   println!("height is {}",height);
   println!("score is {}",score);
}

La plage valide de la variable u8 non signée est comprise entre 0 et 255. Dans l'exemple ci-dessus, les variables reçoivent des valeurs supérieures à 255 (limite supérieure pour une variable entière dans Rust). Lors de l'exécution, le code ci-dessus renverra un avertissement -warning − literal out of range for u8pour les variables de poids, de taille et de score. Les valeurs de dépassement après 255 commenceront à partir de 0, 1, 2, etc. La sortie finale sans avertissement est comme indiqué ci-dessous -

age is 255
weight is 0
height is 1
score is 2

Flotte

Le type de données Float dans Rust peut être classé comme f32 et f64. Le type f32 est un flotteur simple précision et f64 a une double précision. Le type par défaut est f64. Considérez l'exemple suivant pour en savoir plus sur le type de données float.

fn main() {
   let result = 10.00;        //f64 by default
   let interest:f32 = 8.35;
   let cost:f64 = 15000.600;  //double precision
   
   println!("result value is {}",result);
   println!("interest is {}",interest);
   println!("cost is {}",cost);
}

La sortie sera comme indiqué ci-dessous -

interest is 8.35
cost is 15000.6

Coulée de type automatique

La fonte de type automatique n'est pas autorisée dans Rust. Considérez l'extrait de code suivant. Une valeur entière est affectée à la variable floatinterest.

fn main() {
   let interest:f32 = 8;   // integer assigned to float variable
   println!("interest is {}",interest);
}

Le compilateur lance un mismatched types error comme indiqué ci-dessous.

error[E0308]: mismatched types
   --> main.rs:2:22
   |
 2 | let interest:f32=8;
   |    ^ expected f32, found integral variable
   |
   = note: expected type `f32`
      found type `{integer}`
error: aborting due to previous error(s)

Séparateur de nombres

Pour une lisibilité facile des grands nombres, nous pouvons utiliser un séparateur visuel _ souligné pour séparer les chiffres. C'est-à-dire 50 000 peut être écrit comme 50_000. Ceci est illustré dans l'exemple ci-dessous.

fn main() {
   let float_with_separator = 11_000.555_001;
   println!("float value {}",float_with_separator);
   
   let int_with_separator = 50_000;
   println!("int value {}",int_with_separator);
}

La sortie est donnée ci-dessous -

float value 11000.555001
int value 50000

Booléen

Les types booléens ont deux valeurs possibles: true ou false . Utilisez lebool mot-clé pour déclarer une variable booléenne.

Illustration

fn main() {
   let isfun:bool = true;
   println!("Is Rust Programming Fun ? {}",isfun);
}

La sortie du code ci-dessus sera -

Is Rust Programming Fun ? true

Personnage

Le type de données caractère dans Rust prend en charge les nombres, les alphabets, Unicode et les caractères spéciaux. Utilisez lecharmot-clé pour déclarer une variable de type de données caractère. Le type char de Rust représente une valeur scalaire Unicode, ce qui signifie qu'il peut représenter beaucoup plus qu'un simple ASCII. Les valeurs scalaires Unicode vont deU+0000 à U+D7FF et U+E000 à U+10FFFF compris.

Prenons un exemple pour en savoir plus sur le type de données Character.

fn main() {
   let special_character = '@'; //default
   let alphabet:char = 'A';
   let emoji:char = '';
   
   println!("special character is {}",special_character);
   println!("alphabet is {}",alphabet);
   println!("emoji is {}",emoji);
}

La sortie du code ci-dessus sera -

special character is @
alphabet is A
emoji is

Une variable est un stockage nommé que les programmes peuvent manipuler. En termes simples, une variable aide les programmes à stocker des valeurs. Les variables dans Rust sont associées à un type de données spécifique. Le type de données détermine la taille et la disposition de la mémoire de la variable, la plage de valeurs qui peuvent être stockées dans cette mémoire et l'ensemble des opérations qui peuvent être effectuées sur la variable.

Règles de dénomination d'une variable

Dans cette section, nous allons découvrir les différentes règles de dénomination d'une variable.

  • Le nom d'une variable peut être composé de lettres, de chiffres et du caractère de soulignement.

  • Il doit commencer par une lettre ou un trait de soulignement.

  • Les lettres majuscules et minuscules sont distinctes car Rust est sensible à la casse.

Syntaxe

Le type de données est facultatif lors de la déclaration d'une variable dans Rust. Le type de données est déduit de la valeur affectée à la variable.

La syntaxe pour déclarer une variable est donnée ci-dessous.

let variable_name = value;            // no type specified
let variable_name:dataType = value;   //type specified

Illustration

fn main() {
   let fees = 25_000;
   let salary:f64 = 35_000.00;
   println!("fees is {} and salary is {}",fees,salary);
}

La sortie du code ci-dessus sera fees is 25000 and salary is 35000.

Immuable

Par défaut, les variables sont immuables - en lecture seule dans Rust. En d'autres termes, la valeur de la variable ne peut pas être modifiée une fois qu'une valeur est liée à un nom de variable.

Comprenons cela avec un exemple.

fn main() {
   let fees = 25_000;
   println!("fees is {} ",fees);
   fees = 35_000;
   println!("fees changed is {}",fees);
}

La sortie sera comme indiqué ci-dessous -

error[E0384]: re-assignment of immutable variable `fees`
 --> main.rs:6:3
   |
 3 | let fees = 25_000;
   | ---- first assignment to `fees`
...
 6 | fees=35_000;
   | ^^^^^^^^^^^ re-assignment of immutable variable

error: aborting due to previous error(s)

Le message d'erreur indique la cause de l'erreur - vous ne pouvez pas attribuer de valeurs deux fois aux frais variables immuables. C'est l'une des nombreuses façons dont Rust permet aux programmeurs d'écrire du code et tire parti de la sécurité et de la facilité d'accès concurrentiel.

Mutable

Les variables sont immuables par défaut. Préfixez le nom de la variable avecmutmot-clé pour le rendre mutable. La valeur d'une variable mutable peut être modifiée.

La syntaxe pour déclarer une variable mutable est la suivante:

let mut variable_name = value;
let mut variable_name:dataType = value;
Let us understand this with an example

fn main() {
   let mut fees:i32 = 25_000;
   println!("fees is {} ",fees);
   fees = 35_000;
   println!("fees changed is {}",fees);
}

La sortie de l'extrait est donnée ci-dessous -

fees is 25000
fees changed is 35000

Les constantes représentent des valeurs qui ne peuvent pas être modifiées. Si vous déclarez une constante, il n'y a aucun moyen que sa valeur change. Le mot clé pour utiliser les constantes estconst. Les constantes doivent être typées explicitement. Voici la syntaxe pour déclarer une constante.

const VARIABLE_NAME:dataType = value;

Convention de dénomination des constantes de rouille

La convention de dénomination des constantes est similaire à celle des variables. Tous les caractères d'un nom constant sont généralement en majuscules. Contrairement à la déclaration de variables, lelet Le mot clé n'est pas utilisé pour déclarer une constante.

Nous avons utilisé des constantes dans Rust dans l'exemple ci-dessous -

fn main() {
   const USER_LIMIT:i32 = 100;    // Declare a integer constant
   const PI:f32 = 3.14;           //Declare a float constant

   println!("user limit is {}",USER_LIMIT);  //Display value of the constant
   println!("pi value is {}",PI);            //Display value of the constant
}

Variables constantes v / s

Dans cette section, nous découvrirons les facteurs de différenciation entre les constantes et les variables.

  • Les constantes sont déclarées en utilisant le const mot-clé tandis que les variables sont déclarées à l'aide du let mot-clé.

  • Une déclaration de variable peut éventuellement avoir un type de données tandis qu'une déclaration de constante doit spécifier le type de données. Cela signifie que const USER_LIMIT = 100 entraînera une erreur.

  • Une variable déclarée à l'aide du letLe mot-clé est par défaut immuable. Cependant, vous avez la possibilité de le faire muter en utilisant lemutmot-clé. Les constantes sont immuables.

  • Les constantes peuvent être définies uniquement sur une expression constante et non sur le résultat d'un appel de fonction ou de toute autre valeur qui sera calculée au moment de l'exécution.

  • Les constantes peuvent être déclarées dans n'importe quelle portée, y compris la portée globale, ce qui les rend utiles pour les valeurs que de nombreuses parties du code doivent connaître.

Ombrage des variables et des constantes

Rust permet aux programmeurs de déclarer des variables avec le même nom. Dans ce cas, la nouvelle variable remplace la variable précédente.

Comprenons cela avec un exemple.

fn main() {
   let salary = 100.00;
   let salary = 1.50 ; 
   // reads first salary
   println!("The value of salary is :{}",salary);
}

Le code ci-dessus déclare deux variables par le nom salaire. La première déclaration se voit attribuer une valeur de 100,00 tandis que la deuxième déclaration reçoit une valeur de 1,50. La seconde variable masque ou masque la première variable lors de l'affichage de la sortie.

Production

The value of salary is :1.50

Rust prend en charge les variables avec différents types de données lors de l'observation.

Prenons l'exemple suivant.

Le code déclare deux variables par le nom uname. La première déclaration se voit attribuer une valeur de chaîne, tandis que la seconde déclaration reçoit un entier. La fonction len renvoie le nombre total de caractères dans une valeur de chaîne.

fn main() {
   let uname = "Mohtashim";
   let uname = uname.len();
   println!("name changed to integer : {}",uname);
}

Production

name changed to integer: 9

Contrairement aux variables, les constantes ne peuvent pas être ombrées. Si les variables du programme ci-dessus sont remplacées par des constantes, le compilateur lèvera une erreur.

fn main() {
   const NAME:&str = "Mohtashim";
   const NAME:usize = NAME.len(); 
   //Error : `NAME` already defined
   println!("name changed to integer : {}",NAME);
}

Le type de données String dans Rust peut être classé comme suit:

  • Chaîne littérale(&str)

  • Objet String(String)

Chaîne littérale

Les littéraux de chaîne (& str) sont utilisés lorsque la valeur d'une chaîne est connue au moment de la compilation. Les littéraux de chaîne sont un ensemble de caractères codés en dur dans une variable. Par exemple, laissez company = "Tutorials Point" . Les littéraux de chaîne se trouvent dans le module std :: str. Les littéraux de chaîne sont également appelés tranches de chaîne.

L'exemple suivant déclare deux chaînes littérales: société et emplacement .

fn main() {
   let company:&str="TutorialsPoint";
   let location:&str = "Hyderabad";
   println!("company is : {} location :{}",company,location);
}

Les littéraux de chaîne sont statiques par défaut. Cela signifie que les littéraux de chaîne sont garantis valides pendant toute la durée du programme. Nous pouvons également spécifier explicitement la variable comme statique comme indiqué ci-dessous -

fn main() {
   let company:&'static str = "TutorialsPoint";
   let location:&'static str = "Hyderabad";
   println!("company is : {} location :{}",company,location);
}

Le programme ci-dessus générera la sortie suivante -

company is : TutorialsPoint location :Hyderabad

Objet String

Le type d'objet String est fourni dans la bibliothèque standard. Contrairement à la chaîne littérale, le type d'objet chaîne ne fait pas partie du langage de base. Il est défini comme une structure publique dans la bibliothèque standard pub struct String . String est une collection évolutive. Il est de type mutable et encodé en UTF-8. leStringLe type d'objet peut être utilisé pour représenter les valeurs de chaîne fournies lors de l'exécution. L'objet String est alloué dans le tas.

Syntaxe

Pour créer un objet String, nous pouvons utiliser l'une des syntaxes suivantes -

String::new()

La syntaxe ci-dessus crée une chaîne vide

String::from()

Cela crée une chaîne avec une valeur par défaut transmise en paramètre au from() méthode.

L'exemple suivant illustre l'utilisation d'un objet String.

fn main(){
   let empty_string = String::new();
   println!("length is {}",empty_string.len());

   let content_string = String::from("TutorialsPoint");
   println!("length is {}",content_string.len());
}

L'exemple ci-dessus crée deux chaînes: un objet chaîne vide à l'aide de la nouvelle méthode et un objet chaîne à partir d'une chaîne littérale à l'aide de la méthode from .

La sortie est comme indiqué ci-dessous -

length is 0
length is 14

Méthodes courantes - Objet chaîne

Sr.No. Méthode Signature La description
1 Nouveau() pub const fn new () → Chaîne Crée une nouvelle chaîne vide.
2 to_string () fn to_string (& self) → Chaîne Convertit la valeur donnée en String.
3 remplacer() pub fn replace <'a, P> (&' a self, from: P, to: & str) → String Remplace toutes les correspondances d'un modèle par une autre chaîne.
4 as_str () pub fn as_str (& soi) → & str Extrait une tranche de chaîne contenant la chaîne entière.
5 pousser() pub fn push (& mut self, ch: char) Ajoute le caractère donné à la fin de cette chaîne.
6 push_str () pub fn push_str (& mut self, string: & str) Ajoute une tranche de chaîne donnée à la fin de cette chaîne.
sept len () pub fn len (& self) → utiliser Renvoie la longueur de cette chaîne, en octets.
8 réduire() pub fn trim (& auto) → & str Renvoie une tranche de chaîne avec les espaces de début et de fin supprimés.
9 split_whitespace () pub fn split_whitespace (& self) → SplitWhitespace Divise une tranche de chaîne par un espace et renvoie un itérateur.
dix Divisé() pub fn split <'a, P> (&' a self, pat: P) → Split <'a, P>, où P est motif peut être & str, char, ou une fermeture qui détermine la séparation. Renvoie un itérateur sur les sous-chaînes de cette tranche de chaîne, séparés par des caractères correspondant à un modèle.
11 caractères () pub fn chars (& self) → Chars Renvoie un itérateur sur les caractères d'une tranche de chaîne.

Illustration: nouveau ()

Un objet chaîne vide est créé à l'aide du new()et sa valeur est définie sur hello .

fn main(){
   let mut z = String::new();
   z.push_str("hello");
   println!("{}",z);
}

Production

Le programme ci-dessus génère la sortie suivante -

hello

Illustration: to_string ()

Pour accéder à toutes les méthodes de l'objet String, convertissez une chaîne littérale en type d'objet à l'aide du to_string() fonction.

fn main(){
   let name1 = "Hello TutorialsPoint , 
   Hello!".to_string();
   println!("{}",name1);
}

Production

Le programme ci-dessus génère la sortie suivante -

Hello TutorialsPoint , Hello!

Illustration: replace ()

le replace()La fonction prend deux paramètres - le premier paramètre est un modèle de chaîne à rechercher et le second paramètre est la nouvelle valeur à remplacer. Dans l'exemple ci-dessus, Hello apparaît deux fois dans la chaîne name1 .

La fonction replace remplace toutes les occurrences de la chaîne Hello avec Howdy.

fn main(){
   let name1 = "Hello TutorialsPoint , 
   Hello!".to_string();         //String object
   let name2 = name1.replace("Hello","Howdy");    //find and replace
   println!("{}",name2);
}

Production

Le programme ci-dessus génère la sortie suivante -

Howdy TutorialsPoint , Howdy!

Illustration: as_str ()

le as_str() La fonction extrait une tranche de chaîne contenant la chaîne entière.

fn main() {
   let example_string = String::from("example_string");
   print_literal(example_string.as_str());
}
fn print_literal(data:&str ){
   println!("displaying string literal {}",data);
}

Production

Le programme ci-dessus génère la sortie suivante -

displaying string literal example_string

Illustration: pousser ()

le push() function ajoute le caractère donné à la fin de cette chaîne.

fn main(){
   let mut company = "Tutorial".to_string();
   company.push('s');
   println!("{}",company);
}

Production

Le programme ci-dessus génère la sortie suivante -

Tutorials

Illustration: push_str ()

le push_str() function ajoute une tranche de chaîne donnée à la fin d'une chaîne.

fn main(){
   let mut company = "Tutorials".to_string();
   company.push_str(" Point");
   println!("{}",company);
}

Production

Le programme ci-dessus génère la sortie suivante -

Tutorials Point

Illustration: len ()

le len() La fonction renvoie le nombre total de caractères dans une chaîne (espaces compris).

fn main() {
   let fullname = " Tutorials Point";
   println!("length is {}",fullname.len());
}

Production

Le programme ci-dessus génère la sortie suivante -

length is 20

Illustration: trim ()

La fonction trim () supprime les espaces de début et de fin dans une chaîne. Notez que cette fonction ne supprimera pas les espaces en ligne.

fn main() {
   let fullname = " Tutorials Point \r\n";
   println!("Before trim ");
   println!("length is {}",fullname.len());
   println!();
   println!("After trim ");
   println!("length is {}",fullname.trim().len());
}

Production

Le programme ci-dessus génère la sortie suivante -

Before trim
length is 24

After trim
length is 15

Illustration: split_whitespace ()

le split_whitespace()divise la chaîne d'entrée en différentes chaînes. Il renvoie un itérateur donc nous itérons à travers les jetons comme indiqué ci-dessous -

fn main(){
   let msg = "Tutorials Point has good t
   utorials".to_string();
   let mut i = 1;
   
   for token in msg.split_whitespace(){
      println!("token {} {}",i,token);
      i+=1;
   }
}

Production

token 1 Tutorials
token 2 Point
token 3 has
token 4 good
token 5 tutorials

Illustration: chaîne split ()

le split() stringLa méthode retourne un itérateur sur des sous-chaînes d'une tranche de chaîne, séparées par des caractères correspondant à un modèle. La limitation de la méthode split () est que le résultat ne peut pas être stocké pour une utilisation ultérieure. lecollect peut être utilisée pour stocker le résultat renvoyé par split () sous forme de vecteur.

fn main() {
   let fullname = "Kannan,Sudhakaran,Tutorialspoint";

   for token in fullname.split(","){
      println!("token is {}",token);
   }

   //store in a Vector
   println!("\n");
   let tokens:Vec<&str>= fullname.split(",").collect();
   println!("firstName is {}",tokens[0]);
   println!("lastname is {}",tokens[1]);
   println!("company is {}",tokens[2]);
}

L'exemple ci-dessus divise la chaîne fullname, chaque fois qu'il rencontre une virgule (,).

Production

token is Kannan
token is Sudhakaran
token is Tutorialspoint

firstName is Kannan
lastname is Sudhakaran
company is Tutorialspoint

Illustration: chars ()

Les caractères individuels d'une chaîne sont accessibles à l'aide de la méthode chars. Prenons un exemple pour comprendre cela.

fn main(){
   let n1 = "Tutorials".to_string();

   for n in n1.chars(){
      println!("{}",n);
   }
}

Production

T
u
t
o
r
i
a
l
s

Concaténation de chaînes avec l'opérateur +

Une valeur de chaîne peut être ajoutée à une autre chaîne. C'est ce qu'on appelle la concaténation ou l'interpolation. Le résultat de la concaténation de chaînes est un nouvel objet chaîne. L'opérateur + utilise en interne une méthode add . La syntaxe de la fonction add prend deux paramètres. Le premier paramètre est self - l'objet chaîne lui-même et le second paramètre est une référence du second objet chaîne. Ceci est montré ci-dessous -

//add function
add(self,&str)->String { 
   // returns a String object
}

Illustration: Concaténation de chaînes

fn main(){
   let n1 = "Tutorials".to_string();
   let n2 = "Point".to_string();

   let n3 = n1 + &n2; // n2 reference is passed
   println!("{}",n3);
}

La sortie sera comme indiqué ci-dessous

TutorialsPoint

Illustration: Moulage de type

L'exemple suivant illustre la conversion d'un nombre en objet chaîne -

fn main(){
   let number = 2020;
   let number_as_string = number.to_string(); 
   
   // convert number to string
   println!("{}",number_as_string);
   println!("{}",number_as_string=="2020");
}

La sortie sera comme indiqué ci-dessous

2020
true

Illustration: Format! Macro

Une autre façon d'ajouter des objets String ensemble consiste à utiliser une fonction macro appelée format. L'utilisation de Format! est comme indiqué ci-dessous.

fn main(){
   let n1 = "Tutorials".to_string();
   let n2 = "Point".to_string();
   let n3 = format!("{} {}",n1,n2);
   println!("{}",n3);
}

La sortie sera comme indiqué ci-dessous

Tutorials Point

Un opérateur définit une fonction qui sera exécutée sur les données. Les données sur lesquelles les opérateurs travaillent sont appelées opérandes. Considérez l'expression suivante -

7 + 5 = 12

Ici, les valeurs 7, 5 et 12 sont des opérandes, tandis que + et = sont des opérateurs.

Les principaux opérateurs de Rust peuvent être classés comme -

  • Arithmetic
  • Bitwise
  • Comparison
  • Logical
  • Bitwise
  • Conditional

Opérateurs arithmétiques

Supposons que les valeurs des variables a et b soient 10 et 5 respectivement.

Afficher des exemples

Sr.Non Opérateur La description Exemple
1 + (Ajout) renvoie la somme des opérandes a + b vaut 15
2 -(Soustraction) renvoie la différence des valeurs ab est 5
3 * (Multiplication) renvoie le produit des valeurs a * b vaut 50
4 / (Division) effectue une opération de division et renvoie le quotient a / b vaut 2
5 % (Module) effectue l'opération de division et renvoie le reste a% b vaut 0

NOTE - Les opérateurs ++ et - ne sont pas pris en charge dans Rust.

Opérateurs relationnels

Les opérateurs relationnels testent ou définissent le type de relation entre deux entités. Les opérateurs relationnels sont utilisés pour comparer deux valeurs ou plus. Les opérateurs relationnels renvoient une valeur booléenne - true ou false.

Supposons que la valeur de A est 10 et B est 20.

Afficher des exemples

Sr.Non Opérateur La description Exemple
1 > Plus grand que (A> B) est faux
2 < Inférieur à (A <B) est vrai
3 > = Plus grand ou égal à (A> = B) est faux
4 <= Inférieur ou égal à (A <= B) est vrai
5 == Égalité (A == B) est faux
6 ! = Inégal (A! = B) est vrai

Opérateurs logiques

Les opérateurs logiques sont utilisés pour combiner deux ou plusieurs conditions. Les opérateurs logiques renvoient également une valeur booléenne. Supposons que la valeur de la variable A est 10 et B est 20.

Afficher des exemples

Sr.Non Opérateur La description Exemple
1 && (Et) L'opérateur renvoie true uniquement si toutes les expressions spécifiées retournent true (A> 10 && B> 10) est faux
2 || (OU) L'opérateur renvoie true si au moins une des expressions spécifiées renvoie true (A> 10 || B> 10) est vrai
3 ! (NE PAS) L'opérateur renvoie l'inverse du résultat de l'expression. Par exemple:! (> 5) renvoie faux ! (A> 10) est vrai

Opérateurs au niveau du bit

Supposons que la variable A = 2 et B = 3.

Afficher des exemples

Sr.Non Opérateur La description Exemple
1 & (ET au niveau du bit) Il effectue une opération booléenne AND sur chaque bit de ses arguments entiers. (A & B) est égal à 2
2 | (BitWise OU) Il effectue une opération booléenne OU sur chaque bit de ses arguments entiers. (A | B) vaut 3
3 ^ (XOR au niveau du bit) Il effectue une opération OU exclusive booléenne sur chaque bit de ses arguments entiers. OU exclusif signifie que l'opérande un est vrai ou l'opérande deux est vrai, mais pas les deux. (A ^ B) vaut 1
4 ! (Pas au niveau du bit) C'est un opérateur unaire et fonctionne en inversant tous les bits de l'opérande. (! B) vaut -4
5 << (Maj gauche) Il déplace tous les bits de son premier opérande vers la gauche du nombre de places spécifié dans le deuxième opérande. Les nouveaux bits sont remplis de zéros. Décaler une valeur vers la gauche d'une position équivaut à la multiplier par 2, déplacer deux positions équivaut à multiplier par 4, et ainsi de suite. (A << 1) vaut 4
6 >> (décalage vers la droite) Opérateur de décalage binaire vers la droite. La valeur de l'opérande gauche est déplacée vers la droite du nombre de bits spécifié par l'opérande droit. (A >> 1) vaut 1
sept >>> (décalage à droite avec zéro) Cet opérateur est exactement comme l'opérateur >>, sauf que les bits décalés vers la gauche sont toujours nuls. (A >>> 1) est 1

Les structures de prise de décision exigent que le programmeur spécifie une ou plusieurs conditions à évaluer ou tester par le programme, ainsi qu'une ou plusieurs instructions à exécuter si la condition est jugée vraie, et éventuellement d'autres instructions à exécuter si le condition est considérée comme fausse.

Vous trouverez ci-dessous la forme générale d'une structure de prise de décision typique trouvée dans la plupart des langages de programmation -

Sr.Non Déclaration et description
1

if statement

Une instruction if consiste en une expression booléenne suivie d'une ou plusieurs instructions.

2

if...else statement

Une instruction if peut être suivie d'une instruction else facultative , qui s'exécute lorsque l'expression booléenne est fausse.

3

else...if and nested ifstatement

Vous pouvez utiliser une instruction if ou else if dans une autre instruction if ou else if .

4

match statement

Une instruction de correspondance permet à une variable d'être testée par rapport à une liste de valeurs.

Si déclaration

La construction if… else évalue une condition avant qu'un bloc de code ne soit exécuté.

Syntaxe

if boolean_expression {
   // statement(s) will execute if the boolean expression is true
}

Si l'expression booléenne a la valeur true, le bloc de code à l'intérieur de l'instruction if sera exécuté. Si l'expression booléenne est évaluée à false, le premier ensemble de code après la fin de l'instruction if (après l'accolade fermante) sera exécuté.

fn main(){
   let num:i32 = 5;
   if num > 0 {
      println!("number is positive") ;
   }
}

L'exemple ci-dessus s'imprimera number is positive car la condition spécifiée par le bloc if est vraie.

instruction if else

Un if peut être suivi d'une option elsebloquer. Le bloc else s'exécutera si l'expression booléenne testée par l'instruction if est évaluée à false.

Syntaxe

if boolean_expression {
   // statement(s) will execute if the boolean expression is true
} else {
   // statement(s) will execute if the boolean expression is false
}

Organigramme

le ifblock protège l'expression conditionnelle. Le bloc associé à l'instruction if est exécuté si l'expression booléenne prend la valeur true.

Le bloc if peut être suivi d'une instruction else facultative. Le bloc d'instructions associé au bloc else est exécuté si l'expression est évaluée à false.

Illustration - Simple si… sinon

fn main() {
   let num = 12;
   if num % 2==0 {
      println!("Even");
   } else {
      println!("Odd");
   }
}

L'exemple ci-dessus indique si la valeur d'une variable est paire ou impaire. Le bloc if vérifie la divisibilité de la valeur par 2 pour déterminer la même chose. Voici la sortie du code ci-dessus -

Even

Imbriqué si

le else…ifladder est utile pour tester plusieurs conditions. La syntaxe est la suivante -

Syntaxe

if boolean_expression1 {
   //statements if the expression1 evaluates to true
} else if boolean_expression2 {
   //statements if the expression2 evaluates to true
} else {
   //statements if both expression1 and expression2 result to false
}

Lorsque vous utilisez des déclarations if… else… if et else, il y a quelques points à garder à l'esprit.

  • Un if peut avoir zéro ou un autre et il doit venir après tout autre..if.
  • Un if peut avoir zéro à beaucoup d'autre..if et ils doivent venir avant l'autre.
  • Une fois qu'un else..if réussit, aucun des autres..if ou else ne sera testé.

Exemple: else… si échelle

fn main() {
   let num = 2 ;
   if num > 0 {
      println!("{} is positive",num);
   } else if num < 0 {
      println!("{} is negative",num);
   } else {
      println!("{} is neither positive nor negative",num) ;
   }
}

L'extrait de code indique si la valeur est positive, négative ou nulle.

Production

2 is positive

Déclaration de correspondance

L'instruction match vérifie si une valeur actuelle correspond à partir d'une liste de valeurs, ceci est très similaire à l'instruction switch en langage C. En premier lieu, notez que l'expression qui suit le mot-clé match n'a pas à être placée entre parenthèses.

La syntaxe est la suivante.

let expressionResult = match variable_expression {
   constant_expr1 => {
      //statements;
   },
   constant_expr2 => {
      //statements;
   },
   _ => {
      //default
   }
};

Dans l'exemple ci-dessous, state_code correspond à une liste de valeurs MH, KL, KA, GA- si une correspondance est trouvée, une valeur de chaîne est renvoyée à l' état variable . Si aucune correspondance n'est trouvée, le cas par défaut _ correspond et la valeur Unkown est renvoyée.

fn main(){
   let state_code = "MH";
   let state = match state_code {
      "MH" => {println!("Found match for MH"); "Maharashtra"},
      "KL" => "Kerala",
      "KA" => "Karnadaka",
      "GA" => "Goa",
      _ => "Unknown"
   };
   println!("State name is {}",state);
}

Production

Found match for MH
State name is Maharashtra

Il peut y avoir des cas où un bloc de code doit être exécuté à plusieurs reprises. En général, les instructions de programmation sont exécutées séquentiellement: la première instruction d'une fonction est exécutée en premier, suivie de la seconde, et ainsi de suite.

Les langages de programmation fournissent diverses structures de contrôle qui permettent des chemins d'exécution plus compliqués.

Une instruction de boucle nous permet d'exécuter une instruction ou un groupe d'instructions plusieurs fois. Vous trouverez ci-dessous la forme générale d'une instruction de boucle dans la plupart des langages de programmation.

Rust fournit différents types de boucles pour gérer les exigences de bouclage -

  • while
  • loop
  • for

Boucle définie

Une boucle dont le nombre d'itérations est défini / fixe est appelée boucle définie. lefor loop est une implémentation d'une boucle définie.

Pour la boucle

La boucle for exécute le bloc de code un nombre de fois spécifié. Il peut être utilisé pour parcourir un ensemble fixe de valeurs, tel qu'un tableau. La syntaxe de la boucle for est la suivante

Syntaxe

for temp_variable in lower_bound..upper_bound {
   //statements
}

Un exemple de boucle for est illustré ci-dessous

fn main(){
   for x in 1..11{ // 11 is not inclusive
      if x==5 {
         continue;
      }
      println!("x is {}",x);
   }
}

NOTE: que la variable x n'est accessible que dans le bloc for.

Production

x is 1
x is 2
x is 3
x is 4
x is 6
x is 7
x is 8
x is 9
x is 10

Boucle indéfinie

Une boucle indéfinie est utilisée lorsque le nombre d'itérations dans une boucle est indéterminé ou inconnu.

Des boucles indéfinies peuvent être implémentées en utilisant -

Sr.Non Nom et description
1

While

Le temps boucle exécute les instructions à chaque fois que la condition spécifiée est évaluée à true

2

Loop

La boucle est une boucle indéfinie while (vraie)

Illustration - pendant quelque temps

fn main(){
   let mut x = 0;
   while x < 10{
      x+=1;
      println!("inside loop x value is {}",x);
   }
   println!("outside loop x value is {}",x);
}

La sortie est comme indiqué ci-dessous -

inside loop x value is 1
inside loop x value is 2
inside loop x value is 3
inside loop x value is 4
inside loop x value is 5
inside loop x value is 6
inside loop x value is 7
inside loop x value is 8
inside loop x value is 9
inside loop x value is 10
outside loop x value is 10

Illustration −loop

fn main(){
   //while true

   let mut x = 0;
   loop {
      x+=1;
      println!("x={}",x);

      if x==15 {
         break;
      }
   }
}

le breakL'instruction est utilisée pour retirer le contrôle d'une construction. L'utilisation de break in a loop provoque la sortie du programme de la boucle.

Production

x=1
x=2
x=3
x=4
x=5
x=6
x=7
x=8
x=9
x=10
x=11
x=12
x=13
x=14
x=15

Continuer la déclaration

L'instruction continue ignore les instructions suivantes dans l'itération actuelle et ramène le contrôle au début de la boucle. Contrairement à l'instruction break, le continue ne quitte pas la boucle. Il met fin à l'itération en cours et démarre l'itération suivante.

Un exemple de l'instruction continue est donné ci-dessous.

fn main() {

   let mut count = 0;

   for num in 0..21 {
      if num % 2==0 {
         continue;
      }
      count+=1;
   }
   println! (" The count of odd values between 0 and 20 is: {} ",count);
   //outputs 10
}

L'exemple ci-dessus affiche le nombre de valeurs paires entre 0 et 20. La boucle quitte l'itération actuelle si le nombre est pair. Ceci est réalisé en utilisant l'instruction continue.

Le nombre de valeurs impaires entre 0 et 20 est de 10

Les fonctions sont les éléments constitutifs d'un code lisible, maintenable et réutilisable. Une fonction est un ensemble d'instructions pour effectuer une tâche spécifique. Les fonctions organisent le programme en blocs logiques de code. Une fois définies, les fonctions peuvent être appelées pour accéder au code. Cela rend le code réutilisable. De plus, les fonctions facilitent la lecture et la maintenance du code du programme.

Une déclaration de fonction informe le compilateur du nom, du type de retour et des paramètres d'une fonction. Une définition de fonction fournit le corps réel de la fonction.

Sr.Non Description de la fonction
1

Defining a function

La définition de la fonction TA spécifie quoi et comment une tâche spécifique serait effectuée.

2

Calling or invoking a Function

Une fonction doit être appelée pour l'exécuter.

3

Returning Functions

Les fonctions peuvent également renvoyer une valeur avec le contrôle, à l'appelant.

4

Parameterized Function

Les paramètres sont un mécanisme permettant de transmettre des valeurs aux fonctions.

Définition d'une fonction

Une définition de fonction spécifie quoi et comment une tâche spécifique serait effectuée. Avant d'utiliser une fonction, elle doit être définie. Le corps de la fonction contient du code qui doit être exécuté par la fonction. Les règles de dénomination d'une fonction sont similaires à celles d'une variable. Les fonctions sont définies à l'aide dufnmot-clé. La syntaxe pour définir une fonction standard est donnée ci-dessous

Syntaxe

fn function_name(param1,param2..paramN) {
   // function body
}

Une déclaration de fonction peut éventuellement contenir des paramètres / arguments. Les paramètres sont utilisés pour transmettre des valeurs aux fonctions.

Exemple - Définition de fonction simple

//Defining a function
fn fn_hello(){
   println!("hello from function fn_hello ");
}

Appel d'une fonction

Une fonction doit être appelée pour l'exécuter. Ce processus est appeléfunction invocation. Les valeurs des paramètres doivent être transmises lorsqu'une fonction est appelée. La fonction qui appelle une autre fonction est appeléecaller function.

Syntaxe

function_name(val1,val2,valN)

Exemple: appeler une fonction

fn main(){
   //calling a function
   fn_hello();
}

Ici, le main () est la fonction de l'appelant.

Illustration

L'exemple suivant définit une fonction fn_hello(). La fonction imprime un message sur la console. lemain()La fonction invoque la fonction fn_hello () .

fn main(){
   //calling a function
   fn_hello();
}
//Defining a function
fn fn_hello(){
   println!("hello from function fn_hello ");
}

Production

hello from function fn_hello

Renvoyer la valeur d'une fonction

Les fonctions peuvent également renvoyer une valeur avec le contrôle, à l'appelant. Ces fonctions sont appelées fonctions de retour.

Syntaxe

L'une ou l'autre des syntaxes suivantes peut être utilisée pour définir une fonction avec un type de retour.

Avec déclaration de retour

// Syntax1
fn function_name() -> return_type {
   //statements
   return value;
}

Syntaxe abrégée sans instruction de retour

//Syntax2
fn function_name() -> return_type {
   value //no semicolon means this value is returned
}

lllustration

fn main(){
   println!("pi value is {}",get_pi());
}
fn get_pi()->f64 {
   22.0/7.0
}

Production

pi value is 3.142857142857143

Fonction avec paramètres

Les paramètres sont un mécanisme permettant de transmettre des valeurs aux fonctions. Les paramètres font partie de la signature de la fonction. Les valeurs des paramètres sont transmises à la fonction lors de son appel. Sauf indication contraire explicite, le nombre de valeurs passées à une fonction doit correspondre au nombre de paramètres définis.

Les paramètres peuvent être passés à une fonction en utilisant l'une des techniques suivantes -

Passer par valeur

Lorsqu'une méthode est appelée, un nouvel emplacement de stockage est créé pour chaque paramètre de valeur. Les valeurs des paramètres réels y sont copiées. Par conséquent, les modifications apportées au paramètre dans la méthode invoquée n'ont aucun effet sur l'argument.

L'exemple suivant déclare une variable no, qui vaut initialement 5. La variable est passée en paramètre (par valeur) au mutate_no_to_zero()function, qui change la valeur à zéro. Après l'appel de la fonction, lorsque le contrôle revient à la méthode principale, la valeur sera la même.

fn main(){
   let no:i32 = 5;
   mutate_no_to_zero(no);
   println!("The value of no is:{}",no);
}

fn mutate_no_to_zero(mut param_no: i32) {
   param_no = param_no*0;
   println!("param_no value is :{}",param_no);
}

Production

param_no value is :0
The value of no is:5

Passer par référence

Lorsque vous transmettez des paramètres par référence, contrairement aux paramètres de valeur, aucun nouvel emplacement de stockage n'est créé pour ces paramètres. Les paramètres de référence représentent le même emplacement mémoire que les paramètres réels fournis à la méthode. Les valeurs de paramètre peuvent être passées par référence en préfixant le nom de la variable avec un& .

Dans l'exemple donné ci-dessous, nous avons une variable no , qui est initialement 5. Une référence à la variable no est passée aumutate_no_to_zero()fonction. La fonction opère sur la variable d'origine. Après l'appel de fonction, lorsque le contrôle revient à la méthode principale, la valeur de la variable d'origine sera le zéro.

fn main() {
   let mut no:i32 = 5;
   mutate_no_to_zero(&mut no);
   println!("The value of no is:{}",no);
}
fn mutate_no_to_zero(param_no:&mut i32){
   *param_no = 0; //de reference
}

L'opérateur * est utilisé pour accéder à la valeur stockée dans l'emplacement mémoire où la variable param_nopointe vers. Ceci est également connu sous le nom de déréférencement.

La sortie sera -

The value of no is 0.

Passer une chaîne à une fonction

La fonction main () transmet un objet chaîne à la fonction display () .

fn main(){
   let name:String = String::from("TutorialsPoint");
   display(name); 
   //cannot access name after display
}
fn display(param_name:String){
   println!("param_name value is :{}",param_name);
}

Production

param_name value is :TutorialsPoint

Tuple est un type de données composé. Un type scalaire ne peut stocker qu'un seul type de données. Par exemple, une variable i32 ne peut stocker qu'une seule valeur entière. Dans les types composés, nous pouvons stocker plus d'une valeur à la fois et elle peut être de différents types.

Les tuples ont une longueur fixe - une fois déclarés, ils ne peuvent pas grandir ou rétrécir. L'index de tuple commence à partir de0.

Syntaxe

//Syntax1
let tuple_name:(data_type1,data_type2,data_type3) = (value1,value2,value3);

//Syntax2
let tuple_name = (value1,value2,value3);

Illustration

L'exemple suivant affiche les valeurs dans un tuple.

fn main() {
   let tuple:(i32,f64,u8) = (-325,4.9,22);
   println!("{:?}",tuple);
}

La syntaxe println! ("{}", Tuple) ne peut pas être utilisée pour afficher des valeurs dans un tuple. En effet, un tuple est un type composé. Utilisez la syntaxe println! ("{:?}", tuple_name) pour imprimer les valeurs dans un tuple.

Production

(-325, 4.9, 22)

Illustration

L'exemple suivant imprime des valeurs individuelles dans un tuple.

fn main() {
   let tuple:(i32,f64,u8) = (-325,4.9,22);
   println!("integer is :{:?}",tuple.0);
   println!("float is :{:?}",tuple.1);
   println!("unsigned integer is :{:?}",tuple.2);
}

Production

integer is :-325
float is :4.9
unsigned integer is :2

Illustration

L'exemple suivant passe un tuple comme paramètre à une fonction. Les tuples sont passés par valeur aux fonctions.

fn main(){
   let b:(i32,bool,f64) = (110,true,10.9);
   print(b);
}
//pass the tuple as a parameter

fn print(x:(i32,bool,f64)){
   println!("Inside print method");
   println!("{:?}",x);
}

Production

Inside print method
(110, true, 10.9)

Destruction

L'affectation de destruction est une fonctionnalité de rust dans laquelle nous décompressons les valeurs d'un tuple. Ceci est réalisé en attribuant un tuple à des variables distinctes.

Prenons l'exemple suivant -

fn main(){
   let b:(i32,bool,f64) = (30,true,7.9);
   print(b);
}
fn print(x:(i32,bool,f64)){
   println!("Inside print method");
   let (age,is_male,cgpa) = x; //assigns a tuple to 
   distinct variables
   println!("Age is {} , isMale? {},cgpa is 
   {}",age,is_male,cgpa);
}

La variable x est un tuple qui est assigné à l'instruction let. Chaque variable - age, is_male et cgpa contiendra les valeurs correspondantes dans un tuple.

Production

Inside print method
Age is 30 , isMale? true,cgpa is 7.9

Dans ce chapitre, nous allons découvrir un tableau et les différentes fonctionnalités qui lui sont associées. Avant d'en apprendre davantage sur les tableaux, voyons en quoi un tableau est différent d'une variable.

Les variables ont les limitations suivantes -

  • Les variables sont de nature scalaire. En d'autres termes, une déclaration de variable ne peut contenir qu'une seule valeur à la fois. Cela signifie que pour stocker n valeurs dans un programme, n déclaration de variable sera nécessaire. Par conséquent, l'utilisation de variables n'est pas possible lorsque l'on a besoin de stocker une plus grande collection de valeurs.

  • Les variables d'un programme se voient allouer de la mémoire dans un ordre aléatoire, ce qui rend difficile la récupération / lecture des valeurs dans l'ordre de leur déclaration.

Un tableau est une collection homogène de valeurs. En termes simples, un tableau est une collection de valeurs du même type de données.

Caractéristiques d'une baie

Les caractéristiques d'un tableau sont énumérées ci-dessous -

  • Une déclaration de tableau alloue des blocs de mémoire séquentiels.

  • Les tableaux sont statiques. Cela signifie qu'un tableau une fois initialisé ne peut pas être redimensionné.

  • Chaque bloc de mémoire représente un élément de tableau.

  • Les éléments du tableau sont identifiés par un entier unique appelé indice / index de l'élément.

  • Le remplissage des éléments du tableau est appelé initialisation du tableau.

  • Les valeurs des éléments de tableau peuvent être mises à jour ou modifiées mais ne peuvent pas être supprimées.

Déclaration et initialisation de tableaux

Utilisez la syntaxe ci-dessous pour déclarer et initialiser un tableau dans Rust.

Syntaxe

//Syntax1
let variable_name = [value1,value2,value3];

//Syntax2
let variable_name:[dataType;size] = [value1,value2,value3];

//Syntax3
let variable_name:[dataType;size] = [default_value_for_elements,size];

Dans la première syntaxe, le type du tableau est déduit du type de données du premier élément du tableau lors de l'initialisation.

Illustration: matrice simple

L'exemple suivant spécifie explicitement la taille et le type de données du tableau. La syntaxe {:?} De la fonction println! () Est utilisée pour imprimer toutes les valeurs du tableau. La fonction len () est utilisée pour calculer la taille du tableau.

fn main(){
   let arr:[i32;4] = [10,20,30,40];
   println!("array is {:?}",arr);
   println!("array size is :{}",arr.len());
}

Production

array is [10, 20, 30, 40]
array size is :4

Illustration: matrice sans type de données

Le programme suivant déclare un tableau de 4 éléments. Le type de données n'est pas explicitement spécifié lors de la déclaration de variable. Dans ce cas, le tableau sera de type entier. La fonction len () est utilisée pour calculer la taille du tableau.

fn main(){
   let arr = [10,20,30,40];
   println!("array is {:?}",arr);
   println!("array size is :{}",arr.len());
}

Production

array is [10, 20, 30, 40]
array size is :4

Illustration: valeurs par défaut

L'exemple suivant crée un tableau et initialise tous ses éléments avec une valeur par défaut de -1 .

fn main() {
   let arr:[i32;4] = [-1;4];
   println!("array is {:?}",arr);
   println!("array size is :{}",arr.len());
}

Production

array is [-1, -1, -1, -1]
array size is :4

Illustration: tableau avec boucle for

L'exemple suivant parcourt un tableau et imprime les index et leurs valeurs correspondantes. La boucle récupère les valeurs de l'index 0 à 4 (index du dernier élément du tableau).

fn main(){
   let arr:[i32;4] = [10,20,30,40];
   println!("array is {:?}",arr);
   println!("array size is :{}",arr.len());

   for index in 0..4 {
      println!("index is: {} & value is : {}",index,arr[index]);
   }
}

Production

array is [10, 20, 30, 40]
array size is :4
index is: 0 & value is : 10
index is: 1 & value is : 20
index is: 2 & value is : 30
index is: 3 & value is : 40

Illustration: utilisation de la fonction iter ()

La fonction iter () récupère les valeurs de tous les éléments d'un tableau.

fn main(){

let arr:[i32;4] = [10,20,30,40];
   println!("array is {:?}",arr);
   println!("array size is :{}",arr.len());

   for val in arr.iter(){
      println!("value is :{}",val);
   }
}

Production

array is [10, 20, 30, 40]
array size is :4
value is :10
value is :20
value is :30
value is :40

Illustration: tableau Mutable

Le mot clé mut peut être utilisé pour déclarer un tableau mutable. L'exemple suivant déclare un tableau mutable et modifie la valeur du deuxième élément du tableau.

fn main(){
   let mut arr:[i32;4] = [10,20,30,40];
   arr[1] = 0;
   println!("{:?}",arr);
}

Production

[10, 0, 30, 40]

Passage de tableaux en tant que paramètres à des fonctions

Un tableau peut être passé par valeur ou par référence à des fonctions.

Illustration: Passer par valeur

fn main() {
   let arr = [10,20,30];
   update(arr);

   print!("Inside main {:?}",arr);
}
fn update(mut arr:[i32;3]){
   for i in 0..3 {
      arr[i] = 0;
   }
   println!("Inside update {:?}",arr);
}

Production

Inside update [0, 0, 0]
Inside main [10, 20, 30]

Illustration: Passez par référence

fn main() {
   let mut arr = [10,20,30];
   update(&mut arr);
   print!("Inside main {:?}",arr);
}
fn update(arr:&mut [i32;3]){
   for i in 0..3 {
      arr[i] = 0;
   }
   println!("Inside update {:?}",arr);
}

Production

Inside update [0, 0, 0]
Inside main [0, 0, 0]

Déclaration de tableau et constantes

Prenons l'exemple ci-dessous pour comprendre la déclaration et les constantes des tableaux.

fn main() {
   let N: usize = 20;
   let arr = [0; N]; //Error: non-constant used with constant
   print!("{}",arr[10])
}

Le compilateur entraînera une exception. En effet, la longueur d'un tableau doit être connue au moment de la compilation. Ici, la valeur de la variable "N" sera déterminée lors de l'exécution. En d'autres termes, les variables ne peuvent pas être utilisées pour définir la taille d'un tableau.

Cependant, le programme suivant est valide -

fn main() {
   const N: usize = 20; 
   // pointer sized
   let arr = [0; N];

   print!("{}",arr[10])
}

La valeur d'un identifiant précédé du mot clé const est définie au moment de la compilation et ne peut pas être modifiée à l'exécution. usize est de la taille d'un pointeur, donc sa taille réelle dépend de l'architecture pour laquelle vous compilez votre programme.

La mémoire d'un programme peut être allouée de la manière suivante -

  • Stack
  • Heap

Empiler

Une pile suit un dernier dans le premier ordre sorti. Stack stocke les valeurs de données dont la taille est connue au moment de la compilation. Par exemple, une variable de taille fixe i32 est candidate à l'allocation de pile. Sa taille est connue au moment de la compilation. Tous les types scalaires peuvent être stockés dans la pile car la taille est fixe.

Prenons un exemple de chaîne à laquelle une valeur est affectée au moment de l'exécution. La taille exacte d'une telle chaîne ne peut pas être déterminée au moment de la compilation. Ce n'est donc pas un candidat pour l'allocation de pile mais pour l'allocation de tas.

Tas

La mémoire du tas stocke des valeurs de données dont la taille est inconnue au moment de la compilation. Il est utilisé pour stocker des données dynamiques. En termes simples, une mémoire de tas est allouée à des valeurs de données qui peuvent changer tout au long du cycle de vie du programme. Le tas est une zone de la mémoire qui est moins organisée que la pile.

Qu'est-ce que la propriété?

Chaque valeur de Rust a une variable appelée ownerde la valeur. Chaque donnée stockée dans Rust sera associée à un propriétaire. Par exemple, dans la syntaxe - let age = 30, age est le propriétaire de la valeur 30 .

  • Chaque donnée ne peut avoir qu'un seul propriétaire à la fois.

  • Deux variables ne peuvent pas pointer vers le même emplacement mémoire. Les variables pointeront toujours vers différents emplacements de mémoire.

Transfert de propriété

La propriété de la valeur peut être transférée par -

  • Attribuer la valeur d'une variable à une autre variable.

  • Passer de la valeur à une fonction.

  • Valeur renvoyée par une fonction.

Attribution de la valeur d'une variable à une autre variable

Le principal argument de vente de Rust en tant que langage est la sécurité de sa mémoire. La sécurité de la mémoire est assurée par un contrôle strict sur qui peut utiliser quoi et quand les restrictions.

Considérez l'extrait suivant -

fn main(){
   let v = vec![1,2,3]; 
   // vector v owns the object in heap

   //only a single variable owns the heap memory at any given time
   let v2 = v; 
   // here two variables owns heap value,
   //two pointers to the same content is not allowed in rust

   //Rust is very smart in terms of memory access ,so it detects a race condition
   //as two variables point to same heap

   println!("{:?}",v);
}

L'exemple ci-dessus déclare un vecteur v. L'idée de propriété est qu'une seule variable se lie à une ressource, soit v se lie à la ressource ou v2se lie à la ressource. L'exemple ci-dessus génère une erreur - utilisation de la valeur déplacée: `v` . En effet, la propriété de la ressource est transférée vers la v2. Cela signifie que la propriété est déplacée de v à v2 (v2 = v) et v est invalidée après le déplacement.

Passer de la valeur à une fonction

La propriété d'une valeur change également lorsque nous transmettons un objet du tas à une fermeture ou à une fonction.

fn main(){
   let v = vec![1,2,3];     // vector v owns the object in heap
   let v2 = v;              // moves ownership to v2
   display(v2);             // v2 is moved to display and v2 is invalidated
   println!("In main {:?}",v2);    //v2 is No longer usable here
}
fn display(v:Vec<i32>){
   println!("inside display {:?}",v);
}

Renvoyer la valeur d'une fonction

La propriété transmise à la fonction sera invalidée à la fin de l'exécution de la fonction. Une solution pour cela consiste à laisser la fonction renvoyer l'objet possédé à l'appelant.

fn main(){
   let v = vec![1,2,3];       // vector v owns the object in heap
   let v2 = v;                // moves ownership to v2
   let v2_return = display(v2);    
   println!("In main {:?}",v2_return);
}
fn display(v:Vec<i32>)->Vec<i32> { 
   // returning same vector
   println!("inside display {:?}",v);
}

Propriété et types primitifs

Dans le cas des types primitifs, le contenu d'une variable est copié dans une autre. Il n'y a donc pas de transfert de propriété. C'est parce qu'une variable primitive a besoin de moins de ressources qu'un objet. Prenons l'exemple suivant -

fn main(){
   let u1 = 10;
   let u2 = u1;  // u1 value copied(not moved) to u2

   println!("u1 = {}",u1);
}

La sortie sera - 10.

Il est très peu pratique de transmettre la propriété d'une variable à une autre fonction, puis de restituer la propriété. Rust prend en charge un concept, l'emprunt, dans lequel la propriété d'une valeur est transférée temporairement à une entité, puis renvoyée à l'entité propriétaire d'origine.

Considérez ce qui suit -

fn main(){
   // a list of nos
   let v = vec![10,20,30];
   print_vector(v);
   println!("{}",v[0]); // this line gives error
}
fn print_vector(x:Vec<i32>){
   println!("Inside print_vector function {:?}",x);
}

La fonction main invoque une fonction print_vector () . Un vecteur est passé en paramètre à cette fonction. La propriété du vecteur est également transmise à la fonction print_vector () à partir de main () . Le code ci-dessus entraînera une erreur comme indiqué ci-dessous lorsque la fonction main () essaiera d'accéder au vecteur v .

|  print_vector(v);
|     - value moved here
|  println!("{}",v[0]);
|     ^ value used here after move

En effet, une variable ou une valeur ne peut plus être utilisée par la fonction qui la possédait à l'origine une fois que la propriété est transférée à une autre fonction.

Qu'est-ce que l'emprunt?

Lorsqu'une fonction transfère temporairement son contrôle sur une variable / valeur à une autre fonction, pendant un certain temps, elle est appelée emprunt. Ceci est réalisé en passant une référence à la variable(& var_name)plutôt que de passer la variable / valeur elle-même à la fonction. La propriété de la variable / valeur est transférée au propriétaire d'origine de la variable une fois que la fonction à laquelle le contrôle a été passé a terminé l'exécution.

fn main(){
   // a list of nos
   let v = vec![10,20,30];
   print_vector(&v); // passing reference
   println!("Printing the value from main() v[0]={}",v[0]);
}
fn print_vector(x:&Vec<i32>){
   println!("Inside print_vector function {:?}",x);
}

Production

Inside print_vector function [10, 20, 30]
Printing the value from main() v[0] = 10

Références mutables

Une fonction peut modifier une ressource empruntée en utilisant une référence mutable à une telle ressource. Une référence mutable est précédée du préfixe&mut. Les références mutables ne peuvent fonctionner que sur des variables mutables.

Illustration: Mutation d'une référence entière

fn add_one(e: &mut i32) {
   *e+= 1;
}
fn main() {
   let mut i = 3;
   add_one(&mut i);
   println!("{}", i);
}

La fonction main () déclare une variable entière mutable i et transmet une référence mutable de i auadd_one(). La fonction add_one () incrémente la valeur de la variable i de un.

Illustration: muter une référence de chaîne

fn main() {
   let mut name:String = String::from("TutorialsPoint");
   display(&mut name); 
   //pass a mutable reference of name
   println!("The value of name after modification is:{}",name);
}
fn display(param_name:&mut String){
   println!("param_name value is :{}",param_name);
   param_name.push_str(" Rocks"); 
   //Modify the actual string,name
}

La fonction main () transmet une référence mutable du nom de la variable à la fonction display () . La fonction d'affichage ajoute une chaîne supplémentaire à la variable de nom d' origine .

Production

param_name value is :TutorialsPoint
The value of name after modification is:TutorialsPoint Rocks

Une tranche est un pointeur vers un bloc de mémoire. Les tranches peuvent être utilisées pour accéder à des parties de données stockées dans des blocs de mémoire contigus. Il peut être utilisé avec des structures de données telles que des tableaux, des vecteurs et des chaînes. Les tranches utilisent des numéros d'index pour accéder à des parties de données. La taille d'une tranche est déterminée au moment de l'exécution.

Les tranches sont des pointeurs vers les données réelles. Ils sont transmis par référence à des fonctions, également appelées emprunt.

Par exemple, les tranches peuvent être utilisées pour récupérer une partie d'une valeur de chaîne. Une chaîne tranchée est un pointeur vers l'objet chaîne réel. Par conséquent, nous devons spécifier l'index de début et de fin d'une chaîne. L'index commence à 0 comme les tableaux.

Syntaxe

let sliced_value = &data_structure[start_index..end_index]

La valeur d'index minimale est 0 et la valeur d'index maximale est la taille de la structure de données. Notez que end_index ne sera pas inclus dans la chaîne finale.

Le diagramme ci-dessous montre un exemple de didacticiels de chaîne , qui comporte 9 caractères. L'index du premier caractère est 0 et celui du dernier caractère est 8.

Le code suivant récupère 5 caractères de la chaîne (à partir de l'index 4).

fn main() {
   let n1 = "Tutorials".to_string();
   println!("length of string is {}",n1.len());
   let c1 = &n1[4..9]; 
   
   // fetches characters at 4,5,6,7, and 8 indexes
   println!("{}",c1);
}

Production

length of string is 9
rials

Illustration - Découpage d'un tableau d'entiers

La fonction main () déclare un tableau avec 5 éléments. Il invoque leuse_slice()et lui transmet une tranche de trois éléments (pointe vers le tableau de données). Les tranches sont passées par référence. La fonction use_slice () imprime la valeur de la tranche et sa longueur.

fn main(){
   let data = [10,20,30,40,50];
   use_slice(&data[1..4]);
   //this is effectively borrowing elements for a while
}
fn use_slice(slice:&[i32]) { 
   // is taking a slice or borrowing a part of an array of i32s
   println!("length of slice is {:?}",slice.len());
   println!("{:?}",slice);
}

Production

length of slice is 3
[20, 30, 40]

Tranches mutables

le &mut Le mot-clé peut être utilisé pour marquer une tranche comme mutable.

fn main(){
   let mut data = [10,20,30,40,50];
   use_slice(&mut data[1..4]);
   // passes references of 
   20, 30 and 40
   println!("{:?}",data);
}
fn use_slice(slice:&mut [i32]) {
   println!("length of slice is {:?}",slice.len());
   println!("{:?}",slice);
   slice[0] = 1010; // replaces 20 with 1010
}

Production

length of slice is 3
[20, 30, 40]
[10, 1010, 30, 40, 50]

Le code ci-dessus passe une tranche mutable à la fonction use_slice () . La fonction modifie le deuxième élément du tableau d'origine.

Les tableaux sont utilisés pour représenter une collection homogène de valeurs. De même, une structure est un autre type de données défini par l'utilisateur disponible dans Rust qui nous permet de combiner des éléments de données de différents types, y compris une autre structure. Une structure définit les données comme une paire clé-valeur.

Syntaxe - Déclaration d'une structure

Le mot - clé struct est utilisé pour déclarer une structure. Les structures étant typées statiquement, chaque champ de la structure doit être associé à un type de données. Les règles et conventions de dénomination d'une structure sont similaires à celles d'une variable. Le bloc de structure doit se terminer par un point-virgule.

struct Name_of_structure {
   field1:data_type,
   field2:data_type,
   field3:data_type
}

Syntaxe - Initialisation d'une structure

Après avoir déclaré une structure, chaque champ doit recevoir une valeur. C'est ce qu'on appelle l'initialisation.

let instance_name = Name_of_structure {
   field1:value1,
   field2:value2,
   field3:value3
}; 
//NOTE the semicolon
Syntax: Accessing values in a structure
Use the dot notation to access value of a specific field.
instance_name.field1
Illustration
struct Employee {
   name:String,
   company:String,
   age:u32
}
fn main() {
   let emp1 = Employee {
      company:String::from("TutorialsPoint"),
      name:String::from("Mohtashim"),
      age:50
   };
   println!("Name is :{} company is {} age is {}",emp1.name,emp1.company,emp1.age);
}

L'exemple ci-dessus déclare une structure Employee avec trois champs - nom, société et âge des types. Le main () initialise la structure. Il utilise le println! macro pour imprimer les valeurs des champs définis dans la structure.

Production

Name is :Mohtashim company is TutorialsPoint age is 50

Modifier une instance de struct

Pour modifier une instance, la variable d'instance doit être marquée mutable. L'exemple ci-dessous déclare et initialise une structure nommée Employee et modifie ultérieurement la valeur du champ age à 40 de 50.

let mut emp1 = Employee {
   company:String::from("TutorialsPoint"),
   name:String::from("Mohtashim"),
   age:50
};
emp1.age = 40;
println!("Name is :{} company is {} age is 
{}",emp1.name,emp1.company,emp1.age);

Production

Name is :Mohtashim company is TutorialsPoint age is 40

Passer une structure à une fonction

L'exemple suivant montre comment passer une instance de struct en tant que paramètre. La méthode d'affichage prend une instance Employee comme paramètre et imprime les détails.

fn display( emp:Employee) {
   println!("Name is :{} company is {} age is 
   {}",emp.name,emp.company,emp.age);
}

Voici le programme complet -

//declare a structure
struct Employee {
   name:String,
   company:String,
   age:u32
}
fn main() {
   //initialize a structure
   let emp1 = Employee {
      company:String::from("TutorialsPoint"),
      name:String::from("Mohtashim"),
      age:50
   };
   let emp2 = Employee{
      company:String::from("TutorialsPoint"),
      name:String::from("Kannan"),
      age:32
   };
   //pass emp1 and emp2 to display()
   display(emp1);
   display(emp2);
}
// fetch values of specific structure fields using the 
// operator and print it to the console
fn display( emp:Employee){
   println!("Name is :{} company is {} age is 
   {}",emp.name,emp.company,emp.age);
}

Production

Name is :Mohtashim company is TutorialsPoint age is 50
Name is :Kannan company is TutorialsPoint age is 32

Renvoyer une structure depuis une fonction

Considérons une fonction who_is_elder () , qui compare l'âge de deux employés et renvoie le plus âgé.

fn who_is_elder (emp1:Employee,emp2:Employee)->Employee {
   if emp1.age>emp2.age {
      return emp1;
   } else {
      return emp2;
   }
}

Voici le programme complet -

fn main() {
   //initialize structure
   let emp1 = Employee{
      company:String::from("TutorialsPoint"),
      name:String::from("Mohtashim"),
      age:50
   };
   let emp2 = Employee {
      company:String::from("TutorialsPoint"),
      name:String::from("Kannan"),
      age:32
   };
   let elder = who_is_elder(emp1,emp2);
   println!("elder is:");

   //prints details of the elder employee
   display(elder);
}
//accepts instances of employee structure and compares their age
fn who_is_elder (emp1:Employee,emp2:Employee)->Employee {
   if emp1.age>emp2.age {
      return emp1;
   } else {
      return emp2;
   }
}
//display name, comapny and age of the employee
fn display( emp:Employee) {
   println!("Name is :{} company is {} age is {}",emp.name,emp.company,emp.age);
}
//declare a structure
struct Employee {
   name:String,
   company:String,
   age:u32
}

Production

elder is:
Name is :Mohtashim company is TutorialsPoint age is 50

Méthode dans la structure

Les méthodes sont comme des fonctions. Il s'agit d'un groupe logique d'instructions de programmation. Les méthodes sont déclarées avec lefnmot-clé. La portée d'une méthode se trouve dans le bloc de structure.

Les méthodes sont déclarées en dehors du bloc de structure. leimplLe mot clé est utilisé pour définir une méthode dans le contexte d'une structure. Le premier paramètre d'une méthode sera toujoursself, qui représente l'instance appelante de la structure. Les méthodes opèrent sur les données membres d'une structure.

Pour invoquer une méthode, nous devons d'abord instancier la structure. La méthode peut être appelée en utilisant l'instance de la structure.

Syntaxe

struct My_struct {}
impl My_struct { 
   //set the method's context
   fn method_name() { 
      //define a method
   }
}

Illustration

L'exemple suivant définit une structure Rectangle avec des champs - largeur et hauteur . Une zone de méthode est définie dans le contexte de la structure. La méthode area accède aux champs de la structure via le mot-clé self et calcule l'aire d'un rectangle.

//define dimensions of a rectangle
struct Rectangle {
   width:u32, height:u32
}

//logic to calculate area of a rectangle
impl Rectangle {
   fn area(&self)->u32 {
      //use the . operator to fetch the value of a field via the self keyword
      self.width * self.height
   }
}

fn main() {
   // instanatiate the structure
   let small = Rectangle {
      width:10,
      height:20
   };
   //print the rectangle's area
   println!("width is {} height is {} area of Rectangle 
   is {}",small.width,small.height,small.area());
}

Production

width is 10 height is 20 area of Rectangle is 200

Méthode statique dans la structure

Les méthodes statiques peuvent être utilisées comme méthodes utilitaires. Ces méthodes existent avant même que la structure ne soit instanciée. Les méthodes statiques sont appelées à l'aide du nom de la structure et sont accessibles sans instance. Contrairement aux méthodes normales, une méthode statique ne prendra pas le paramètre & self .

Syntaxe - Déclaration d'une méthode statique

Une méthode statique telle que des fonctions et d'autres méthodes peut éventuellement contenir des paramètres.

impl Structure_Name {
   //static method that creates objects of the Point structure
   fn method_name(param1: datatype, param2: datatype) -> return_type {
      // logic goes here
   }
}

Syntaxe - Invocation d'une méthode statique

La syntaxe structure_name :: est utilisée pour accéder à une méthode statique.

structure_name::method_name(v1,v2)

Illustration

L'exemple suivant utilise la méthode getInstance comme classe de fabrique qui crée et retourne des instances de la structure Point .

//declare a structure
struct Point {
   x: i32,
   y: i32,
}
impl Point {
   //static method that creates objects of the Point structure
   fn getInstance(x: i32, y: i32) -> Point {
      Point { x: x, y: y }
   }
   //display values of the structure's field
   fn display(&self){
      println!("x ={} y={}",self.x,self.y );
   }
}
fn main(){
   // Invoke the static method
   let p1 = Point::getInstance(10,20);
   p1.display();
}

Production

x =10 y=20

Dans la programmation Rust, lorsque nous devons sélectionner une valeur dans une liste de variantes possibles, nous utilisons des types de données d'énumération. Un type énuméré est déclaré à l'aide du mot clé enum . Voici la syntaxe de enum -

enum enum_name {
   variant1,
   variant2,
   variant3
}

Illustration: utilisation d'une énumération

L'exemple déclare une énumération - GenderCategory , qui a des variantes comme Homme et Femme. L' impression! La macro affiche la valeur de l'énumération. Le compilateur lancera une erreur dont le trait std :: fmt :: Debug n'est pas implémenté pour GenderCategory . L'attribut # [derive (Debug)] est utilisé pour supprimer cette erreur.

// The `derive` attribute automatically creates the implementation
// required to make this `enum` printable with `fmt::Debug`.
#[derive(Debug)]
enum GenderCategory {
   Male,Female
}
fn main() {
   let male = GenderCategory::Male;
   let female = GenderCategory::Female;

   println!("{:?}",male);
   println!("{:?}",female);
}

Production

Male
Female

Struct et Enum

L'exemple suivant définit une structure Person. Le champ gender est du type GenderCategory (qui est une énumération) et peut être attribué à Male ou Female comme valeur.

// The `derive` attribute automatically creates the 
implementation
// required to make this `enum` printable with 
`fmt::Debug`.

#[derive(Debug)]
enum GenderCategory {
   Male,Female
}

// The `derive` attribute automatically creates the implementation
// required to make this `struct` printable with `fmt::Debug`.
#[derive(Debug)]
struct Person {
   name:String,
   gender:GenderCategory
}

fn main() {
   let p1 = Person {
      name:String::from("Mohtashim"),
      gender:GenderCategory::Male
   };
   let p2 = Person {
      name:String::from("Amy"),
      gender:GenderCategory::Female
   };
   println!("{:?}",p1);
   println!("{:?}",p2);
}

L'exemple crée des objets p1 et p2 de type Person et initialise les attributs, le nom et le sexe de chacun de ces objets.

Production

Person { name: "Mohtashim", gender: Male }
Person { name: "Amy", gender: Female }

Option Enum

Option est une énumération prédéfinie dans la bibliothèque standard Rust. Cette énumération a deux valeurs - Some (data) et None.

Syntaxe

enum Option<T> {
   Some(T),      //used to return a value
   None          // used to return null, as Rust doesn't support 
   the null keyword
}

Ici, le type T représente la valeur de tout type.

Rust ne prend pas en charge le mot clé null . La valeur None , dans enumOption , peut être utilisée par une fonction pour renvoyer une valeur nulle. S'il y a des données à renvoyer, la fonction peut renvoyer Some (data) .

Comprenons cela avec un exemple -

Le programme définit une fonction is_even () , avec un type de retour Option. La fonction vérifie si la valeur passée est un nombre pair. Si l'entrée est paire, une valeur true est renvoyée, sinon la fonction renvoie None .

fn main() {
   let result = is_even(3);
   println!("{:?}",result);
   println!("{:?}",is_even(30));
}
fn is_even(no:i32)->Option<bool> {
   if no%2 == 0 {
      Some(true)
   } else {
      None
   }
}

Production

None
Some(true)

Énoncé de correspondance et énumération

L' instruction match peut être utilisée pour comparer des valeurs stockées dans une énumération. L'exemple suivant définit une fonction, print_size , qui prend CarType enum comme paramètre. La fonction compare les valeurs des paramètres avec un ensemble prédéfini de constantes et affiche le message approprié.

enum CarType {
   Hatch,
   Sedan,
   SUV
}
fn print_size(car:CarType) {
   match car {
      CarType::Hatch => {
         println!("Small sized car");
      },
      CarType::Sedan => {
         println!("medium sized car");
      },
      CarType::SUV =>{
         println!("Large sized Sports Utility car");
      }
   }
}
fn main(){
   print_size(CarType::SUV);
   print_size(CarType::Hatch);
   print_size(CarType::Sedan);
}

Production

Large sized Sports Utility car
Small sized car
medium sized car

Match avec option

L'exemple de la fonction is_even , qui retourne le type Option, peut également être implémenté avec l'instruction match comme indiqué ci-dessous -

fn main() {
   match is_even(5) {
      Some(data) => {
         if data==true {
            println!("Even no");
         }
      },
      None => {
         println!("not even");
      }
   }
}
fn is_even(no:i32)->Option<bool> {
   if no%2 == 0 {
      Some(true)
   } else {
      None
   }
}

Production

not even

Correspondance et énumération avec le type de données

Il est possible d'ajouter un type de données à chaque variante d'une énumération. Dans l'exemple suivant, les variantes Name et Usr_ID de l'énumération sont respectivement de type String et integer. L'exemple suivant montre l'utilisation de l'instruction match avec une énumération ayant un type de données.

// The `derive` attribute automatically creates the implementation
// required to make this `enum` printable with `fmt::Debug`.
#[derive(Debug)]
enum GenderCategory {
   Name(String),Usr_ID(i32)
}
fn main() {
   let p1 = GenderCategory::Name(String::from("Mohtashim"));
   let p2 = GenderCategory::Usr_ID(100);
   println!("{:?}",p1);
   println!("{:?}",p2);

   match p1 {
      GenderCategory::Name(val)=> {
         println!("{}",val);
      }
      GenderCategory::Usr_ID(val)=> {
         println!("{}",val);
      }
   }
}

Production

Name("Mohtashim")
Usr_ID(100)
Mohtashim

Un groupe logique de code est appelé un module. Plusieurs modules sont compilés dans une unité appeléecrate. Les programmes Rust peuvent contenir une caisse binaire ou une caisse de bibliothèque. Une caisse binaire est un projet exécutable qui a une méthode main () . Une caisse de bibliothèque est un groupe de composants qui peuvent être réutilisés dans d'autres projets. Contrairement à une caisse binaire, une caisse de bibliothèque n'a pas de point d'entrée (méthode main ()). L'outil Cargo est utilisé pour gérer les caisses dans Rust. Par exemple, le module de réseau contient des fonctions liées au réseau et le module graphique contient des fonctions liées au dessin. Les modules sont similaires aux espaces de noms dans d'autres langages de programmation. Les caisses tierces peuvent être téléchargées à l'aide de la cargaison de crates.io .

Sr.Non Terme et description
1

crate

Est une unité de compilation dans Rust; Crate est compilé en binaire ou en bibliothèque.

2

cargo

L'outil officiel de gestion des paquets Rust pour les caisses.

3

module

Regroupe logiquement le code dans une caisse.

4

crates.io

Le registre officiel des packages Rust.

Syntaxe

//public module
pub mod a_public_module {
   pub fn a_public_function() {
      //public function
   }
   fn a_private_function() {
      //private function
   }
}
//private module
mod a_private_module {
   fn a_private_function() {
   }
}

Les modules peuvent être publics ou privés. Les composants d'un module privé ne sont pas accessibles par d'autres modules. Les modules de Rust sont privés par défaut. Au contraire, les fonctions d'un module public sont accessibles par d'autres modules. Les modules doivent être précédés depubmot-clé pour le rendre public. Les fonctions d'un module public doivent également être rendues publiques.

Illustration: définition d'un module

L'exemple définit un module public - les films . Le module contient une fonction play () qui accepte un paramètre et imprime sa valeur.

pub mod movies {
   pub fn play(name:String) {
      println!("Playing movie {}",name);
   }
}
fn main(){
   movies::play("Herold and Kumar".to_string());
}

Production

Playing movie Herold and Kumar

Utiliser un mot-clé

Le mot-clé use permet d'importer un module public.

Syntaxe

use public_module_name::function_name;

Illustration

pub mod movies {
   pub fn play(name:String) {
      println!("Playing movie {}",name);
   }
}
use movies::play;
fn main(){
   play("Herold and Kumar ".to_string());
}

Production

Playing movie Herold and Kumar

Modules imbriqués

Les modules peuvent également être imbriqués. Le module comédie est imbriqué dans le module anglais , qui est en outre imbriqué dans le module films . L'exemple ci-dessous définit une fonction play dans le module movies / english / comedy .

pub mod movies {
   pub mod english {
      pub mod comedy {
         pub fn play(name:String) {
            println!("Playing comedy movie {}",name);
         }
      }
   }
}
use movies::english::comedy::play; 
// importing a public module

fn main() {
   // short path syntax
   play("Herold and Kumar".to_string());
   play("The Hangover".to_string());

   //full path syntax
   movies::english::comedy::play("Airplane!".to_string());
}

Production

Playing comedy movie Herold and Kumar
Playing comedy movie The Hangover
Playing comedy movie Airplane!

Illustration - Créer une caisse de bibliothèque et consommer dans une caisse binaire

Créons une caisse de bibliothèque nommée movie_lib, qui contient un module movies. Pour construire lemovie_lib caisse de bibliothèque, nous utiliserons l'outil cargo.

Étape 1 - Créer un dossier de projet

Créez un dossier movie-app suivi d'un sous-dossier movie-lib . Une fois le dossier et le sous-dossier créés, créez unsrcdossier et un fichier Cargo.toml dans ce répertoire. Le code source doit aller dans le dossier src . Créez les fichiers lib.rs et movies.rs dans le dossier src. Le fichier Cargo.toml contiendra les métadonnées du projet comme le numéro de version, le nom de l'auteur, etc.

La structure du répertoire du projet sera comme indiqué ci-dessous -

movie-app
   movie-lib/
      -->Cargo.toml
      -->src/
         lib.rs
         movies.rs

Étape 2 - Modifiez le fichier Cargo.toml pour ajouter des métadonnées de projet

[package]
name = "movies_lib"
version = "0.1.0"
authors = ["Mohtashim"]

Étape 3 - Modifiez le fichier lib.rs.

Ajoutez la définition de module suivante à ce fichier.

pub mod movies;

La ligne ci-dessus crée un module public - movies.

Étape 4 - Modifiez le fichier movies.rs

Ce fichier définira toutes les fonctions du module films.

pub fn play(name:String){
   println!("Playing movie {} :movies-app",name);
}

Le code ci-dessus définit une fonction play() qui accepte un paramètre et l’imprime sur la console.

Étape 5 - Construisez la caisse de la bibliothèque

Créer une application à l'aide du cargo buildpour vérifier si la caisse de la bibliothèque est correctement structurée. Assurez-vous que vous êtes à la racine du projet - le dossier movie-app. Le message suivant sera affiché dans le terminal si la construction réussit.

D:\Rust\movie-lib> cargo build
   Compiling movies_lib v0.1.0 (file:///D:/Rust/movie-lib)
   Finished dev [unoptimized + debuginfo] target(s) in 0.67s

Étape 6 - Créez une application de test

Créer un autre dossier movie-lib-testdans le dossier movie-app suivi d'un fichier Cargo.toml et du dossier src. Ce projet devrait avoir la méthode principale car il s'agit d'une caisse binaire, qui consommera la caisse de bibliothèque créée précédemment. Créez un fichier main.rs dans le dossier src. La structure des dossiers sera comme indiqué.

movie-app
   movie-lib 
   // already completed

   movie-lib-test/
      -->Cargo.toml
      -->src/
         main.rs

Étape 7 - Ajoutez ce qui suit dans le fichier Cargo.toml

[package]
name = "test_for_movie_lib"
version = "0.1.0"
authors = ["Mohtashim"]

[dependencies]
movies_lib = { path = "../movie-lib" }

NOTE- Le chemin d'accès au dossier de la bibliothèque est défini comme dépendances. Le diagramme suivant montre le contenu des deux projets.

Étape 8 - Ajoutez ce qui suit au fichier main.rs

extern crate movies_lib;
use movies_lib::movies::play;
fn main() {
   println!("inside main of test ");
   play("Tutorialspoint".to_string())
}

Le code ci-dessus importe un package externe appelé movies_lib. Vérifiez le Cargo.toml du projet actuel pour vérifier le nom de la caisse.

Étape 9 - Utilisation de la construction de fret et de la course de fret

Nous utiliserons la construction de la cargaison et l'exécution de la cargaison pour construire le projet binaire et l'exécuter comme indiqué ci-dessous -

La bibliothèque de collection standard de Rust fournit des implémentations efficaces des structures de données de programmation générales les plus courantes. Ce chapitre traite de l'implémentation des collections couramment utilisées - Vector, HashMap et HashSet.

Vecteur

Un vecteur est un tableau redimensionnable. Il stocke les valeurs dans des blocs de mémoire contigus. La structure prédéfinie Vec peut être utilisée pour créer des vecteurs. Certaines caractéristiques importantes d'un vecteur sont:

  • Un vecteur peut grossir ou diminuer au moment de l'exécution.

  • Un vecteur est une collection homogène.

  • Un vecteur stocke les données sous forme de séquence d'éléments dans un ordre particulier. Chaque élément d'un vecteur reçoit un numéro d'index unique. L'index commence à 0 et va jusqu'à n-1 où, n est la taille de la collection. Par exemple, dans une collection de 5 éléments, le premier élément sera à l'index 0 et le dernier élément sera à l'index 4.

  • Un vecteur ajoutera des valeurs uniquement à (ou près de) la fin. En d'autres termes, un vecteur peut être utilisé pour implémenter une pile.

  • La mémoire d'un vecteur est allouée dans le tas.

Syntaxe - Création d'un vecteur

let mut instance_name = Vec::new();

La méthode statique new () de la structure Vec est utilisée pour créer une instance vectorielle.

Alternativement, un vecteur peut également être créé en utilisant le vec! macro. La syntaxe est la suivante -

let vector_name = vec![val1,val2,val3]

Le tableau suivant répertorie certaines fonctions couramment utilisées de la structure Vec.

Sr.Non Méthode Signature et description
1 Nouveau()

pub fn new()->Vect

Construit un nouveau Vec vide. Le vecteur ne sera pas alloué tant que les éléments ne seront pas poussés dessus.

2 pousser()

pub fn push(&mut self, value: T)

Ajoute un élément à l'arrière d'une collection.

3 retirer()

pub fn remove(&mut self, index: usize) -> T

Supprime et renvoie l'élément à l'index de position dans le vecteur, en décalant tous les éléments après celui-ci vers la gauche.

4 contient ()

pub fn contains(&self, x: &T) -> bool

Renvoie true si la tranche contient un élément avec la valeur donnée.

5 len ()

pub fn len(&self) -> usize

Renvoie le nombre d'éléments dans le vecteur, également appelé sa «longueur».

Illustration: Création d'un vecteur - nouveau ()

Pour créer un vecteur, nous utilisons la méthode statique new -

fn main() {
   let mut v = Vec::new();
   v.push(20);
   v.push(30);
   v.push(40);

   println!("size of vector is :{}",v.len());
   println!("{:?}",v);
}

L'exemple ci-dessus crée un vecteur en utilisant la méthode statique new () qui est définie dans la structure Vec . La fonction push (val) ajoute la valeur passée en paramètre à la collection. La fonction len () renvoie la longueur du vecteur.

Production

size of vector is :3
[20, 30, 40]

Illustration: création d'un vecteur - vec! Macro

Le code suivant crée un vecteur à l'aide de vec! macro. Le type de données du vecteur est déduit de la première valeur qui lui est affectée.

fn main() {
   let v = vec![1,2,3];
   println!("{:?}",v);
}

Production

[1, 2, 3]

Comme mentionné précédemment, un vecteur ne peut contenir que des valeurs du même type de données. L'extrait de code suivant générera une erreur [E0308]: erreur de types incompatibles .

fn main() {
   let v = vec![1,2,3,"hello"];
   println!("{:?}",v);
}

Illustration: pousser ()

Ajoute un élément à la fin d'une collection.

fn main() {
   let mut v = Vec::new();
   v.push(20);
   v.push(30);
   v.push(40);
   
   println!("{:?}",v);
}

Production

[20, 30, 40]

Illustration: remove ()

Supprime et renvoie l'élément à l'index de position dans le vecteur, en décalant tous les éléments après celui-ci vers la gauche.

fn main() {
   let mut v = vec![10,20,30];
   v.remove(1);
   println!("{:?}",v);
}

Production

[10, 30]

Illustration - contient ()

Renvoie true si la tranche contient un élément avec la valeur donnée -

fn main() {
   let v = vec![10,20,30];
   if v.contains(&10) {
      println!("found 10");
   }
   println!("{:?}",v);
}

Production

found 10
[10, 20, 30]

Illustration: len ()

Renvoie le nombre d'éléments dans le vecteur, également appelé sa «longueur».

fn main() {
   let v = vec![1,2,3];
   println!("size of vector is :{}",v.len());
}

Production

size of vector is :3

Accéder aux valeurs d'un vecteur

Les éléments individuels d'un vecteur sont accessibles en utilisant leurs numéros d'index correspondants. L'exemple suivant crée une annonce vectorielle imprime la valeur du premier élément.

fn main() {
   let mut v = Vec::new();
   v.push(20);
   v.push(30);

   println!("{:?}",v[0]);
}
Output: `20`

Les valeurs d'un vecteur peuvent également être récupérées à l'aide d'une référence à la collection.

fn main() {
   let mut v = Vec::new();
   v.push(20);
   v.push(30);
   v.push(40);
   v.push(500);

   for i in &v {
      println!("{}",i);
   }
   println!("{:?}",v);
}

Production

20
30
40
500
[20, 30, 40, 500]

HashMap

Une carte est une collection de paires clé-valeur (appelées entrées). Aucune entrée dans une carte ne peut avoir la même clé. En bref, une carte est une table de consultation. Un HashMap stocke les clés et les valeurs dans une table de hachage. Les entrées sont stockées dans un ordre arbitraire. La clé est utilisée pour rechercher des valeurs dans le HashMap. La structure HashMap est définie dans lestd::collectionsmodule. Ce module doit être importé explicitement pour accéder à la structure HashMap.

Syntaxe: Création d'un HashMap

let mut instance_name = HashMap::new();

La méthode statique new () de la structure HashMap est utilisée pour créer un objet HashMap. Cette méthode crée un HashMap vide.

Les fonctions couramment utilisées de HashMap sont discutées ci-dessous -

Sr.Non Méthode Signature et description
1 insérer()

pub fn insert(&mut self, k: K, v: V) -> Option

Insère une paire clé / valeur, si aucune clé, Aucune est renvoyée. Après la mise à jour, l'ancienne valeur est renvoyée.

2 len ()

pub fn len(&self) -> usize

Renvoie le nombre d'éléments de la carte.

3 avoir()

pub fn get<Q: ?Sized>(&lself, k: &Q) -> Option<&V> where K:Borrow Q:Hash+ Eq

Renvoie une référence à la valeur correspondant à la clé.

4 iter ()

pub fn iter(&self) -> Iter<K, V>

Un itérateur visitant toutes les paires clé-valeur dans un ordre arbitraire. Le type d'élément de l'itérateur est (& 'a K, &' a V).

5 contient_key

pub fn contains_key<Q: ?Sized>(&self, k: &Q) -> bool

Renvoie true si la carte contient une valeur pour la clé spécifiée.

6 retirer()

pub fn remove_entry<Q: ?Sized>(&mut self, k: &Q) -> Option<(K, V)>

Supprime une clé de la carte, renvoyant la clé et la valeur stockées si la clé était auparavant dans la carte.

Illustration: insérer ()

Insère une paire clé / valeur dans le HashMap.

use std::collections::HashMap;
fn main(){
   let mut stateCodes = HashMap::new();
   stateCodes.insert("KL","Kerala");
   stateCodes.insert("MH","Maharashtra");
   println!("{:?}",stateCodes);
}

Le programme ci-dessus crée un HashMap et l'initialise avec 2 paires clé-valeur.

Production

{"KL": "Kerala", "MH": "Maharashtra"}

Illustration: len ()

Renvoie le nombre d'éléments de la carte

use std::collections::HashMap;
fn main() {
   let mut stateCodes = HashMap::new();
   stateCodes.insert("KL","Kerala");
   stateCodes.insert("MH","Maharashtra");
   println!("size of map is {}",stateCodes.len());
}

L'exemple ci-dessus crée un HashMap et imprime le nombre total d'éléments qu'il contient.

Production

size of map is 2

Illustration - get ()

Renvoie une référence à la valeur correspondant à la clé. L'exemple suivant récupère la valeur de la clé KL dans le HashMap.

use std::collections::HashMap;
fn main() {
   let mut stateCodes = HashMap::new();
   stateCodes.insert("KL","Kerala");
   stateCodes.insert("MH","Maharashtra");
   println!("size of map is {}",stateCodes.len());
   println!("{:?}",stateCodes);

   match stateCodes.get(&"KL") {
      Some(value)=> {
         println!("Value for key KL is {}",value);
      }
      None => {
         println!("nothing found");
      }
   }
}

Production

size of map is 2
{"KL": "Kerala", "MH": "Maharashtra"}
Value for key KL is Kerala

Illustration - Iter ()

Renvoie un itérateur contenant une référence à toutes les paires clé-valeur dans un ordre arbitraire.

use std::collections::HashMap;
fn main() {
   let mut stateCodes = HashMap::new();
   stateCodes.insert("KL","Kerala");
   stateCodes.insert("MH","Maharashtra");

   for (key, val) in stateCodes.iter() {
      println!("key: {} val: {}", key, val);
   }
}

Production

key: MH val: Maharashtra
key: KL val: Kerala

Illustration: contains_key ()

Renvoie true si la carte contient une valeur pour la clé spécifiée.

use std::collections::HashMap;
fn main() {
   let mut stateCodes = HashMap::new();
   stateCodes.insert("KL","Kerala");
   stateCodes.insert("MH","Maharashtra");
   stateCodes.insert("GJ","Gujarat");

   if stateCodes.contains_key(&"GJ") {
      println!("found key");
   }
}

Production

found key

Illustration: remove ()

Supprime une clé de la carte.

use std::collections::HashMap;
fn main() {
   let mut stateCodes = HashMap::new();
   stateCodes.insert("KL","Kerala");
   stateCodes.insert("MH","Maharashtra");
   stateCodes.insert("GJ","Gujarat");

   println!("length of the hashmap {}",stateCodes.len());
   stateCodes.remove(&"GJ");
   println!("length of the hashmap after remove() {}",stateCodes.len());
}

Production

length of the hashmap 3
length of the hashmap after remove() 2

HashSet

HashSet est un ensemble de valeurs uniques de type T. L'ajout et la suppression de valeurs est rapide, et il est rapide de demander si une valeur donnée est dans l'ensemble ou non. La structure HashSet est définie dans le module std :: collections. Ce module doit être importé explicitement pour accéder à la structure HashSet.

Syntaxe: Création d'un HashSet

let mut hash_set_name = HashSet::new();

La méthode statique, nouvelle , de la structure HashSet est utilisée pour créer un HashSet. Cette méthode crée un HashSet vide.

Le tableau suivant répertorie certaines des méthodes couramment utilisées de la structure HashSet.

Sr.Non Méthode Signature et description
1 insérer()

pub fn insert(&mut self, value: T) -> bool

Ajoute une valeur à l'ensemble. Si l'ensemble n'avait pas cette valeur présente, true est renvoyé, sinon false.

2 len ()

pub fn len(&self) -> usize

Renvoie le nombre d'éléments de l'ensemble.

3 avoir()

pub fn get<Q:?Sized>(&self, value: &Q) -> Option<&T> where T: Borrow,Q: Hash + Eq,

Renvoie une référence à la valeur de l'ensemble, le cas échéant, égale à la valeur donnée.

4 iter ()

pub fn iter(&self) -> Iter

Renvoie un itérateur visitant tous les éléments dans un ordre arbitraire. Le type d'élément itérateur est & 'a T.

5 contient_key

pub fn contains<Q: ?Sized>(&self, value: &Q) -> bool

Renvoie true si l'ensemble contient une valeur.

6 retirer()

pub fn remove<Q: ?Sized>(&mut self, value: &Q) -> bool

Supprime une valeur de l'ensemble. Renvoie true si la valeur était présente dans l'ensemble.

Illustration - insérer ()

Ajoute une valeur à l'ensemble. Un HashSet n'ajoute pas de valeurs en double à la collection.

use std::collections::HashSet;
fn main() {
   let mut names = HashSet::new();

   names.insert("Mohtashim");
   names.insert("Kannan");
   names.insert("TutorialsPoint");
   names.insert("Mohtashim");//duplicates not added

   println!("{:?}",names);
}

Production

{"TutorialsPoint", "Kannan", "Mohtashim"}

Illustration: len ()

Renvoie le nombre d'éléments de l'ensemble.

use std::collections::HashSet;
fn main() {
   let mut names = HashSet::new();
   names.insert("Mohtashim");
   names.insert("Kannan");
   names.insert("TutorialsPoint");
   println!("size of the set is {}",names.len());
}

Production

size of the set is 3

Illustration - Iter ()

Récupère un itérateur visitant tous les éléments dans un ordre arbitraire.

use std::collections::HashSet;
fn main() {
   let mut names = HashSet::new();
   names.insert("Mohtashim");
   names.insert("Kannan");
   names.insert("TutorialsPoint");
   names.insert("Mohtashim");

   for name in names.iter() {
      println!("{}",name);
   }
}

Production

TutorialsPoint
Mohtashim
Kannan

Illustration: get ()

Renvoie une référence à la valeur de l'ensemble, le cas échéant, qui est égale à la valeur donnée.

use std::collections::HashSet;
fn main() {
   let mut names = HashSet::new();
   names.insert("Mohtashim");
   names.insert("Kannan");
   names.insert("TutorialsPoint");
   names.insert("Mohtashim");

   match names.get(&"Mohtashim"){
      Some(value)=>{
         println!("found {}",value);
      }
      None =>{
         println!("not found");
      }
   }
   println!("{:?}",names);
}

Production

found Mohtashim
{"Kannan", "Mohtashim", "TutorialsPoint"}

Illustration - contient ()

Renvoie true si l'ensemble contient une valeur.

use std::collections::HashSet;

fn main() {
   let mut names = HashSet::new();
   names.insert("Mohtashim");
   names.insert("Kannan");
   names.insert("TutorialsPoint");

   if names.contains(&"Kannan") {
      println!("found name");
   }  
}

Production

found name

Illustration: remove ()

Supprime une valeur de l'ensemble.

use std::collections::HashSet;

fn main() {
   let mut names = HashSet::new();
   names.insert("Mohtashim");
   names.insert("Kannan");
   names.insert("TutorialsPoint");
   println!("length of the Hashset: {}",names.len());
   names.remove(&"Kannan");
   println!("length of the Hashset after remove() : {}",names.len());
}

Production

length of the Hashset: 3
length of the Hashset after remove() : 2

Dans Rust, les erreurs peuvent être classées en deux grandes catégories, comme indiqué dans le tableau ci-dessous.

Sr.Non Nom et description Usage
1

Recoverable

Erreurs qui peuvent être traitées

Énumération des résultats
2

UnRecoverable

Erreurs qui ne peuvent pas être traitées

macro de panique

Une erreur récupérable est une erreur qui peut être corrigée. Un programme peut réessayer l'opération qui a échoué ou spécifier un autre plan d'action lorsqu'il rencontre une erreur récupérable. Les erreurs récupérables ne provoquent pas l'échec brutal d'un programme. Un exemple d'erreur récupérable est l'erreur File Not Found .

Des erreurs irrécupérables provoquent l'échec brutal d'un programme. Un programme ne peut pas revenir à son état normal si une erreur irrémédiable se produit. Il ne peut pas réessayer l'opération qui a échoué ou annuler l'erreur. Un exemple d'erreur irrémédiable est de tenter d'accéder à un emplacement au-delà de la fin d'un tableau.

Contrairement à d'autres langages de programmation, Rust n'a pas d'exceptions. Il renvoie une énumération Result <T, E> pour les erreurs récupérables, alors qu'il appelle lepanicmacro si le programme rencontre une erreur irrémédiable. La macro panique entraîne la fermeture brutale du programme.

Macro de panique et erreurs irrécupérables

panique! macro permet à un programme de se terminer immédiatement et de fournir une rétroaction à l'appelant du programme. Il doit être utilisé lorsqu'un programme atteint un état irrécupérable.

fn main() {
   panic!("Hello");
   println!("End of main"); //unreachable statement
}

Dans l'exemple ci-dessus, le programme se terminera immédiatement lorsqu'il rencontrera la panique! macro.

Production

thread 'main' panicked at 'Hello', main.rs:3

Illustration: panique! macro

fn main() {
   let a = [10,20,30];
   a[10]; //invokes a panic since index 10 cannot be reached
}

La sortie est comme indiqué ci-dessous -

warning: this expression will panic at run-time
--> main.rs:4:4
  |
4 | a[10];
  | ^^^^^ index out of bounds: the len is 3 but the index is 10

$main
thread 'main' panicked at 'index out of bounds: the len 
is 3 but the index is 10', main.rs:4
note: Run with `RUST_BACKTRACE=1` for a backtrace.

Un programme peut provoquer la panique! macro si les règles métier ne sont pas respectées, comme indiqué dans l'exemple ci-dessous -

fn main() {
   let no = 13; 
   //try with odd and even
   if no%2 == 0 {
      println!("Thank you , number is even");
   } else {
      panic!("NOT_AN_EVEN"); 
   }
   println!("End of main");
}

L'exemple ci-dessus renvoie une erreur si la valeur affectée à la variable est impaire.

Production

thread 'main' panicked at 'NOT_AN_EVEN', main.rs:9
note: Run with `RUST_BACKTRACE=1` for a backtrace.

Énumération des résultats et erreurs récupérables

Résultat Enum - <T, E> peut être utilisé pour gérer les erreurs récupérables. Il a deux variantes -OK et Err. T et E sont des paramètres de type générique. T représente le type de la valeur qui sera renvoyée en cas de succès dans la variante OK, et E représente le type d'erreur qui sera retourné en cas d'échec dans la variante Err.

enum Result<T,E> {
   OK(T),
   Err(E)
}

Comprenons cela à l'aide d'un exemple -

use std::fs::File;
fn main() {
   let f = File::open("main.jpg"); 
   //this file does not exist
   println!("{:?}",f);
}

Le programme renvoie OK (Fichier) si le fichier existe déjà et Err (Erreur) si le fichier n'est pas trouvé.

Err(Error { repr: Os { code: 2, message: "No such file or directory" } })

Voyons maintenant comment gérer la variante Err.

L'exemple suivant gère une erreur renvoyée lors de l'ouverture du fichier à l'aide de match déclaration

use std::fs::File;
fn main() {
   let f = File::open("main.jpg");   // main.jpg doesn't exist
   match f {
      Ok(f)=> {
         println!("file found {:?}",f);
      },
      Err(e)=> {
         println!("file not found \n{:?}",e);   //handled error
      }
   }
   println!("end of main");
}

NOTE- Le programme imprime la fin de l' événement principal même si le fichier n'a pas été trouvé. Cela signifie que le programme a géré correctement l'erreur.

Production

file not found
Os { code: 2, kind: NotFound, message: "The system cannot find the file specified." }
end of main

Illustration

La fonction is_even renvoie une erreur si le nombre n'est pas un nombre pair. La fonction main () gère cette erreur.

fn main(){
   let result = is_even(13);
   match result {
      Ok(d)=>{
         println!("no is even {}",d);
      },
      Err(msg)=>{
         println!("Error msg is {}",msg);
      }
   }
   println!("end of main");
}
fn is_even(no:i32)->Result<bool,String> {
   if no%2==0 {
      return Ok(true);
   } else {
      return Err("NOT_AN_EVEN".to_string());
   }
}

NOTE- Puisque la fonction main gère correctement l'erreur, la fin de l' instruction principale est imprimée.

Production

Error msg is NOT_AN_EVEN
end of main

dérouler () et attendre ()

La bibliothèque standard contient quelques méthodes d'assistance que les deux énumérations - Result <T, E> et Option <T> implémentent. Vous pouvez les utiliser pour simplifier les cas d'erreur où vous ne vous attendez vraiment pas à ce que les choses échouent. En cas de succès d'une méthode, la fonction "dérouler" est utilisée pour extraire le résultat réel.

Sr.Non Méthode Signature et description
1 déballer

unwrap(self): T

S'attend à ce que le soi soit Ok / Some et renvoie la valeur contenue à l'intérieur. Si c'estErr ou None au lieu de cela, il soulève une panique avec le contenu de l'erreur affiché.

2 attendre

expect(self, msg: &str): T

Se comporte comme un dépliage, sauf qu'il génère un message personnalisé avant de paniquer en plus du contenu de l'erreur.

déballer()

La fonction unwrap () renvoie le résultat réel qu'une opération réussit. Il renvoie une panique avec un message d'erreur par défaut si une opération échoue. Cette fonction est un raccourci pour l'instruction match. Ceci est illustré dans l'exemple ci-dessous -

fn main(){
   let result = is_even(10).unwrap();
   println!("result is {}",result);
   println!("end of main");
}
fn is_even(no:i32)->Result<bool,String> {
   if no%2==0 {
      return Ok(true);
   } else {
      return Err("NOT_AN_EVEN".to_string());
   }
}
result is true
end of main

Modifiez le code ci-dessus pour passer un nombre impair au is_even() fonction.

La fonction unwrap () paniquera et retournera un message d'erreur par défaut comme indiqué ci-dessous

thread 'main' panicked at 'called `Result::unwrap()` on 
an `Err` value: "NOT_AN_EVEN"', libcore\result.rs:945:5
note: Run with `RUST_BACKTRACE=1` for a backtrace

attendre()

Le programme peut renvoyer un message d'erreur personnalisé en cas de panique. Ceci est illustré dans l'exemple suivant -

use std::fs::File;
fn main(){
   let f = File::open("pqr.txt").expect("File not able to open");
   //file does not exist
   println!("end of main");
}

La fonction expect () est similaire à unwrap (). La seule différence est qu'un message d'erreur personnalisé peut être affiché à l'aide de expect.

Production

thread 'main' panicked at 'File not able to open: Error { repr: Os 
{ code: 2, message: "No such file or directory" } }', src/libcore/result.rs:860
note: Run with `RUST_BACKTRACE=1` for a backtrace.

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. Les génériques permettent d'écrire du code plus concis et plus propre en réduisant la duplication de code et en fournissant une 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

Ce chapitre explique comment accepter les valeurs de l'entrée standard (clavier) et afficher les valeurs vers la sortie standard (console). Dans ce chapitre, nous aborderons également le passage des arguments de ligne de commande.

Types de lecteurs et d'écrivains

Les fonctionnalités de bibliothèque standard de Rust pour l'entrée et la sortie sont organisées autour de deux traits -

  • Read
  • Write
Sr.Non Trait et description Exemple
1

Read

Les types qui implémentent Read ont des méthodes pour l'entrée orientée octet. Ils sont appelés lecteurs

Stdin, fichier
2

Write

Les types qui implémentent l'écriture prennent en charge la sortie de texte orientée octet et UTF-8. Ils sont appelés écrivains.

Stdout, fichier

Lire le trait

Readerssont des composants à partir desquels votre programme peut lire des octets. Les exemples incluent la lecture des entrées du clavier, des fichiers, etc.read_line() La méthode de ce trait peut être utilisée pour lire des données, une ligne à la fois, à partir d'un fichier ou d'un flux d'entrée standard.

Sr.Non Trait Méthode et description
1 Lis

read_line(&mut line)->Result

Lit une ligne de texte et l'ajoute à la ligne, qui est une chaîne. La valeur de retour est un io :: Result, le nombre d'octets lus.

Illustration − Reading from the Console − stdin()

Rust programs might have to accept values from the user at runtime. The following example reads values from the standard input (Keyboard) and prints it to the console.

fn main(){
   let mut line = String::new();
   println!("Enter your name :");
   let b1 = std::io::stdin().read_line(&mut line).unwrap();
   println!("Hello , {}", line);
   println!("no of bytes read , {}", b1);
}

The stdin() function returns a handle to the standard input stream of the current process, to which the read_line function can be applied. This function tries to read all the characters present in the input buffer when it encounters an end-of-line character.

Output

Enter your name :
Mohtashim
Hello , Mohtashim
no of bytes read , 10

Write Trait

Writers are components that your program can write bytes to. Examples include printing values to the console, writing to files, etc. The write() method of this trait can be used to write data to a file or standard output stream.

Sr.No Trait Method & Description
1 Write

write(&buf)->Result

Writes some of the bytes in the slice buf to the underlying stream. It returns an io::Result, the number of bytes written.

Illustration - Writing to the Console - stdout()

The print! or println! macros can be used to display text on the console. However, you can also use the write() standard library function to display some text to the standard output.

Let us consider an example to understand this.

use std::io::Write;
fn main() {
   let b1 = std::io::stdout().write("Tutorials ".as_bytes()).unwrap();
   let b2 = std::io::stdout().write(String::from("Point").as_bytes()).unwrap();
   std::io::stdout().write(format!("\nbytes written {}",(b1+b2)).as_bytes()).unwrap();
}

Output

Tutorials Point
bytes written 15

The stdout() standard library function returns a handle to the standard output stream of the current process, to which the write function can be applied. The write() method returns an enum, Result. The unwrap() is a helper method to extract the actual result from the enumeration. The unwrap method will send panic if an error occurs.

NOTE − File IO is discussed in the next chapter.

CommandLine Arguments

CommandLine arguments are passed to a program before executing it. They are like parameters passed to functions. CommandLine parameters can be used to pass values to the main() function. The std::env::args() returns the commandline arguments.

Illustration

The following example passes values as commandLine arguments to the main() function. The program is created in a file name main.rs.

//main.rs
fn main(){
   let cmd_line = std::env::args();
   println!("No of elements in arguments is :{}",cmd_line.len()); 
   //print total number of values passed
   for arg in cmd_line {
      println!("[{}]",arg); //print all values passed 
      as commandline arguments
   }
}

The program will generate a file main.exe once compiled. Multiple command line parameters should be separated by space. Execute main.exe from the terminal as main.exe hello tutorialspoint.

NOTEhello and tutorialspoint are commandline arguments.

Output

No of elements in arguments is :3
[main.exe]
[hello]
[tutorialspoint]

The output shows 3 arguments as the main.exe is the first argument.

Illustration

The following program calculates the sum of values passed as commandline arguments. A list integer values separated by space is passed to program.

fn main(){
   let cmd_line = std::env::args();
   println!("No of elements in arguments is 
   :{}",cmd_line.len()); 
   // total number of elements passed

   let mut sum = 0;
   let mut has_read_first_arg = false;

   //iterate through all the arguments and calculate their sum

   for arg in cmd_line {
      if has_read_first_arg { //skip the first argument since it is the exe file name
         sum += arg.parse::<i32>().unwrap();
      }
      has_read_first_arg = true; 
      // set the flag to true to calculate sum for the subsequent arguments.
   }
   println!("sum is {}",sum);
}

On executing the program as main.exe 1 2 3 4, the output will be −

No of elements in arguments is :5
sum is 10

In addition to reading and writing to console, Rust allows reading and writing to files.

The File struct represents a file. It allows a program to perform read-write operations on a file. All methods in the File struct return a variant of the io::Result enumeration.

The commonly used methods of the File struct are listed in the table below −

Sr.No Module Method Signature Description
1 std::fs::File open() pub fn open<P: AsRef>(path: P) -> Result The open static method can be used to open a file in read-only mode.
2 std::fs::File create() pub fn create<P: AsRef>(path: P) -> Result Static method opens a file in write-only mode. If the file already existed, the old content is destroyed. Otherwise, a new file is created.
3 std::fs::remove_file remove_file() pub fn remove_file<P: AsRef>(path: P) -> Result<()> Removes a file from the filesystem. There is no guarantee that the file is immediately deleted.
4 std::fs::OpenOptions append() pub fn append(&mut self, append: bool) -> &mut OpenOptions Sets the option for the append mode of file.
5 std::io::Writes write_all() fn write_all(&mut self, buf: &[u8]) -> Result<()> Attempts to write an entire buffer into this write.
6 std::io::Read read_to_string() fn read_to_string(&mut self, buf: &mut String) -> Result Reads all bytes until EOF in this source, appending them to buf.

Write to a File

Let us see an example to understand how to write a file.

The following program creates a file 'data.txt'. The create() method is used to create a file. The method returns a file handle if the file is created successfully. The last line write_all function will write bytes in newly created file. If any of the operations fail, the expect() function returns an error message.

use std::io::Write;
fn main() {
   let mut file = std::fs::File::create("data.txt").expect("create failed");
   file.write_all("Hello World".as_bytes()).expect("write failed");
   file.write_all("\nTutorialsPoint".as_bytes()).expect("write failed");
   println!("data written to file" );
}

Output

data written to file

Read from a File

The following program reads the contents in a file data.txt and prints it to the console. The "open" function is used to open an existing file. An absolute or relative path to the file is passed to the open() function as a parameter. The open() function throws an exception if the file does not exist, or if it is not accessible for whatever reason. If it succeeds, a file handle to such file is assigned to the "file" variable.

The "read_to_string" function of the "file" handle is used to read contents of that file into a string variable.

use std::io::Read;

fn main(){
   let mut file = std::fs::File::open("data.txt").unwrap();
   let mut contents = String::new();
   file.read_to_string(&mut contents).unwrap();
   print!("{}", contents);
}

Output

Hello World
TutorialsPoint

Delete a file

The following example uses the remove_file() function to delete a file. The expect() function returns a custom message in case an error occurs.

use std::fs;
fn main() {
   fs::remove_file("data.txt").expect("could not remove file");
   println!("file is removed");
}

Output

file is removed

Append data to a file

The append() function writes data to the end of the file. This is shown in the example given below −

use std::fs::OpenOptions;
use std::io::Write;

fn main() {
   let mut file = OpenOptions::new().append(true).open("data.txt").expect(
      "cannot open file");
   file.write_all("Hello World".as_bytes()).expect("write failed");
   file.write_all("\nTutorialsPoint".as_bytes()).expect("write failed");
   println!("file append success");
}

Output

file append success

Copy a file

The following example copies the contents in a file to a new file.

use std::io::Read;
use std::io::Write;

fn main() {
   let mut command_line: std::env::Args = std::env::args();
   command_line.next().unwrap();
   // skip the executable file name
   // accept the source file
   let source = command_line.next().unwrap();
   // accept the destination file
   let destination = command_line.next().unwrap();
   let mut file_in = std::fs::File::open(source).unwrap();
   let mut file_out = std::fs::File::create(destination).unwrap();
   let mut buffer = [0u8; 4096];
   loop {
      let nbytes = file_in.read(&mut buffer).unwrap();
      file_out.write(&buffer[..nbytes]).unwrap();
      if nbytes < buffer.len() { break; }
   }
}

Execute the above program as main.exe data.txt datacopy.txt. Two command line arguments are passed while executing the file −

  • the path to the source file
  • the destination file

Cargo is the package manager for RUST. This acts like a tool and manages Rust projects.

Some commonly used cargo commands are listed in the table below −

Sr.No Command & Description
1

cargo build

Compiles the current project.

2

cargo check

Analyzes the current project and report errors, but don't build object files.

3

cargo run

Builds and executes src/main.rs.

4

cargo clean

Removes the target directory.

5

cargo update

Updates dependencies listed in Cargo.lock.

6

cargo new

Creates a new cargo project.

Cargo helps to download third party libraries. Therefore, it acts like a package manager. You can also build your own libraries. Cargo is installed by default when you install Rust.

To create a new cargo project, we can use the commands given below.

Create a binary crate

cargo new project_name --bin

Create a library crate

cargo new project_name --lib

To check the current version of cargo, execute the following command −

cargo --version

Illustration - Create a Binary Cargo project

The game generates a random number and prompts the user to guess the number.

Step 1 - Create a project folder

Open the terminal and type the following command cargo new guess-game-app --bin.

This will create the following folder structure.

guess-game-app/
   -->Cargo.toml
   -->src/
      main.rs

The cargo new command is used to create a crate. The --bin flag indicates that the crate being created is a binary crate. Public crates are stored in a central repository called crates.io https://crates.io/.

Step 2 - Include references to external libraries

This example needs to generate a random number. Since the internal standard library does not provide random number generation logic, we need to look at external libraries or crates. Let us use rand crate which is available at crates.io website crates.io

The https://crates.io/crates/rand is a rust library for random number generation. Rand provides utilities to generate random numbers, to convert them to useful types and distributions, and some randomness-related algorithms.

The following diagram shows crate.io website and search result for rand crate.

Copy the version of rand crate to the Cargo.toml file rand = "0.5.5".

[package]
name = "guess-game-app"
version = "0.1.0"
authors = ["Mohtashim"]

[dependencies]
rand = "0.5.5"

Step 3: Compile the Project

Navigate to the project folder. Execute the command cargo build on the terminal window −

Updating registry `https://github.com/rust-lang/crates.io-index`
Downloading rand v0.5.5
Downloading rand_core v0.2.2
Downloading winapi v0.3.6
Downloading rand_core v0.3.0
   Compiling winapi v0.3.6
   Compiling rand_core v0.3.0
   Compiling rand_core v0.2.2
   Compiling rand v0.5.5
   Compiling guess-game-app v0.1.0 
   (file:///E:/RustWorks/RustRepo/Code_Snippets/cargo-projects/guess-game-app)
   Finished dev [unoptimized + debuginfo] target(s) in 1m 07s

The rand crate and all transitive dependencies (inner dependencies of rand) will be automatically downloaded.

Step 4 - Understanding the Business Logic

Let us now see how the business logic works for the number guessing game −

  • Game initially generates a random number.

  • A user is asked to enter input and guess the number.

  • If number is less than the generated number, a message “Too low” is printed.

  • If number is greater than the generated number, a message “Too high” is printed.

  • If the user enters the number generated by the program, the game exits.

Step 5 - Edit the main.rs file

Ajoutez la logique métier au fichier main.rs.

use std::io;
extern crate rand; 
//importing external crate
use rand::random;
fn get_guess() -> u8 {
   loop {
      println!("Input guess") ;
      let mut guess = String::new();
      io::stdin().read_line(&mut guess)
         .expect("could not read from stdin");
      match guess.trim().parse::<u8>(){ //remember to trim input to avoid enter spaces
         Ok(v) => return v,
         Err(e) => println!("could not understand input {}",e)
      }
   }
}
fn handle_guess(guess:u8,correct:u8)-> bool {
   if guess < correct {
      println!("Too low");
      false

   } else if guess> correct {
      println!("Too high");
      false
   } else {
      println!("You go it ..");
      true
   }
}
fn main() {
   println!("Welcome to no guessing game");

   let correct:u8 = random();
   println!("correct value is {}",correct);
   loop {
      let guess = get_guess();
      if handle_guess(guess,correct){
         break;
      }
   }
}

Étape 6 - Compilez et exécutez le projet

Exécutez la commande cargo run sur le terminal. Assurez-vous que le terminal pointe vers le répertoire du projet.

Welcome to no guessing game
correct value is 97
Input guess
20
Too low
Input guess
100
Too high
Input guess
97
You got it ..

Dans ce chapitre, nous allons apprendre comment les itérateurs et les fermetures fonctionnent dans RUST.

Itérateurs

Un itérateur permet d'itérer sur une collection de valeurs telles que des tableaux, des vecteurs, des cartes, etc. Les itérateurs implémentent le trait Iterator qui est défini dans la bibliothèque standard de Rust. La méthode iter () renvoie un objet itérateur de la collection. Les valeurs d'un objet itérateur sont appelées éléments. La méthode next () de l'itérateur peut être utilisée pour parcourir les éléments. La méthode next () renvoie une valeur None lorsqu'elle atteint la fin de la collection.

L'exemple suivant utilise un itérateur pour lire les valeurs d'un tableau.

fn main() {
   //declare an array
   let a = [10,20,30];

   let mut iter = a.iter(); 
   // fetch an iterator object for the array
   println!("{:?}",iter);

   //fetch individual values from the iterator object
   println!("{:?}",iter.next());
   println!("{:?}",iter.next());
   println!("{:?}",iter.next());
   println!("{:?}",iter.next());
}

Production

Iter([10, 20, 30])
Some(10)
Some(20)
Some(30)
None

Si une collection telle que array ou Vector implémente le trait Iterator, elle peut être parcourue en utilisant la syntaxe for ... in comme indiqué ci-dessous.

fn main() {
   let a = [10,20,30];
   let iter = a.iter();
   for data in iter{
      print!("{}\t",data);
   }
}

Production

10 20 30

Les 3 méthodes suivantes retournent un objet itérateur à partir d'une collection, où T représente les éléments d'une collection.

Sr.Non Méthodes et description
1

iter()

donne un itérateur sur & T (référence à T)

2

into_iter()

donne un itérateur sur T

3

iter_mut()

donne un itérateur sur & mut T

Illustration: iter ()

La fonction iter () utilise le concept d'emprunt. Il renvoie une référence à chaque élément de la collection, laissant la collection intacte et disponible pour une réutilisation après la boucle.

fn main() {
   let names = vec!["Kannan", "Mohtashim", "Kiran"];
   for name in names.iter() {
      match name {
         &"Mohtashim" => println!("There is a rustacean among us!"),
         _ => println!("Hello {}", name),
      }
   }
   println!("{:?}",names); 
   // reusing the collection after iteration
}

Production

Hello Kannan
There is a rustacean among us!
Hello Kiran
["Kannan", "Mohtashim", "Kiran"]

Illustration - into_iter ()

Cette fonction utilise le concept de propriété. Il déplace les valeurs de la collection dans un objet iter, c'est-à-dire que la collection est consommée et qu'elle n'est plus disponible pour une réutilisation.

fn main(){
   let names = vec!["Kannan", "Mohtashim", "Kiran"];
   for name in names.into_iter() {
      match name {
         "Mohtashim" => println!("There is a rustacean among us!"),
         _ => println!("Hello {}", name),
      }
   }
   // cannot reuse the collection after iteration
   //println!("{:?}",names); 
   //Error:Cannot access after ownership move
}

Production

Hello Kannan
There is a rustacean among us!
Hello Kiran

Illustration - pour et iter_mut ()

Cette fonction est comme la fonction iter () . Cependant, cette fonction peut modifier les éléments de la collection.

fn main() {
   let mut names = vec!["Kannan", "Mohtashim", "Kiran"];
   for name in names.iter_mut() {
      match name {
         &mut "Mohtashim" => println!("There is a rustacean among us!"),
         _ => println!("Hello {}", name),
      }
   }
   println!("{:?}",names);
   //// reusing the collection after iteration
}

Production

Hello Kannan
There is a rustacean among us!
Hello Kiran
["Kannan", "Mohtashim", "Kiran"]

Fermeture

La fermeture fait référence à une fonction dans une autre fonction. Ce sont des fonctions anonymes - des fonctions sans nom. La fermeture peut être utilisée pour affecter une fonction à une variable. Cela permet à un programme de passer une fonction en tant que paramètre à d'autres fonctions. La fermeture est également connue sous le nom de fonction en ligne. Les variables de la fonction externe sont accessibles par des fonctions en ligne.

Syntaxe: définir une fermeture

Une définition de fermeture peut éventuellement avoir des paramètres. Les paramètres sont entourés de deux barres verticales.

let closure_function = |parameter| {
   //logic
}

La syntaxe invoquant une fermeture implémente Fntraits. Ainsi, il peut être invoqué avec() syntaxe.

closure_function(parameter);    //invoking

Illustration

L'exemple suivant définit une fermeture is_even dans la fonction main () . La fermeture renvoie vrai si un nombre est pair et renvoie faux si le nombre est impair.

fn main(){
   let is_even = |x| {
      x%2==0
   };
   let no = 13;
   println!("{} is even ? {}",no,is_even(no));
}

Production

13 is even ? false

Illustration

fn main(){
   let val = 10; 
   // declared outside
   let closure2 = |x| {
      x + val //inner function accessing outer fn variable
   };
   println!("{}",closure2(2));
}

La fonction main () déclare une variable val et une fermeture. La fermeture accède à la variable déclarée dans la fonction externe main () .

Production

12

Rust alloue tout sur la pile par défaut. Vous pouvez stocker des éléments sur le tas en les enveloppant dans des pointeurs intelligents comme Box . Des types tels que Vec et String aident implicitement l'allocation de tas. Les pointeurs intelligents implémentent les traits répertoriés dans le tableau ci-dessous. Ces traits des pointeurs intelligents les différencient d'une structure ordinaire -

Sr.Non Nom du trait Paquet et description
1 Deref

std::ops::Deref

Utilisé pour les opérations de déréférencement immuables, comme * v.

2 Laissez tomber

std::ops::Drop

Utilisé pour exécuter du code lorsqu'une valeur est hors de portée. Ceci est parfois appelé un destructeur

Dans ce chapitre, nous découvrirons les Boxpointeur intelligent. Nous allons également apprendre à créer un pointeur intelligent personnalisé comme Box.

Boîte

Le pointeur intelligent Box, également appelé boîte, vous permet de stocker des données sur le tas plutôt que sur la pile. La pile contient le pointeur vers les données du tas. Une boîte n'a pas de surcharge de performances, autre que le stockage de leurs données sur le tas.

Voyons comment utiliser une boîte pour stocker une valeur i32 sur le tas.

fn main() {
   let var_i32 = 5; 
   //stack
   let b = Box::new(var_i32); 
   //heap
   println!("b = {}", b);
}

Production

b = 5

Pour accéder à une valeur pointée par une variable, utilisez le déréférencement. Le * est utilisé comme opérateur de déréférencement. Voyons comment utiliser le déréférencement avec Box.

fn main() {
   let x = 5; 
   //value type variable
   let y = Box::new(x); 
   //y points to a new value 5 in the heap

   println!("{}",5==x);
   println!("{}",5==*y); 
   //dereferencing y
}

La variable x est un type valeur avec la valeur 5. Ainsi, l'expression 5 == x retournera true. La variable y pointe vers le tas. Pour accéder à la valeur dans le tas, nous devons déréférencer en utilisant * y. * y renvoie la valeur 5. Ainsi, l'expression 5 == * y renvoie vrai.

Production

true
true

Illustration - Trait Deref

Le trait Deref, fourni par la bibliothèque standard, nous oblige à implémenter une méthode nommée deref , qui emprunte self et renvoie une référence aux données internes. L'exemple suivant crée une structure MyBox , qui est un type générique. Il met en œuvre le trait Deref . Cette caractéristique nous aide à accéder aux valeurs de tas encapsulées par y en utilisant * y .

use std::ops::Deref;
struct MyBox<T>(T);
impl<T> MyBox<T> { 
   // Generic structure with static method new
   fn new(x:T)-> MyBox<T> {
      MyBox(x)
   }
}
impl<T> Deref for MyBox<T> {
   type Target = T;
   fn deref(&self) -> &T {
      &self.0 //returns data
   }
}
fn main() {
   let x = 5;
   let y = MyBox::new(x); 
   // calling static method
   
   println!("5==x is {}",5==x);
   println!("5==*y is {}",5==*y); 
   // dereferencing y
   println!("x==*y is {}",x==*y);
   //dereferencing y
}

Production

5==x is true
5==*y is true
x==*y is true

Illustration - Trait de goutte

Le trait Drop contient la méthode drop () . Cette méthode est appelée lorsqu'une structure qui a implémenté ce trait est hors de portée. Dans certains langages, le programmeur doit appeler du code pour libérer de la mémoire ou des ressources à chaque fois qu'il finit d'utiliser une instance d'un pointeur intelligent. Dans Rust, vous pouvez obtenir une désallocation automatique de la mémoire à l'aide du trait Drop.

use std::ops::Deref;

struct MyBox<T>(T);
impl<T> MyBox<T> {
   fn new(x:T)->MyBox<T>{
      MyBox(x)
   }
}
impl<T> Deref for MyBox<T> {
   type Target = T;
      fn deref(&self) -< &T {
      &self.0
   }
}
impl<T> Drop for MyBox<T>{
   fn drop(&mut self){
      println!("dropping MyBox object from memory ");
   }
}
fn main() {
   let x = 50;
   MyBox::new(x);
   MyBox::new("Hello");
}

Dans l'exemple ci-dessus, la méthode drop sera appelée deux fois lorsque nous créons deux objets dans le tas.

dropping MyBox object from memory
dropping MyBox object from memory

En programmation simultanée, différentes parties d'un programme s'exécutent indépendamment. D'autre part, dans la programmation parallèle, différentes parties d'un programme s'exécutent en même temps. Les deux modèles sont tout aussi importants car davantage d'ordinateurs tirent parti de leurs multiples processeurs.

Fils

Nous pouvons utiliser des threads pour exécuter des codes simultanément. Dans les systèmes d'exploitation actuels, le code d'un programme exécuté est exécuté dans un processus et le système d'exploitation gère plusieurs processus à la fois. Dans votre programme, vous pouvez également avoir des parties indépendantes qui s'exécutent simultanément. Les fonctionnalités qui exécutent ces pièces indépendantes sont appelées threads.

Créer un fil

le thread::spawnLa fonction est utilisée pour créer un nouveau thread. La fonction spawn prend une fermeture comme paramètre. La fermeture définit le code qui doit être exécuté par le thread. L'exemple suivant imprime du texte à partir d'un fil principal et d'autres textes à partir d'un nouveau fil.

//import the necessary modules
use std::thread;
use std::time::Duration;

fn main() {
   //create a new thread
   thread::spawn(|| {
      for i in 1..10 {
         println!("hi number {} from the spawned thread!", i);
         thread::sleep(Duration::from_millis(1));
      }
   });
   //code executed by the main thread
   for i in 1..5 {
      println!("hi number {} from the main thread!", i);
      thread::sleep(Duration::from_millis(1));
   }
}

Production

hi number 1 from the main thread!
hi number 1 from the spawned thread!
hi number 2 from the main thread!
hi number 2 from the spawned thread!
hi number 3 from the main thread!
hi number 3 from the spawned thread!
hi number 4 from the spawned thread!
hi number 4 from the main thread!

Le thread principal imprime les valeurs de 1 à 4.

NOTE- Le nouveau fil sera arrêté à la fin du fil principal. La sortie de ce programme peut être un peu différente à chaque fois.

le thread::sleepLa fonction force un thread à arrêter son exécution pendant une courte durée, permettant à un thread différent de s'exécuter. Les threads se relaient probablement, mais ce n'est pas garanti - cela dépend de la façon dont le système d'exploitation planifie les threads. Dans cette exécution, le thread principal est imprimé en premier, même si l'instruction d'impression du thread généré apparaît en premier dans le code. De plus, même si le thread généré est programmé pour imprimer des valeurs jusqu'à 9, il n'est arrivé qu'à 5 avant que le thread principal ne s'arrête.

Joindre les poignées

Un thread généré peut ne pas avoir la possibilité de s'exécuter ou de s'exécuter complètement. C'est parce que le thread principal se termine rapidement. La fonction spawn <F, T> (f: F) -> JoinHandlelt; T> renvoie un JoinHandle. La méthode join () sur JoinHandle attend que le thread associé se termine.

use std::thread;
use std::time::Duration;

fn main() {
   let handle = thread::spawn(|| {
      for i in 1..10 {
         println!("hi number {} from the spawned thread!", i);
         thread::sleep(Duration::from_millis(1));
      }
   });
   for i in 1..5 {
      println!("hi number {} from the main thread!", i);
      thread::sleep(Duration::from_millis(1));
   }
   handle.join().unwrap();
}

Production

hi number 1 from the main thread!
hi number 1 from the spawned thread!
hi number 2 from the spawned thread!
hi number 2 from the main thread!
hi number 3 from the spawned thread!
hi number 3 from the main thread!
hi number 4 from the main thread!
hi number 4 from the spawned thread!
hi number 5 from the spawned thread!
hi number 6 from the spawned thread!
hi number 7 from the spawned thread!
hi number 8 from the spawned thread!
hi number 9 from the spawned thread!

Le thread principal et le thread généré continuent de basculer.

NOTE - Le thread principal attend que le thread généré se termine à cause de l'appel au join() méthode.