Rust - Panduan Cepat

Rust adalah bahasa pemrograman tingkat sistem, yang dikembangkan oleh Graydon Hoare. Mozilla Labs kemudian memperoleh program tersebut.

Bahasa Pemrograman Sistem Aplikasi v / s

Bahasa pemrograman aplikasi seperti Java / C # digunakan untuk membangun perangkat lunak, yang menyediakan layanan kepada pengguna secara langsung. Mereka membantu kami membangun aplikasi bisnis seperti spreadsheet, pengolah kata, aplikasi web, atau aplikasi seluler.

Bahasa pemrograman sistem seperti C / C ++ digunakan untuk membangun platform perangkat lunak dan perangkat lunak. Mereka dapat digunakan untuk membangun sistem operasi, mesin permainan, kompiler, dll. Bahasa pemrograman ini memerlukan interaksi perangkat keras yang tinggi.

Sistem dan bahasa pemrograman aplikasi menghadapi dua masalah utama -

  • Sulit untuk menulis kode aman.
  • Sulit untuk menulis kode multi-utas.

Mengapa Rust?

Rust berfokus pada tiga gol -

  • Safety
  • Speed
  • Concurrency

Bahasa ini dirancang untuk mengembangkan perangkat lunak yang sangat andal dan cepat dengan cara yang sederhana. Rust dapat digunakan untuk menulis program tingkat tinggi ke program khusus perangkat keras.

Performa

Bahasa pemrograman Rust tidak memiliki Garbage Collector (GC) dengan desain. Ini meningkatkan kinerja saat runtime.

Keamanan memori pada waktu kompilasi

Perangkat lunak yang dibuat menggunakan Rust aman dari masalah memori seperti pointer yang menggantung, buffer overruns, dan kebocoran memori.

Aplikasi multi-utas

Kepemilikan Rust dan aturan keamanan memori menyediakan konkurensi tanpa data race.

Dukungan untuk Perakitan Web (WASM)

Perakitan Web membantu menjalankan algoritme intensif komputasi tinggi di browser, pada perangkat yang disematkan, atau di mana pun. Ini berjalan dengan kecepatan kode asli. Rust dapat dikompilasi ke Web Assembly untuk eksekusi yang cepat dan andal.

Pemasangan Karat menjadi mudah rustup, alat berbasis konsol untuk mengelola versi Rust dan alat terkait.

Instalasi di Windows

Mari kita pelajari cara menginstal RUST di Windows.

  • Penginstalan Visual Studio 2013 atau lebih tinggi dengan alat C ++ wajib untuk menjalankan program Rust di windows. Pertama, unduh Visual Studio dari sini VS 2013 Express

  • Unduh dan pasang rustup alat untuk windows. rustup-init.exetersedia untuk diunduh di sini - Rust Lang

  • Klik dua kali rustup-init.exemengajukan. Setelah mengklik, layar berikut akan muncul.

  • Tekan enter untuk penginstalan default. Setelah instalasi selesai, layar berikut akan muncul.

  • Dari layar instalasi, jelas bahwa file terkait Rust disimpan di folder -

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

Isi folder tersebut adalah -

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
  • Cargoadalah manajer paket untuk Rust. Untuk memverifikasi jikacargo diinstal, jalankan perintah berikut -

C:\Users\Admin>cargo -V
cargo 1.29.0 (524a578d7 2018-08-05)
  • Kompiler untuk Rust adalah rustc. Untuk memverifikasi versi kompiler, jalankan perintah berikut -

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

Instalasi di Linux / Mac

Untuk memasang rustup di Linux atau macOS, buka terminal dan masukkan perintah berikut.

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

Perintah mengunduh skrip dan memulai penginstalan rustupalat, yang menginstal Rust versi stabil terbaru. Anda mungkin diminta memasukkan kata sandi Anda. Jika penginstalan berhasil, baris berikut akan muncul -

Rust is installed now. Great!

Skrip instalasi secara otomatis menambahkan Rust ke PATH sistem Anda setelah login berikutnya. Untuk mulai menggunakan Rust segera alih-alih memulai ulang terminal Anda, jalankan perintah berikut di shell Anda untuk menambahkan Rust ke PATH sistem Anda secara manual -

$ source $HOME/.cargo/env

Atau, Anda dapat menambahkan baris berikut ke ~ / .bash_profile Anda -

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

NOTE - Ketika Anda mencoba untuk mengkompilasi program Rust dan mendapatkan kesalahan yang menunjukkan bahwa penaut tidak dapat dijalankan, itu berarti penaut tidak diinstal pada sistem Anda dan Anda harus menginstalnya secara manual.

Menggunakan Tutorial Point Coding Ground untuk RUST

Read-Evaluate-Print Loop (REPL) adalah shell interaktif yang mudah digunakan untuk menyusun dan menjalankan program komputer. Jika Anda ingin mengkompilasi dan menjalankan program Rust secara online di dalam browser, gunakan Tutorialspoint Coding Ground .

Bab ini menjelaskan sintaks dasar bahasa Rust melalui a HelloWorld contoh.

  • Membuat HelloWorld-App folder dan arahkan ke folder itu di terminal

C:\Users\Admin>mkdir HelloWorld-App
C:\Users\Admin>cd HelloWorld-App
C:\Users\Admin\HelloWorld-App>
  • Untuk membuat file Rust, jalankan perintah berikut -

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

File program Rust memiliki ekstensi .rs. Perintah di atas membuat file kosongHello.rsdan membukanya di CATATANpad. Tambahkan kode yang diberikan di bawah ini ke file ini -

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

Program di atas mendefinisikan sebuah fungsi main fn main () . Kata kunci fn digunakan untuk mendefinisikan sebuah fungsi. The main () adalah fungsi yang telah ditetapkan yang bertindak sebagai titik masuk ke program. println! adalah makro yang telah ditentukan di Rust. Ini digunakan untuk mencetak string (di sini Halo) ke konsol. Panggilan makro selalu ditandai dengan tanda seru - ! .

  • Kompilasi file Hello.rs file menggunakan rustc.

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

Setelah kompilasi program berhasil, file yang dapat dieksekusi ( nama_file.exe ) dibuat. Untuk memverifikasi apakah file .exe dibuat, jalankan perintah berikut.

C:\Users\Admin\HelloWorld-App>dir
//lists the files in folder
Hello.exe
Hello.pdb
Hello.rs
  • Jalankan file Hello.exe dan verifikasi hasilnya.

Apa itu makro?

Rust menyediakan sistem makro yang kuat yang memungkinkan meta-pemrograman. Seperti yang Anda lihat pada contoh sebelumnya, makro terlihat seperti fungsi, kecuali namanya diakhiri dengan bang (!), Tetapi alih-alih menghasilkan panggilan fungsi, makro diperluas menjadi kode sumber yang dikompilasi dengan program lainnya. Oleh karena itu, mereka menyediakan lebih banyak fitur runtime ke program tidak seperti fungsi. Makro adalah versi fungsi yang diperluas.

Menggunakan println! Makro - Sintaks

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

Komentar di Rust

Komentar adalah cara untuk meningkatkan keterbacaan program. Komentar dapat digunakan untuk memasukkan informasi tambahan tentang program seperti pembuat kode, petunjuk tentang fungsi / konstruksi, dll. Kompilator mengabaikan komentar.

Rust mendukung jenis komentar berikut -

  • Komentar satu baris (//) - Teks apa pun di antara // dan akhir baris dianggap sebagai komentar

  • Komentar multi-baris (/ * * /) - Komentar ini dapat mencakup beberapa baris.

Contoh

//this is single line comment

/* This is a
   Multi-line comment
*/

Jalankan secara online

Program Rust dapat dijalankan secara online melalui Tutorialspoint Coding Ground . Tulis program HelloWorld di tab Editor dan klik Execute untuk melihat hasilnya.

Sistem Jenis mewakili berbagai jenis nilai yang didukung oleh bahasa. Sistem Jenis memeriksa validitas nilai yang diberikan, sebelum disimpan atau dimanipulasi oleh program. Ini memastikan bahwa kode berperilaku seperti yang diharapkan. Sistem Jenis lebih jauh memungkinkan untuk petunjuk kode yang lebih kaya dan dokumentasi otomatis juga.

Rust adalah bahasa yang diketik secara statis. Setiap nilai di Rust memiliki tipe data tertentu. Kompilator dapat secara otomatis menyimpulkan tipe data variabel berdasarkan nilai yang diberikan padanya.

Deklarasikan Variabel

Menggunakan let kata kunci untuk mendeklarasikan variabel.

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);
}

Dalam contoh di atas, tipe data variabel akan disimpulkan dari nilai yang diberikan padanya. Misalnya, Rust akan menetapkan tipe data string ke variabel company_string , tipe data float ke rating_float , dll.

Hasil cetak! makro membutuhkan dua argumen -

  • Sintaks khusus {} , yang merupakan placeholder
  • Nama variabel atau konstanta

Placeholder akan diganti dengan nilai variabel

Output dari potongan kode di atas adalah -

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

Jenis Skalar

Jenis skalar mewakili satu nilai. Misalnya, 10,3.14, 'c'. Karat memiliki empat jenis skalar utama.

  • Integer
  • Floating-point
  • Booleans
  • Characters

Kami akan mempelajari tentang setiap jenis di bagian selanjutnya.

Bilangan bulat

Integer adalah bilangan tanpa komponen pecahan. Sederhananya, tipe data integer digunakan untuk merepresentasikan bilangan bulat.

Integer selanjutnya dapat diklasifikasikan sebagai Signed dan Unsigned. Bilangan bulat bertanda tangan dapat menyimpan nilai negatif dan positif. Bilangan bulat tak bertanda tangan hanya dapat menyimpan nilai positif. Penjelasan rinci jika tipe integer diberikan di bawah -

Sr.No. Ukuran Tertanda Tidak ditandatangani
1 8 bit i8 u8
2 16 bit i16 u16
3 32 bit i32 u32
4 64 bit i64 u64
5 128 bit i128 u128
6 Lengkungan isize menggunakan

Ukuran bilangan bulat bisa berupa lengkungan . Ini berarti ukuran tipe data akan diturunkan dari arsitektur mesin. Bilangan bulat yang ukurannya arch adalah 32 bit pada mesin x86 dan 64 bit pada mesin x64. Bilangan bulat lengkung terutama digunakan saat mengindeks beberapa jenis koleksi.

Ilustrasi

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);
}

Outputnya akan seperti yang diberikan di bawah ini -

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

Kode di atas akan mengembalikan kesalahan kompilasi jika Anda mengganti nilai usia dengan nilai floating-point.

Rentang Integer

Setiap varian bertanda tangan dapat menyimpan angka dari - (2 ^ (n-1) hingga 2 ^ (n-1) -1 , dengan n adalah jumlah bit yang digunakan varian. Misalnya, i8 dapat menyimpan angka dari - (2 ^ 7) menjadi 2 ^ 7 -1 - di sini kita mengganti n dengan 8.

Setiap varian unsigned dapat menyimpan angka dari 0 hingga (2 ^ n) -1 . Misalnya, u8 dapat menyimpan angka dari 0 hingga 2 ^ 7 , yang sama dengan 0 hingga 255.

Integer Overflow

Kelebihan bilangan bulat terjadi ketika nilai yang ditetapkan ke variabel bilangan bulat melebihi rentang yang ditentukan Rust untuk tipe data. Mari kita pahami ini dengan sebuah contoh -

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);
}

Rentang valid dari variabel unsigned u8 adalah 0 sampai 255. Dalam contoh di atas, variabel diberi nilai lebih dari 255 (batas atas untuk variabel integer di Rust). Saat dieksekusi, kode di atas akan mengembalikan peringatan -warning − literal out of range for u8untuk variabel berat, tinggi dan skor. Nilai overflow setelah 255 akan dimulai dari 0, 1, 2, dll. Output akhir tanpa peringatan seperti yang ditunjukkan di bawah ini -

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

Mengapung

Tipe data float di Rust dapat diklasifikasikan sebagai f32 dan f64. Jenis f32 adalah float presisi tunggal, dan f64 memiliki presisi ganda. Jenis defaultnya adalah f64. Pertimbangkan contoh berikut untuk memahami lebih lanjut tentang tipe data 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);
}

Outputnya akan seperti yang ditunjukkan di bawah ini -

interest is 8.35
cost is 15000.6

Pengecoran Jenis Otomatis

Pengecoran tipe otomatis tidak diperbolehkan di Rust. Perhatikan cuplikan kode berikut. Nilai integer diberikan ke variabel floatinterest.

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

Kompilator melempar file mismatched types error seperti yang diberikan di bawah ini.

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)

Pemisah Nomor

Untuk memudahkan pembacaan angka besar, kita dapat menggunakan pemisah visual _ garis bawah untuk memisahkan digit. Itu 50.000 dapat ditulis sebagai 50_000. Ini ditunjukkan pada contoh di bawah ini.

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);
}

Outputnya diberikan di bawah ini -

float value 11000.555001
int value 50000

Boolean

Jenis Boolean memiliki dua kemungkinan nilai - benar atau salah . Menggunakanbool kata kunci untuk mendeklarasikan variabel boolean.

Ilustrasi

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

Output dari kode di atas akan menjadi -

Is Rust Programming Fun ? true

Karakter

Tipe data karakter di Rust mendukung angka, huruf, Unicode dan karakter khusus. Menggunakancharkata kunci untuk mendeklarasikan variabel tipe data karakter. Jenis karakter Rust mewakili Nilai Skalar Unicode, yang berarti dapat mewakili lebih dari sekadar ASCII. Nilai Skalar Unicode berkisar dariU+0000 untuk U+D7FF dan U+E000 untuk U+10FFFF inklusif.

Mari kita pertimbangkan contoh untuk memahami lebih lanjut tentang tipe data Karakter.

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);
}

Output dari kode di atas akan menjadi -

special character is @
alphabet is A
emoji is

Variabel adalah penyimpanan bernama yang dapat dimanipulasi oleh program. Sederhananya, variabel membantu program menyimpan nilai. Variabel di Rust dikaitkan dengan tipe data tertentu. Jenis data menentukan ukuran dan tata letak memori variabel, kisaran nilai yang dapat disimpan dalam memori itu dan rangkaian operasi yang dapat dilakukan pada variabel.

Aturan Penamaan Variabel

Pada bagian ini, kita akan belajar tentang berbagai aturan penamaan variabel.

  • Nama variabel dapat terdiri dari huruf, angka, dan karakter garis bawah.

  • Ini harus dimulai dengan huruf atau garis bawah.

  • Huruf besar dan kecil berbeda karena Rust peka huruf besar / kecil.

Sintaksis

Tipe data adalah opsional saat mendeklarasikan variabel di Rust. Tipe data disimpulkan dari nilai yang diberikan ke variabel.

Sintaks untuk mendeklarasikan variabel diberikan di bawah ini.

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

Ilustrasi

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

Output dari kode di atas adalah fees is 25000 and salary is 35000.

Kekal

Secara default, variabel tidak dapat diubah - hanya baca di Rust. Dengan kata lain, nilai variabel tidak dapat diubah setelah nilai terikat ke nama variabel.

Mari kita pahami ini dengan sebuah contoh.

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

Outputnya akan seperti yang ditunjukkan di bawah ini -

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)

Pesan kesalahan menunjukkan penyebab kesalahan - Anda tidak dapat menetapkan nilai dua kali ke biaya variabel yang tidak dapat diubah. Ini adalah salah satu dari banyak cara Rust memungkinkan pemrogram untuk menulis kode dan memanfaatkan keamanan dan kemudahan konkurensi.

Yg mungkin berubah

Variabel tidak dapat diubah secara default. Awali nama variabel denganmutkata kunci agar bisa berubah. Nilai variabel yang bisa berubah dapat diubah.

Sintaks untuk mendeklarasikan variabel yang bisa berubah adalah seperti yang ditunjukkan di bawah ini -

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);
}

Output dari potongan tersebut diberikan di bawah ini -

fees is 25000
fees changed is 35000

Konstanta mewakili nilai yang tidak dapat diubah. Jika Anda mendeklarasikan sebuah konstanta maka tidak mungkin nilainya berubah. Kata kunci untuk menggunakan konstanta adalahconst. Konstanta harus diketik secara eksplisit. Berikut ini adalah sintaks untuk mendeklarasikan sebuah konstanta.

const VARIABLE_NAME:dataType = value;

Konvensi Penamaan Konstan Rust

Konvensi penamaan untuk Konstanta mirip dengan variabel. Semua karakter dalam nama konstan biasanya menggunakan huruf besar. Tidak seperti mendeklarasikan variabel, filelet kata kunci tidak digunakan untuk mendeklarasikan sebuah konstanta.

Kami telah menggunakan konstanta di Rust pada contoh di bawah ini -

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
}

Konstanta v / s Variabel

Pada bagian ini, kita akan belajar tentang faktor-faktor pembeda antara konstanta dan variabel.

  • Konstanta dideklarasikan menggunakan const kata kunci sementara variabel dideklarasikan menggunakan let kata kunci.

  • Deklarasi variabel secara opsional dapat memiliki tipe data sedangkan deklarasi konstan harus menentukan tipe data. Ini berarti const USER_LIMIT = 100 akan menghasilkan kesalahan.

  • Variabel yang dideklarasikan menggunakan letkata kunci secara default tidak dapat diubah. Namun, Anda memiliki opsi untuk memutasinya menggunakanmutkata kunci. Konstanta tidak dapat diubah.

  • Konstanta hanya dapat disetel ke ekspresi konstan dan bukan hasil pemanggilan fungsi atau nilai lain yang akan dihitung saat runtime.

  • Konstanta dapat dideklarasikan dalam cakupan apa pun, termasuk cakupan global, yang membuatnya berguna untuk nilai yang perlu diketahui oleh banyak bagian kode.

Membayangi Variabel dan Konstanta

Rust memungkinkan pemrogram untuk mendeklarasikan variabel dengan nama yang sama. Dalam kasus seperti itu, variabel baru menggantikan variabel sebelumnya.

Mari kita pahami ini dengan sebuah contoh.

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

Kode di atas mendeklarasikan dua variabel dengan nama gaji. Deklarasi pertama diberi 100,00 sedangkan deklarasi kedua diberi nilai 1,50. Variabel kedua membayangi atau menyembunyikan variabel pertama saat menampilkan keluaran.

Keluaran

The value of salary is :1.50

Rust mendukung variabel dengan tipe data berbeda saat membayangi.

Perhatikan contoh berikut.

Kode tersebut mendeklarasikan dua variabel dengan nama uname. Deklarasi pertama diberi nilai string, sedangkan deklarasi kedua diberi integer. Fungsi len mengembalikan jumlah karakter dalam nilai string.

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

Keluaran

name changed to integer: 9

Tidak seperti variabel, konstanta tidak dapat dibayangi. Jika variabel pada program di atas diganti dengan konstanta, compiler akan menampilkan error.

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

Tipe data String di Rust dapat diklasifikasikan menjadi berikut -

  • String Literal(&str)

  • Objek String(String)

String Literal

String literal (& str) digunakan ketika nilai string diketahui pada waktu kompilasi. Literal string adalah sekumpulan karakter, yang di-hardcode menjadi variabel. Misalnya, mari company = "Tutorials Point" . Literal string ditemukan di modul std :: str. Literal string juga dikenal sebagai irisan string.

Contoh berikut mendeklarasikan dua string literal - perusahaan dan lokasi .

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

Literal string bersifat statis secara default. Ini berarti literal string dijamin valid selama durasi seluruh program. Kami juga dapat secara eksplisit menentukan variabel sebagai statis seperti yang ditunjukkan di bawah ini -

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

Program di atas akan menghasilkan keluaran sebagai berikut -

company is : TutorialsPoint location :Hyderabad

Objek String

Jenis objek String disediakan di Perpustakaan Standar. Tidak seperti string literal, tipe objek string bukanlah bagian dari bahasa inti. Ini didefinisikan sebagai struktur publik di pustaka standar struct String . String adalah koleksi yang bisa tumbuh. Ini bisa berubah dan tipe yang dikodekan UTF-8. ItuStringtipe objek dapat digunakan untuk mewakili nilai string yang disediakan saat runtime. Objek string dialokasikan di heap.

Sintaksis

Untuk membuat objek String, kita dapat menggunakan salah satu sintaks berikut -

String::new()

Sintaks di atas membuat string kosong

String::from()

Ini membuat string dengan beberapa nilai default yang diteruskan sebagai parameter ke from() metode.

Contoh berikut mengilustrasikan penggunaan objek 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());
}

Contoh di atas membuat dua string - objek string kosong menggunakan metode baru dan objek string dari string literal menggunakan metode from .

Outputnya seperti yang ditunjukkan di bawah ini -

length is 0
length is 14

Metode Umum - Objek String

Sr.No. metode Tanda tangan Description
1 new() pub const fn new() → String Creates a new empty String.
2 to_string() fn to_string(&self) → String Converts the given value to a String.
3 replace() pub fn replace<'a, P>(&'a self, from: P, to: &str) → String Replaces all matches of a pattern with another string.
4 as_str() pub fn as_str(&self) → &str Extracts a string slice containing the entire string.
5 push() pub fn push(&mut self, ch: char) Appends the given char to the end of this String.
6 push_str() pub fn push_str(&mut self, string: &str) Appends a given string slice onto the end of this String.
7 len() pub fn len(&self) → usize Returns the length of this String, in bytes.
8 trim() pub fn trim(&self) → &str Returns a string slice with leading and trailing whitespace removed.
9 split_whitespace() pub fn split_whitespace(&self) → SplitWhitespace Splits a string slice by whitespace and returns an iterator.
10 split() pub fn split<'a, P>(&'a self, pat: P) → Split<'a, P> , where P is pattern can be &str, char, or a closure that determines the split. Returns an iterator over substrings of this string slice, separated by characters matched by a pattern.
11 chars() pub fn chars(&self) → Chars Returns an iterator over the chars of a string slice.

Illustration: new()

An empty string object is created using the new() method and its value is set to hello.

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

Output

The above program generates the following output −

hello

Illustration: to_string()

To access all methods of String object, convert a string literal to object type using the to_string() function.

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

Output

The above program generates the following output −

Hello TutorialsPoint , Hello!

Illustration: replace()

The replace() function takes two parameters − the first parameter is a string pattern to search for and the second parameter is the new value to be replaced. In the above example, Hello appears two times in the name1 string.

The replace function replaces all occurrences of the string Hello with Howdy.

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

Output

The above program generates the following output −

Howdy TutorialsPoint , Howdy!

Illustration: as_str()

The as_str() function extracts a string slice containing the entire string.

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);
}

Output

The above program generates the following output −

displaying string literal example_string

Illustration: push()

The push() function appends the given char to the end of this String.

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

Output

The above program generates the following output −

Tutorials

Illustration: push_str()

The push_str() function appends a given string slice onto the end of a String.

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

Output

The above program generates the following output −

Tutorials Point

Illustration: len()

The len() function returns the total number of characters in a string (including spaces).

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

Output

The above program generates the following output −

length is 20

Illustration: trim()

The trim() function removes leading and trailing spaces in a string. NOTE that this function will not remove the inline spaces.

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());
}

Output

The above program generates the following output −

Before trim
length is 24

After trim
length is 15

Illustration:split_whitespace()

The split_whitespace() splits the input string into different strings. It returns an iterator so we are iterating through the tokens as shown below −

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;
   }
}

Output

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

Illustration: split() string

The split() string method returns an iterator over substrings of a string slice, separated by characters matched by a pattern. The limitation of the split() method is that the result cannot be stored for later use. The collect method can be used to store the result returned by split() as a vector.

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]);
}

The above example splits the string fullname, whenever it encounters a comma (,).

Output

token is Kannan
token is Sudhakaran
token is Tutorialspoint

firstName is Kannan
lastname is Sudhakaran
company is Tutorialspoint

Illustration: chars()

Individual characters in a string can be accessed using the chars method. Let us consider an example to understand this.

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

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

Output

T
u
t
o
r
i
a
l
s

Concatenation of Strings with + operator

A string value can be appended to another string. This is called concatenation or interpolation. The result of string concatenation is a new string object. The + operator internally uses an add method. The syntax of the add function takes two parameters. The first parameter is self – the string object itself and the second parameter is a reference of the second string object. This is shown below −

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

Illustration: String Concatenation

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

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

The Output will be as given below

TutorialsPoint

Illustration: Type Casting

The following example illustrates converting a number to a string object −

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");
}

The Output will be as given below

2020
true

Illustration: Format! Macro

Another way to add to String objects together is using a macro function called format. The use of Format! is as shown below.

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

The Output will be as given below

Tutorials Point

An operator defines some function that will be performed on the data. The data on which operators work are called operands. Consider the following expression −

7 + 5 = 12

Here, the values 7, 5, and 12 are operands, while + and = are operators.

The major operators in Rust can be classified as −

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

Arithmetic Operators

Assume the values in variables a and b are 10 and 5 respectively.

Show Examples

Sr.No Operator Description Example
1 +(Addition) returns the sum of the operands a+b is 15
2 -(Subtraction) returns the difference of the values a-b is 5
3 * (Multiplication) returns the product of the values a*b is 50
4 / (Division) performs division operation and returns the quotient a / b is 2
5 % (Modulus) performs division operation and returns the remainder a % b is 0

NOTE − The ++ and -- operators are not supported in Rust.

Relational Operators

Relational Operators test or define the kind of relationship between two entities. Relational operators are used to compare two or more values. Relational operators return a Boolean value − true or false.

Assume the value of A is 10 and B is 20.

Show Examples

Sr.No Operator Description Example
1 > Greater than (A > B) is False
2 < Lesser than (A < B) is True
3 >= Greater than or equal to (A >= B) is False
4 <= Lesser than or equal to (A <= B) is True
5 == Equality (A == B) is fals
6 != Not equal (A != B) is True

Logical Operators

Logical Operators are used to combine two or more conditions. Logical operators too, return a Boolean value. Assume the value of variable A is 10 and B is 20.

Show Examples

Sr.No Operator Description Example
1 && (And) The operator returns true only if all the expressions specified return true (A > 10 && B > 10) is False
2 ||(OR) The operator returns true if at least one of the expressions specified return true (A > 10 || B >10) is True
3 ! (NOT) The operator returns the inverse of the expression’s result. For E.g.: !(>5) returns false !(A >10 ) is True

Bitwise Operators

Assume variable A = 2 and B = 3.

Show Examples

Sr.No Operator Description Example
1 & (Bitwise AND) It performs a Boolean AND operation on each bit of its integer arguments. (A & B) is 2
2 | (BitWise OR) It performs a Boolean OR operation on each bit of its integer arguments. (A | B) is 3
3 ^ (Bitwise XOR) It performs a Boolean exclusive OR operation on each bit of its integer arguments. Exclusive OR means that either operand one is true or operand two is true, but not both. (A ^ B) is 1
4 ! (Bitwise Not) It is a unary operator and operates by reversing all the bits in the operand. (!B) is -4
5 << (Left Shift) It moves all the bits in its first operand to the left by the number of places specified in the second operand. New bits are filled with zeros. Shifting a value left by one position is equivalent to multiplying it by 2, shifting two positions is equivalent to multiplying by 4, and so on. (A << 1) is 4
6 >> (Right Shift) Binary Right Shift Operator. The left operand’s value is moved right by the number of bits specified by the right operand. (A >> 1) is 1
7 >>> (Right shift with Zero) This operator is just like the >> operator, except that the bits shifted to the left are always zero. (A >>> 1) is 1

Decision-making structures require that the programmer specify one or more conditions to be evaluated or tested by the program, along with a statement or statements to be executed if the condition is determined to be true, and optionally, other statements to be executed if the condition is determined to be false.

Shown below is the general form of a typical decision-making structure found in most of the programming languages −

Sr.No Statement & Description
1

if statement

An if statement consists of a Boolean expression followed by one or more statements.

2

if...else statement

An if statement can be followed by an optional else statement, which executes when the Boolean expression is false.

3

else...if and nested ifstatement

You can use one if or else if statement inside another if or else if statement(s).

4

match statement

A match statement allows a variable to be tested against a list of values.

If Statement

The if…else construct evaluates a condition before a block of code is executed.

Syntax

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

If the Boolean expression evaluates to true, then the block of code inside the if statement will be executed. If the Boolean expression evaluates to false, then the first set of code after the end of the if statement (after the closing curly brace) will be executed.

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

The above example will print number is positive as the condition specified by the if block is true.

if else statement

An if can be followed by an optional else block. The else block will execute if the Boolean expression tested by the if statement evaluates to false.

Syntax

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

FlowChart

The if block guards the conditional expression. The block associated with the if statement is executed if the Boolean expression evaluates to true.

The if block may be followed by an optional else statement. The instruction block associated with the else block is executed if the expression evaluates to false.

Illustration - Simple if…else

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

The above example prints whether the value in a variable is even or odd. The if block checks the divisibility of the value by 2 to determine the same. Here is the output of the above code −

Even

Nested If

The else…if ladder is useful to test multiple conditions. The syntax is as shown below −

Syntax

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
}

When using if…else…if and else statements, there are a few points to keep in mind.

  • An if can have zero or one else's and it must come after any else..if.
  • An if can have zero to many else..if and they must come before the else.
  • Once an else..if succeeds, none of the remaining else..if or else will be tested.

Example: else…if ladder

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) ;
   }
}

The snippet displays whether the value is positive, negative or zero.

Output

2 is positive

Match Statement

The match statement checks if a current value is matching from a list of values, this is very much similar to the switch statement in C language. In the first place, notice that the expression following the match keyword does not have to be enclosed in parentheses.

The syntax is as shown below.

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

In the example given below, state_code is matched with a list of values MH, KL, KA, GA − if any match is found, a string value is returned to variable state. If no match is found, the default case _ matches and value Unkown is returned.

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);
}

Output

Found match for MH
State name is Maharashtra

There may be instances, where a block of code needs to be executed repeatedly. In general, programming instructions are executed sequentially: The first statement in a function is executed first, followed by the second, and so on.

Programming languages provide various control structures that allow for more complicated execution paths.

A loop statement allows us to execute a statement or group of statements multiple times. Given below is the general form of a loop statement in most of the programming languages.

Rust provides different types of loops to handle looping requirements −

  • while
  • loop
  • for

Definite Loop

A loop the number of iterations of which is definite/fixed is termed as a definite loop. The for loop is an implementation of a definite loop.

For Loop

The for loop executes the code block for a specified number of times. It can be used to iterate over a fixed set of values, such as an array. The syntax of the for loop is as given below

Syntax

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

An example of a for loop is as shown below

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

NOTE: that the variable x is only accessible within the for block.

Output

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

Indefinite Loop

An indefinite loop is used when the number of iterations in a loop is indeterminate or unknown.

Indefinite loops can be implemented using −

Sr.No Name & Description
1

While

The while loop executes the instructions each time the condition specified evaluates to true

2

Loop

The loop is a while(true) indefinite loop

Illustration − for while

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);
}

The output is as shown below −

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;
      }
   }
}

Itu breakpernyataan digunakan untuk mengambil kendali dari sebuah konstruksi. Menggunakan break in a loop menyebabkan program keluar dari loop.

Keluaran

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

Pernyataan Lanjutan

Pernyataan lanjutkan melompati pernyataan berikutnya dalam iterasi saat ini dan mengambil kendali kembali ke awal pengulangan. Berbeda dengan pernyataan break, continue tidak keluar dari loop. Ini menghentikan iterasi saat ini dan memulai iterasi berikutnya.

Contoh pernyataan lanjutkan diberikan di bawah ini.

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
}

Contoh di atas menampilkan jumlah nilai genap antara 0 dan 20. Loop keluar dari iterasi saat ini jika angkanya genap. Ini dicapai dengan menggunakan pernyataan lanjutkan.

Jumlah nilai ganjil antara 0 dan 20 adalah 10

Fungsi adalah blok bangunan kode yang dapat dibaca, dipelihara, dan dapat digunakan kembali. Fungsi adalah sekumpulan pernyataan untuk melakukan tugas tertentu. Fungsi mengatur program menjadi blok-blok kode logis. Setelah ditentukan, fungsi dapat dipanggil ke kode akses. Ini membuat kode dapat digunakan kembali. Selain itu, fungsi memudahkan untuk membaca dan memelihara kode program.

Deklarasi fungsi memberi tahu compiler tentang nama fungsi, tipe kembalian, dan parameter. Definisi fungsi memberikan tubuh sebenarnya dari fungsi tersebut.

Sr Tidak Deskripsi fungsi
1

Defining a function

Definisi fungsi TA menentukan apa dan bagaimana tugas tertentu akan dilakukan.

2

Calling or invoking a Function

Sebuah fungsi harus dipanggil untuk menjalankannya.

3

Returning Functions

Fungsi juga dapat mengembalikan nilai bersama dengan kontrol, kembali ke pemanggil.

4

Parameterized Function

Parameter adalah mekanisme untuk meneruskan nilai ke fungsi.

Mendefinisikan Fungsi

Definisi fungsi menentukan apa dan bagaimana tugas tertentu akan dilakukan. Sebelum menggunakan fungsi, itu harus ditentukan. Badan fungsi berisi kode yang harus dijalankan oleh fungsi tersebut. Aturan penamaan fungsi mirip dengan variabel. Fungsi ditentukan menggunakanfnkata kunci. Sintaks untuk mendefinisikan fungsi standar diberikan di bawah ini

Sintaksis

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

Deklarasi fungsi secara opsional dapat berisi parameter / argumen. Parameter digunakan untuk meneruskan nilai ke fungsi.

Contoh - Definisi fungsi sederhana

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

Memanggil Fungsi

Sebuah fungsi harus dipanggil untuk menjalankannya. Proses ini disebut sebagaifunction invocation. Nilai untuk parameter harus diteruskan saat fungsi dipanggil. Fungsi yang memanggil fungsi lain disebutcaller function.

Sintaksis

function_name(val1,val2,valN)

Contoh: Memanggil Fungsi

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

Di sini, main () adalah fungsi pemanggil.

Ilustrasi

Contoh berikut mendefinisikan sebuah fungsi fn_hello(). Fungsi mencetak pesan ke konsol. Itumain()function memanggil fungsi fn_hello () .

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

Keluaran

hello from function fn_hello

Mengembalikan Nilai dari Fungsi

Fungsi juga dapat mengembalikan nilai bersama dengan kontrol, kembali ke pemanggil. Fungsi seperti itu disebut fungsi balik.

Sintaksis

Salah satu dari sintaks berikut dapat digunakan untuk mendefinisikan fungsi dengan tipe kembalian.

Dengan pernyataan kembali

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

Sintaks singkatan tanpa pernyataan kembali

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

ilustrasi

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

Keluaran

pi value is 3.142857142857143

Fungsi dengan Parameter

Parameter adalah mekanisme untuk meneruskan nilai ke fungsi. Parameter merupakan bagian dari tanda tangan fungsi. Nilai parameter diteruskan ke fungsi selama pemanggilannya. Kecuali ditentukan secara eksplisit, jumlah nilai yang diteruskan ke fungsi harus sesuai dengan jumlah parameter yang ditentukan.

Parameter dapat diteruskan ke suatu fungsi menggunakan salah satu dari teknik berikut -

Lewati Nilai

Saat metode dipanggil, lokasi penyimpanan baru dibuat untuk setiap parameter nilai. Nilai-nilai dari parameter aktual disalin ke dalamnya. Oleh karena itu, perubahan yang dilakukan pada parameter di dalam metode yang dipanggil tidak berpengaruh pada argumen.

Contoh berikut mendeklarasikan variabel no, yang awalnya 5. Variabel tersebut dikirimkan sebagai parameter (berdasarkan nilai) ke mutate_no_to_zero()functionnction, yang mengubah nilainya menjadi nol. Setelah pemanggilan fungsi ketika kontrol kembali ke metode utama, nilainya akan sama.

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);
}

Keluaran

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

Lewati Referensi

Saat Anda meneruskan parameter dengan referensi, tidak seperti parameter nilai, lokasi penyimpanan baru tidak dibuat untuk parameter ini. Parameter referensi mewakili lokasi memori yang sama dengan parameter aktual yang disuplai ke metode. Nilai parameter dapat diteruskan dengan referensi dengan mengawali nama variabel dengan& .

Dalam contoh yang diberikan di bawah ini, kita memiliki variabel no , yang awalnya 5. Referensi ke variabel no diteruskan kemutate_no_to_zero()fungsi. Fungsi tersebut beroperasi pada variabel asli. Setelah pemanggilan fungsi, saat kontrol kembali ke metode utama, nilai variabel asli akan menjadi nol.

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
}

Operator * digunakan untuk mengakses nilai yang disimpan di lokasi memori variabel param_nomenunjuk ke. Ini juga dikenal sebagai dereferencing.

Outputnya adalah -

The value of no is 0.

Meneruskan string ke suatu fungsi

Fungsi main () meneruskan objek string ke fungsi 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);
}

Keluaran

param_name value is :TutorialsPoint

Tuple adalah tipe data gabungan. Jenis skalar hanya dapat menyimpan satu jenis data. Misalnya, variabel i32 hanya dapat menyimpan satu nilai integer. Dalam tipe gabungan, kita dapat menyimpan lebih dari satu nilai pada satu waktu dan bisa dari tipe yang berbeda.

Tupel memiliki panjang tetap - setelah dinyatakan tidak dapat tumbuh atau menyusut. Indeks tupel dimulai dari0.

Sintaksis

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

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

Ilustrasi

Contoh berikut menampilkan nilai dalam tupel.

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

The println! ( "{}", Tuple) sintaks tidak dapat digunakan untuk menampilkan nilai dalam tuple. Ini karena tupel adalah tipe gabungan. Gunakan sintaks println! ("{:?}", tuple_name) untuk mencetak nilai dalam tupel.

Keluaran

(-325, 4.9, 22)

Ilustrasi

Contoh berikut mencetak nilai individual dalam tupel.

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);
}

Keluaran

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

Ilustrasi

Contoh berikut mengirimkan tupel sebagai parameter ke suatu fungsi. Tupel diteruskan oleh nilai ke fungsi.

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);
}

Keluaran

Inside print method
(110, true, 10.9)

Menghancurkan

Tugas penghancuran adalah fitur karat di mana kami membongkar nilai tupel. Ini dicapai dengan menetapkan tupel ke variabel berbeda.

Perhatikan contoh berikut -

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);
}

Variabel x adalah tupel yang ditugaskan ke pernyataan let. Setiap variabel - age, is_male dan cgpa akan berisi nilai yang sesuai dalam tupel.

Keluaran

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

Dalam bab ini, kita akan belajar tentang sebuah array dan berbagai fitur yang terkait dengannya. Sebelum kita belajar tentang array, mari kita lihat bagaimana array berbeda dari variabel.

Variabel memiliki batasan berikut -

  • Variabel bersifat skalar. Dengan kata lain, deklarasi variabel hanya dapat berisi satu nilai dalam satu waktu. Ini berarti bahwa untuk menyimpan nilai n dalam deklarasi variabel n program akan dibutuhkan. Oleh karena itu, penggunaan variabel tidak layak ketika seseorang perlu menyimpan kumpulan nilai yang lebih besar.

  • Variabel dalam program dialokasikan memori dalam urutan acak, sehingga menyulitkan untuk mengambil / membaca nilai dalam urutan deklarasinya.

Array adalah kumpulan nilai yang homogen. Sederhananya, array adalah kumpulan nilai dari tipe data yang sama.

Fitur Array

Fitur dari sebuah array adalah sebagai berikut -

  • Deklarasi array mengalokasikan blok memori berurutan.

  • Array bersifat statis. Ini berarti bahwa array yang telah diinisialisasi tidak dapat diubah ukurannya.

  • Setiap blok memori mewakili elemen array.

  • Elemen array diidentifikasi oleh integer unik yang disebut subskrip / indeks elemen.

  • Mengisi elemen array dikenal sebagai inisialisasi array.

  • Nilai elemen array dapat diperbarui atau dimodifikasi tetapi tidak dapat dihapus.

Mendeklarasikan dan Menginisialisasi Array

Gunakan sintaks yang diberikan di bawah ini untuk mendeklarasikan dan menginisialisasi array di Rust.

Sintaksis

//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];

Dalam sintaks pertama, tipe array disimpulkan dari tipe data elemen pertama array selama inisialisasi.

Ilustrasi: Array Sederhana

Contoh berikut secara eksplisit menentukan ukuran dan tipe data larik. The {:}?! Sintaks dari println () adalah fungsi yang digunakan untuk mencetak semua nilai dalam array. Fungsi len () digunakan untuk menghitung ukuran array.

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

Keluaran

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

Ilustrasi: Array tanpa tipe data

Program berikut mendeklarasikan array dari 4 elemen. Jenis data tidak secara eksplisit ditentukan selama deklarasi variabel. Dalam kasus ini, array akan berjenis integer. Fungsi len () digunakan untuk menghitung ukuran array.

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

Keluaran

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

Ilustrasi: Nilai default

Contoh berikut membuat array dan menginisialisasi semua elemennya dengan nilai default -1 .

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

Keluaran

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

Ilustrasi: Larik dengan for loop

Contoh berikut melakukan iterasi melalui larik dan mencetak indeks dan nilainya yang sesuai. Loop mengambil nilai dari indeks 0 hingga 4 (indeks elemen array terakhir).

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]);
   }
}

Keluaran

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

Ilustrasi: Menggunakan fungsi iter ()

Fungsi iter () mengambil nilai dari semua elemen dalam sebuah array.

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);
   }
}

Keluaran

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

Ilustrasi: Array yang bisa berubah

Kata kunci mut dapat digunakan untuk mendeklarasikan array yang bisa berubah. Contoh berikut mendeklarasikan array yang bisa berubah dan mengubah nilai elemen array kedua.

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

Keluaran

[10, 0, 30, 40]

Meneruskan Array sebagai Parameter ke Fungsi

Sebuah array bisa diteruskan dengan nilai atau dengan referensi ke fungsi.

Ilustrasi: Meneruskan nilai

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);
}

Keluaran

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

Ilustrasi: Lewati referensi

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);
}

Keluaran

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

Deklarasi Array dan Konstanta

Mari kita pertimbangkan contoh yang diberikan di bawah ini untuk memahami deklarasi dan konstanta array.

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

Kompilator akan menghasilkan pengecualian. Ini karena panjang array harus diketahui pada waktu kompilasi. Di sini, nilai variabel "N" akan ditentukan saat runtime. Dengan kata lain, variabel tidak dapat digunakan untuk menentukan ukuran array.

Namun, program berikut ini valid -

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

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

Nilai pengenal yang diawali dengan kata kunci const ditentukan pada waktu kompilasi dan tidak dapat diubah saat runtime. usize berukuran pointer, jadi ukuran sebenarnya bergantung pada arsitektur yang Anda kompilasi untuk program Anda.

Memori untuk suatu program dapat dialokasikan sebagai berikut -

  • Stack
  • Heap

Tumpukan

Tumpukan mengikuti urutan terakhir dalam urutan keluar pertama. Stack menyimpan nilai data yang ukurannya diketahui pada waktu kompilasi. Misalnya, variabel i32 ukuran tetap adalah kandidat untuk alokasi tumpukan. Ukurannya diketahui pada waktu kompilasi. Semua jenis skalar dapat disimpan dalam tumpukan karena ukurannya tetap.

Pertimbangkan contoh string, yang diberi nilai pada waktu proses. Ukuran pasti dari string tersebut tidak dapat ditentukan pada waktu kompilasi. Jadi ini bukan kandidat untuk alokasi tumpukan tetapi untuk alokasi heap.

Tumpukan

Memori heap menyimpan nilai data yang ukurannya tidak diketahui pada waktu kompilasi. Ini digunakan untuk menyimpan data dinamis. Sederhananya, memori heap dialokasikan ke nilai data yang dapat berubah sepanjang siklus hidup program. Heap adalah area dalam memori yang kurang terorganisir jika dibandingkan dengan stack.

Apa itu Kepemilikan?

Setiap nilai di Rust memiliki variabel yang disebut ownerdari nilai. Setiap data yang disimpan di Rust akan memiliki pemilik yang terkait dengannya. Misalnya, dalam sintaks - misalkan usia = 30, usia adalah pemilik nilai 30 .

  • Setiap data hanya dapat dimiliki oleh satu pemilik dalam satu waktu.

  • Dua variabel tidak dapat menunjuk ke lokasi memori yang sama. Variabel akan selalu mengarah ke lokasi memori yang berbeda.

Mentransfer Kepemilikan

Kepemilikan nilai dapat ditransfer oleh -

  • Menetapkan nilai dari satu variabel ke variabel lain.

  • Meneruskan nilai ke suatu fungsi.

  • Mengembalikan nilai dari suatu fungsi.

Menetapkan nilai dari satu variabel ke variabel lain

Nilai jual utama Rust sebagai bahasa adalah keamanan memorinya. Keamanan memori dicapai dengan kontrol ketat pada siapa yang dapat menggunakan apa dan kapan batasannya.

Pertimbangkan cuplikan berikut -

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);
}

Contoh di atas mendeklarasikan vektor v. Ide kepemilikan adalah bahwa hanya satu variabel yang terikat ke sumber daya v mengikat ke sumber daya atau v2mengikat ke sumber daya. Contoh di atas melontarkan kesalahan - penggunaan nilai yang dipindahkan: `v` . Ini karena kepemilikan sumber daya ditransfer ke v2. Artinya kepemilikan dipindahkan dari v ke v2 (v2 = v) dan v tidak valid setelah perpindahan.

Meneruskan nilai ke suatu fungsi

Kepemilikan nilai juga berubah saat kita meneruskan objek di heap ke closure atau fungsi.

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);
}

Mengembalikan nilai dari suatu fungsi

Kepemilikan yang diteruskan ke fungsi akan dibatalkan saat eksekusi fungsi selesai. Salah satu solusi untuk ini adalah membiarkan fungsi mengembalikan objek yang dimiliki kembali ke pemanggil.

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);
}

Kepemilikan dan Jenis Primitif

Dalam kasus tipe primitif, konten dari satu variabel disalin ke variabel lain. Jadi, tidak ada perpindahan kepemilikan yang terjadi. Ini karena variabel primitif membutuhkan lebih sedikit sumber daya daripada objek. Perhatikan contoh berikut -

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

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

Outputnya akan menjadi - 10.

Sangat merepotkan untuk meneruskan kepemilikan variabel ke fungsi lain dan kemudian mengembalikan kepemilikannya. Rust mendukung konsep, peminjaman, di mana kepemilikan nilai ditransfer sementara ke entitas dan kemudian dikembalikan ke entitas pemilik asli.

Pertimbangkan hal berikut -

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);
}

Fungsi utama memanggil fungsi print_vector () . Vektor dilewatkan sebagai parameter untuk fungsi ini. Kepemilikan vektor juga diteruskan ke fungsi print_vector () dari main () . Kode di atas akan menghasilkan kesalahan seperti yang ditunjukkan di bawah ini ketika fungsi main () mencoba mengakses vektor v .

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

Ini karena variabel atau nilai tidak dapat lagi digunakan oleh fungsi yang semula dimiliki setelah kepemilikan dipindahkan ke fungsi lain.

Apa itu Meminjam?

Ketika suatu fungsi mentransfer kontrolnya atas variabel / nilai ke fungsi lain untuk sementara, untuk sementara disebut peminjaman. Ini dicapai dengan meneruskan referensi ke variabel(& var_name)daripada meneruskan variabel / nilai itu sendiri ke fungsi tersebut. Kepemilikan variabel / nilai ditransfer ke pemilik asli variabel setelah fungsi yang diberikan kontrol menyelesaikan eksekusi.

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);
}

Keluaran

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

Referensi yang Dapat Diubah

Sebuah fungsi bisa memodifikasi sumber daya yang dipinjam dengan menggunakan referensi yang bisa berubah ke sumber daya tersebut. Referensi yang bisa berubah diawali dengan&mut. Referensi yang dapat diubah hanya dapat beroperasi pada variabel yang dapat diubah.

Ilustrasi: Mutasi referensi integer

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

Fungsi main () mendeklarasikan variabel integer yang bisa berubah i dan meneruskan referensi yang bisa berubah dari i keadd_one(). Add_one () menambah nilai variabel i dengan satu.

Ilustrasi: Mutasi referensi string

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
}

Fungsi main () meneruskan referensi yang bisa berubah dari nama variabel ke fungsi display () . Fungsi tampilan menambahkan string tambahan ke variabel nama asli .

Keluaran

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

Slice adalah penunjuk ke blok memori. Irisan dapat digunakan untuk mengakses bagian data yang disimpan dalam blok memori yang berdekatan. Ini dapat digunakan dengan struktur data seperti array, vektor, dan string. Irisan menggunakan nomor indeks untuk mengakses bagian data. Ukuran potongan ditentukan saat runtime.

Irisan adalah penunjuk ke data aktual. Mereka diteruskan dengan mengacu pada fungsi, yang juga dikenal sebagai peminjaman.

Misalnya, irisan dapat digunakan untuk mengambil sebagian dari nilai string. String yang diiris adalah penunjuk ke objek string yang sebenarnya. Oleh karena itu, kita perlu menentukan indeks awal dan akhir dari sebuah String. Indeks dimulai dari 0 seperti array.

Sintaksis

let sliced_value = &data_structure[start_index..end_index]

Nilai indeks minimum adalah 0 dan nilai indeks maksimum adalah ukuran struktur data. Perhatikan bahwa end_index tidak akan dimasukkan dalam string terakhir.

Diagram di bawah ini menunjukkan Tutorial string sampel , yang memiliki 9 karakter. Indeks karakter pertama adalah 0 dan karakter terakhir adalah 8.

Kode berikut mengambil 5 karakter dari string (mulai dari indeks 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);
}

Keluaran

length of string is 9
rials

Ilustrasi - Mengiris array integer

Fungsi main () mendeklarasikan larik dengan 5 elemen. Ini memanggiluse_slice()berfungsi dan memberikan potongan tiga elemen (menunjuk ke larik data). Irisan dilewatkan dengan referensi. Fungsi use_slice () mencetak nilai potongan dan panjangnya.

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);
}

Keluaran

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

Irisan yang Dapat Diubah

Itu &mut kata kunci dapat digunakan untuk menandai irisan sebagai bisa berubah.

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
}

Keluaran

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

Kode di atas meneruskan potongan yang bisa berubah ke fungsi use_slice () . Fungsi tersebut mengubah elemen kedua dari larik asli.

Array digunakan untuk merepresentasikan kumpulan nilai yang homogen. Demikian pula, struktur adalah tipe data yang ditentukan pengguna lain yang tersedia di Rust yang memungkinkan kita untuk menggabungkan item data dari berbagai jenis, termasuk struktur lain. Struktur mendefinisikan data sebagai pasangan nilai kunci.

Sintaks - Mendeklarasikan struktur

Kata kunci struct digunakan untuk mendeklarasikan struktur. Karena struktur diketik secara statis, setiap bidang dalam struktur harus dikaitkan dengan tipe data. Aturan penamaan dan konvensi untuk struktur seperti variabel. Blok struktur harus diakhiri dengan titik koma.

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

Sintaks - Menginisialisasi struktur

Setelah mendeklarasikan struct, setiap bidang harus diberi nilai. Ini dikenal sebagai inisialisasi.

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);
}

Contoh di atas mendeklarasikan struct Karyawan dengan tiga bidang - nama, perusahaan, dan usia jenis. Main () menginisialisasi struktur. Ini menggunakan println! makro untuk mencetak nilai bidang yang ditentukan dalam struktur.

Keluaran

Name is :Mohtashim company is TutorialsPoint age is 50

Mengubah instance struct

Untuk mengubah sebuah instance, variabel instance harus ditandai dapat berubah. Contoh di bawah ini mendeklarasikan dan menginisialisasi struktur bernama Employee dan kemudian mengubah nilai bidang usia menjadi 40 dari 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);

Keluaran

Name is :Mohtashim company is TutorialsPoint age is 40

Meneruskan struct ke suatu fungsi

Contoh berikut menunjukkan cara melewatkan instance struct sebagai parameter. Metode tampilan mengambil contoh Karyawan sebagai parameter dan mencetak detailnya.

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

Berikut adalah program lengkapnya -

//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);
}

Keluaran

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

Mengembalikan struct dari suatu fungsi

Mari kita pertimbangkan fungsi who_is_elder () , yang membandingkan usia dua karyawan dan mengembalikan yang lebih tua.

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

Berikut adalah program lengkapnya -

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
}

Keluaran

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

Metode dalam Struktur

Metode itu seperti fungsi. Mereka adalah kelompok instruksi pemrograman yang logis. Metode dideklarasikan denganfnkata kunci. Ruang lingkup suatu metode berada dalam blok struktur.

Metode dideklarasikan di luar blok struktur. Ituimplkata kunci digunakan untuk mendefinisikan metode dalam konteks struktur. Parameter pertama dari suatu metode akan selaluself, yang merepresentasikan instance pemanggilan struktur. Metode beroperasi pada anggota data suatu struktur.

Untuk memanggil sebuah metode, pertama-tama kita perlu membuat contoh strukturnya. Metode ini dapat dipanggil menggunakan instance struktur.

Sintaksis

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

Ilustrasi

Contoh berikut mendefinisikan struktur Persegi Panjang dengan bidang - lebar dan tinggi . Sebuah metode wilayah didefinisikan dalam konteks struktur itu. Metode area mengakses bidang struktur melalui kata kunci mandiri dan menghitung luas persegi panjang.

//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());
}

Keluaran

width is 10 height is 20 area of Rectangle is 200

Metode Statis dalam Struktur

Metode statis dapat digunakan sebagai metode utilitas. Metode ini ada bahkan sebelum struktur dibuat. Metode statis dipanggil menggunakan nama struktur dan dapat diakses tanpa sebuah instance. Tidak seperti metode normal, metode statis tidak akan mengambil parameter & self .

Sintaks - Mendeklarasikan metode statis

Metode statis seperti fungsi dan metode lain secara opsional dapat berisi parameter.

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

Sintaks - Memanggil metode statis

The structure_name :: sintaks yang digunakan untuk mengakses metode statis.

structure_name::method_name(v1,v2)

Ilustrasi

Contoh berikut menggunakan metode getInstance sebagai kelas pabrik yang membuat dan mengembalikan instance Titik struktur .

//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();
}

Keluaran

x =10 y=20

Dalam pemrograman Rust, ketika kita harus memilih nilai dari daftar kemungkinan varian, kita menggunakan tipe data enumerasi. Jenis enumerasi dideklarasikan menggunakan kata kunci enum . Berikut ini adalah sintaks enum -

enum enum_name {
   variant1,
   variant2,
   variant3
}

Ilustrasi: Menggunakan Enumerasi

Contoh tersebut mendeklarasikan enum - GenderCategory , yang memiliki varian sebagai Pria dan Wanita. The cetak! makro menampilkan nilai enum. Kompilator akan menampilkan kesalahan yang sifat std :: fmt :: Debug tidak diimplementasikan untuk GenderCategory . Atribut # [derive (Debug)] digunakan untuk menekan kesalahan ini.

// 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);
}

Keluaran

Male
Female

Struct dan Enum

Contoh berikut mendefinisikan Person struktur. Jenis kelamin bidang adalah dari jenis GenderCategory (yang merupakan enum) dan dapat ditetapkan sebagai Pria atau Wanita sebagai nilai.

// 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);
}

Contoh ini membuat objek p1 dan p2 bertipe Person dan menginisialisasi atribut, nama, dan jenis kelamin untuk masing-masing objek ini.

Keluaran

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

Opsi Enum

Opsi adalah enum yang telah ditentukan di pustaka standar Rust. Enum ini memiliki dua nilai - Some (data) dan None.

Sintaksis

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

Di sini, tipe T mewakili nilai tipe apa pun.

Rust tidak mendukung kata kunci null . Nilai None , dalam enumOption , dapat digunakan oleh fungsi untuk mengembalikan nilai null. Jika ada data untuk dikembalikan, fungsi dapat mengembalikan Some (data) .

Mari kita pahami ini dengan sebuah contoh -

Program ini mendefinisikan fungsi is_even () , dengan Option tipe pengembalian. Fungsi tersebut memverifikasi jika nilai yang diteruskan adalah bilangan genap. Jika inputnya genap, maka nilai true dikembalikan, jika tidak fungsi mengembalikan 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
   }
}

Keluaran

None
Some(true)

Pernyataan Pertandingan dan Enum

The pertandingan Pernyataan dapat digunakan untuk membandingkan nilai yang tersimpan dalam sebuah enum. Contoh berikut mendefinisikan sebuah fungsi, print_size , yang menggunakan enum CarType sebagai parameter. Fungsi ini membandingkan nilai parameter dengan sekumpulan konstanta yang telah ditentukan sebelumnya dan menampilkan pesan yang sesuai.

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);
}

Keluaran

Large sized Sports Utility car
Small sized car
medium sized car

Cocokkan dengan Opsi

Contoh fungsi is_even , yang mengembalikan tipe Option, juga dapat diimplementasikan dengan pernyataan match seperti yang ditunjukkan di bawah ini -

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
   }
}

Keluaran

not even

Cocokkan & Enum dengan Jenis Data

Dimungkinkan untuk menambahkan tipe data ke setiap varian enum. Dalam contoh berikut, varian Name dan Usr_ID dari enum masing-masing adalah tipe String dan integer. Contoh berikut menunjukkan penggunaan pernyataan kecocokan dengan enum yang memiliki tipe data.

// 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);
      }
   }
}

Keluaran

Name("Mohtashim")
Usr_ID(100)
Mohtashim

Sekelompok kode logis disebut Modul. Beberapa modul dikompilasi menjadi satu unit yang disebutcrate. Program Rust mungkin berisi peti biner atau peti perpustakaan. Sebuah peti biner adalah proyek yang dapat dieksekusi yang memiliki metode main () . Peti perpustakaan adalah sekelompok komponen yang dapat digunakan kembali dalam proyek lain. Tidak seperti peti biner, peti perpustakaan tidak memiliki titik masuk (metode main ()). Alat Kargo digunakan untuk mengelola peti di Rust. Misalnya, modul jaringan berisi fungsi terkait jaringan dan modul grafis berisi fungsi terkait gambar. Modul mirip dengan namespace dalam bahasa pemrograman lain. Peti pihak ketiga dapat diunduh menggunakan kargo dari crates.io .

Sr Tidak Istilah & Deskripsi
1

crate

Apakah unit kompilasi di Rust; Crate dikompilasi ke biner atau perpustakaan.

2

cargo

Alat manajemen paket Rust resmi untuk peti.

3

module

Secara logis mengelompokkan kode di dalam peti.

4

crates.io

Registri paket Rust resmi.

Sintaksis

//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() {
   }
}

Modul dapat bersifat publik atau pribadi. Komponen dalam modul privat tidak dapat diakses oleh modul lain. Modul di Rust bersifat pribadi secara default. Sebaliknya, fungsi dalam modul publik dapat diakses oleh modul lain. Modul harus diawali denganpubkata kunci untuk menjadikannya publik. Fungsi dalam modul publik juga harus dipublikasikan.

Ilustrasi: Mendefinisikan Modul

Contoh ini mendefinisikan modul publik - film . Modul berisi fungsi play () yang menerima parameter dan mencetak nilainya.

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

Keluaran

Playing movie Herold and Kumar

Gunakan Kata Kunci

Kata kunci use membantu mengimpor modul publik.

Sintaksis

use public_module_name::function_name;

Ilustrasi

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

Keluaran

Playing movie Herold and Kumar

Modul Bersarang

Modul juga bisa disarangkan. The komedi modul bersarang dalam bahasa Inggris modul, yang kemudian bersarang di film modul. Contoh yang diberikan di bawah ini mendefinisikan sebuah fungsi bermain di dalam modul film / bahasa Inggris / komedi .

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());
}

Keluaran

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

Ilustrasi - Buat Peti Perpustakaan dan Konsumsi di Peti Biner

Mari kita buat peti perpustakaan bernama movie_lib, yang berisi modul movies. Untuk membangunmovie_lib peti perpustakaan, kami akan menggunakan alat tersebut cargo.

Langkah 1 - Buat folder Proyek

Buat aplikasi film folder diikuti dengan sub-folder movie-lib . Setelah folder dan sub-folder dibuat, buat filesrcfolder dan file Cargo.toml di direktori ini. Kode sumber harus masuk ke folder src . Buat file lib.rs dan film.rs di folder src. File Cargo.toml akan berisi metadata proyek seperti nomor versi, nama penulis, dll.

Struktur direktori proyek akan seperti yang ditunjukkan di bawah ini -

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

Langkah 2 - Edit file Cargo.toml untuk menambahkan metadata proyek

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

Langkah 3 - Edit file lib.rs.

Tambahkan definisi modul berikut ke file ini.

pub mod movies;

Baris di atas membuat modul publik - movies.

Langkah 4 - Edit file film.rs

File ini akan menentukan semua fungsi untuk modul film.

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

Kode di atas mendefinisikan sebuah fungsi play() yang menerima parameter dan mencetaknya ke konsol.

Langkah 5 - Bangun peti perpustakaan

Bangun aplikasi menggunakan cargo buildperintah untuk memverifikasi apakah peti perpustakaan terstruktur dengan benar. Pastikan Anda berada di root proyek - folder aplikasi film. Pesan berikut akan ditampilkan di terminal jika build berhasil.

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

Langkah 6 - Buat aplikasi pengujian

Buat folder lain movie-lib-testdi folder aplikasi film diikuti oleh file Cargo.toml dan folder src. Proyek ini harus memiliki metode utama karena ini adalah peti biner, yang akan menggunakan peti perpustakaan yang dibuat sebelumnya. Buat file main.rs di folder src. Struktur folder akan seperti yang ditunjukkan.

movie-app
   movie-lib 
   // already completed

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

Langkah 7 - Tambahkan yang berikut ini di file Cargo.toml

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

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

NOTE- Jalur ke folder perpustakaan ditetapkan sebagai dependensi. Diagram berikut menunjukkan isi dari kedua proyek tersebut.

Langkah 8 - Tambahkan berikut ini ke file main.rs

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

Kode di atas mengimpor paket eksternal yang disebut movies_lib. Periksa Cargo.toml proyek saat ini untuk memverifikasi nama peti.

Langkah 9 - Penggunaan build kargo dan kargo

Kami akan menggunakan pembangunan kargo dan menjalankan kargo untuk membangun proyek biner dan melaksanakannya seperti yang ditunjukkan di bawah ini -

Pustaka koleksi standar Rust menyediakan implementasi yang efisien dari struktur data pemrograman tujuan umum yang paling umum. Bab ini membahas implementasi dari koleksi yang umum digunakan - Vector, HashMap dan HashSet.

Vektor

Vektor adalah larik yang dapat diubah ukurannya. Ini menyimpan nilai dalam blok memori yang berdekatan. Struktur yang telah ditentukan Vec dapat digunakan untuk membuat vektor. Beberapa fitur penting dari Vector adalah -

  • Vektor dapat tumbuh atau menyusut saat runtime.

  • Vektor adalah koleksi yang homogen.

  • Vektor menyimpan data sebagai urutan elemen dalam urutan tertentu. Setiap elemen dalam Vektor diberi nomor indeks unik. Indeks dimulai dari 0 dan naik ke n-1 di mana, n adalah ukuran koleksi. Misalnya, dalam kumpulan 5 elemen, elemen pertama akan berada di indeks 0 dan elemen terakhir akan berada di indeks 4.

  • Vektor hanya akan menambahkan nilai ke (atau mendekati) akhir. Dengan kata lain, Vektor dapat digunakan untuk mengimplementasikan tumpukan.

  • Memori untuk Vektor dialokasikan di heap.

Sintaks - Membuat Vektor

let mut instance_name = Vec::new();

Metode statis new () dari struktur Vec digunakan untuk membuat instance vektor.

Sebagai alternatif, vektor juga dapat dibuat menggunakan vec! makro. Sintaksnya seperti yang diberikan di bawah ini -

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

Tabel berikut mencantumkan beberapa fungsi struktur Vec yang umum digunakan.

Sr Tidak metode Tanda Tangan & Deskripsi
1 baru()

pub fn new()->Vect

Membangun Vec baru yang kosong. Vektor tidak akan dialokasikan sampai elemen didorong ke atasnya.

2 Dorong()

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

Menambahkan elemen ke belakang koleksi.

3 menghapus()

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

Menghapus dan mengembalikan elemen pada indeks posisi dalam vektor, menggeser semua elemen setelahnya ke kiri.

4 mengandung()

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

Mengembalikan nilai benar jika irisan berisi elemen dengan nilai yang diberikan.

5 len ()

pub fn len(&self) -> usize

Mengembalikan jumlah elemen dalam vektor, juga disebut sebagai 'panjangnya'.

Ilustrasi: Membuat Vektor - new ()

Untuk membuat vektor, kami menggunakan metode statis 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);
}

Contoh di atas membuat Vektor menggunakan metode statis new () yang didefinisikan dalam struktur Vec . Fungsi push (val) menambahkan nilai yang diteruskan sebagai parameter ke koleksi. Fungsi len () mengembalikan panjang vektor.

Keluaran

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

Ilustrasi: Membuat Vektor - vec! Makro

Kode berikut membuat vektor menggunakan vec! makro. Tipe data dari vektor disimpulkan nilai pertama yang ditugaskan padanya.

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

Keluaran

[1, 2, 3]

Seperti yang disebutkan sebelumnya, vektor hanya dapat berisi nilai dengan tipe data yang sama. Cuplikan berikut akan menampilkan kesalahan [E0308]: kesalahan jenis tidak cocok .

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

Ilustrasi: push ()

Menambahkan elemen ke akhir koleksi.

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

Keluaran

[20, 30, 40]

Ilustrasi: hapus ()

Menghapus dan mengembalikan elemen pada indeks posisi dalam vektor, menggeser semua elemen setelahnya ke kiri.

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

Keluaran

[10, 30]

Ilustrasi - berisi ()

Mengembalikan nilai true jika irisan berisi elemen dengan nilai yang diberikan -

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

Keluaran

found 10
[10, 20, 30]

Ilustrasi: len ()

Mengembalikan jumlah elemen dalam vektor, juga disebut sebagai 'panjangnya'.

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

Keluaran

size of vector is :3

Mengakses nilai dari Vektor

Elemen individu dalam vektor dapat diakses menggunakan nomor indeks yang sesuai. Contoh berikut membuat iklan vektor mencetak nilai elemen pertama.

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

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

Nilai dalam vektor juga bisa diambil menggunakan referensi ke koleksi.

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);
}

Keluaran

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

HashMap

Peta adalah kumpulan pasangan nilai kunci (disebut entri). Tidak ada dua entri dalam peta yang dapat memiliki kunci yang sama. Singkatnya, peta adalah tabel pencarian. HashMap menyimpan kunci dan nilai dalam tabel hash. Entri disimpan dalam urutan sewenang-wenang. Kuncinya digunakan untuk mencari nilai di HashMap. Struktur HashMap ditentukan di filestd::collectionsmodul. Modul ini harus diimpor secara eksplisit untuk mengakses struktur HashMap.

Sintaks: Membuat HashMap

let mut instance_name = HashMap::new();

Metode statis new () dari struktur HashMap digunakan untuk membuat objek HashMap. Metode ini membuat HashMap kosong.

Fungsi HashMap yang umum digunakan dibahas di bawah ini -

Sr Tidak metode Tanda Tangan & Deskripsi
1 memasukkan()

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

Menyisipkan pasangan kunci / nilai, jika tidak ada kunci maka Tidak ada yang dikembalikan. Setelah pembaruan, nilai lama dikembalikan.

2 len ()

pub fn len(&self) -> usize

Mengembalikan jumlah elemen di peta.

3 Dapatkan()

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

Mengembalikan referensi ke nilai yang sesuai dengan kunci.

4 iter ()

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

Sebuah iterator yang mengunjungi semua pasangan nilai kunci dalam urutan arbitrer. Jenis elemen iterator adalah (& 'a K, &' a V).

5 berisi_key

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

Mengembalikan nilai benar jika peta berisi nilai untuk kunci yang ditentukan.

6 menghapus()

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

Menghapus kunci dari peta, mengembalikan kunci dan nilai yang disimpan jika kunci tersebut sebelumnya ada di peta.

Ilustrasi: insert ()

Menyisipkan pasangan kunci / nilai ke dalam HashMap.

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

Program di atas membuat HashMap dan menginisialisasinya dengan 2 pasangan nilai kunci.

Keluaran

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

Ilustrasi: len ()

Mengembalikan jumlah elemen di peta

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());
}

Contoh di atas membuat HashMap dan mencetak jumlah total elemen di dalamnya.

Keluaran

size of map is 2

Ilustrasi - get ()

Mengembalikan referensi ke nilai yang sesuai dengan kunci. Contoh berikut mengambil nilai untuk kunci KL di 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");
      }
   }
}

Keluaran

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

Ilustrasi - iter ()

Menampilkan iterator yang berisi referensi ke semua pasangan nilai kunci dalam urutan arbitrer.

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);
   }
}

Keluaran

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

Ilustrasi: contains_key ()

Mengembalikan nilai benar jika peta berisi nilai untuk kunci yang ditentukan.

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");
   }
}

Keluaran

found key

Ilustrasi: hapus ()

Menghapus kunci dari peta.

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());
}

Keluaran

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

HashSet

HashSet adalah sekumpulan nilai unik tipe T. Menambahkan dan menghapus nilai cepat, dan cepat untuk menanyakan apakah nilai yang diberikan ada di set atau tidak. Struktur HashSet didefinisikan dalam modul std :: collections. Modul ini harus diimpor secara eksplisit untuk mengakses struktur HashSet.

Sintaks: Membuat HashSet

let mut hash_set_name = HashSet::new();

Metode statis, baru , dari struktur HashSet digunakan untuk membuat HashSet. Metode ini membuat HashSet kosong.

Tabel berikut mencantumkan beberapa metode yang umum digunakan dari struktur HashSet.

Sr Tidak metode Tanda Tangan & Deskripsi
1 memasukkan()

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

Menambahkan nilai ke set. Jika himpunan tidak memiliki nilai ini, true dikembalikan jika tidak salah.

2 len ()

pub fn len(&self) -> usize

Mengembalikan jumlah elemen dalam set.

3 Dapatkan()

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

Mengembalikan referensi ke nilai dalam himpunan, jika ada yang sama dengan nilai yang diberikan.

4 iter ()

pub fn iter(&self) -> Iter

Menampilkan iterator yang mengunjungi semua elemen dalam urutan arbitrer. Jenis elemen iterator adalah & 'a T.

5 berisi_key

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

Mengembalikan nilai benar jika set berisi nilai.

6 menghapus()

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

Menghapus nilai dari himpunan. Mengembalikan nilai benar jika nilainya ada di set.

Ilustrasi - masukkan ()

Menambahkan nilai ke set. HashSet tidak menambahkan nilai duplikat ke koleksi.

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);
}

Keluaran

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

Ilustrasi: len ()

Mengembalikan jumlah elemen dalam set.

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());
}

Keluaran

size of the set is 3

Ilustrasi - iter ()

Menjalankan kembali iterator yang mengunjungi semua elemen dalam urutan arbitrer.

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);
   }
}

Keluaran

TutorialsPoint
Mohtashim
Kannan

Ilustrasi: get ()

Mengembalikan referensi ke nilai dalam himpunan, jika ada, yang sama dengan nilai yang diberikan.

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);
}

Keluaran

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

Ilustrasi - berisi ()

Mengembalikan nilai benar jika set berisi nilai.

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");
   }  
}

Keluaran

found name

Ilustrasi: hapus ()

Menghapus nilai dari himpunan.

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());
}

Keluaran

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

Di Rust, kesalahan dapat diklasifikasikan menjadi dua kategori utama seperti yang ditunjukkan pada tabel di bawah ini.

Sr Tidak Nama & Deskripsi Pemakaian
1

Recoverable

Kesalahan yang bisa ditangani

Hasil enum
2

UnRecoverable

Kesalahan yang tidak bisa ditangani

makro panik

Kesalahan yang dapat dipulihkan adalah kesalahan yang bisa diperbaiki. Sebuah program dapat mencoba kembali operasi yang gagal atau menentukan tindakan alternatif ketika menemui kesalahan yang dapat dipulihkan. Kesalahan yang dapat dipulihkan tidak menyebabkan program gagal secara tiba-tiba. Contoh kesalahan yang dapat dipulihkan adalah kesalahan File Not Found .

Kesalahan yang tidak dapat dipulihkan menyebabkan program gagal secara tiba-tiba. Sebuah program tidak dapat kembali ke keadaan normal jika terjadi kesalahan yang tidak dapat diperbaiki. Itu tidak dapat mencoba kembali operasi yang gagal atau membatalkan kesalahan. Contoh kesalahan yang tidak dapat dipulihkan adalah mencoba mengakses lokasi di luar akhir larik.

Tidak seperti bahasa pemrograman lainnya, Rust tidak memiliki pengecualian. Ini mengembalikan Hasil enum <T, E> untuk kesalahan yang dapat dipulihkan, sementara itu memanggilpanicmakro jika program mengalami kesalahan yang tidak dapat dipulihkan. The panik makro menyebabkan program untuk keluar tiba-tiba.

Makro Panik dan Kesalahan yang Tidak Dapat Dipulihkan

panik! makro memungkinkan program untuk segera dihentikan dan memberikan umpan balik kepada pemanggil program. Ini harus digunakan ketika program mencapai keadaan tidak dapat dipulihkan.

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

Dalam contoh di atas, program akan segera dihentikan saat mengalami kepanikan! makro.

Keluaran

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

Ilustrasi: panik! makro

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

Outputnya seperti yang ditunjukkan di bawah ini -

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.

Sebuah program dapat menimbulkan kepanikan! makro jika aturan bisnis dilanggar seperti yang ditunjukkan pada contoh di bawah ini -

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");
}

Contoh di atas mengembalikan kesalahan jika nilai yang diberikan ke variabel ganjil.

Keluaran

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

Hasil Enum dan Kesalahan yang Dapat Dipulihkan

Hasil Enum - <T, E> dapat digunakan untuk menangani kesalahan yang dapat dipulihkan. Ini memiliki dua varian -OK dan Err. T dan E adalah parameter tipe generik. T mewakili jenis nilai yang akan dikembalikan dalam kasus sukses dalam varian OK, dan E mewakili jenis kesalahan yang akan dikembalikan dalam kasus kegagalan dalam varian Err.

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

Mari kita pahami ini dengan bantuan contoh -

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

Program mengembalikan OK (File) jika file sudah ada dan Err (Error) jika file tidak ditemukan.

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

Sekarang mari kita lihat bagaimana menangani varian Err.

Contoh berikut menangani kesalahan yang dikembalikan saat membuka file menggunakan match pernyataan

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- Program cetakan mengakhiri satu utama acara meskipun berkas tidak ditemukan. Ini berarti program telah menangani kesalahan dengan baik.

Keluaran

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

Ilustrasi

Fungsi is_even mengembalikan kesalahan jika bilangan tersebut bukan bilangan genap. Fungsi main () menangani kesalahan ini.

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- Karena fungsi utama menangani kesalahan dengan baik, akhir dari pernyataan utama dicetak.

Keluaran

Error msg is NOT_AN_EVEN
end of main

unwrap () dan harapkan ()

Pustaka standar berisi beberapa metode pembantu yang keduanya enum - Result <T, E> dan Option <T> diimplementasikan. Anda dapat menggunakannya untuk menyederhanakan kasus kesalahan di mana Anda benar-benar tidak mengharapkan semuanya gagal. Jika berhasil dari suatu metode, fungsi "buka" digunakan untuk mengekstrak hasil sebenarnya.

Sr Tidak metode Tanda Tangan & Deskripsi
1 membuka

unwrap(self): T

Harapkan self menjadi Ok / Some dan mengembalikan nilai yang terkandung di dalamnya. Jika memangErr atau None sebaliknya, ini menimbulkan kepanikan dengan konten kesalahan yang ditampilkan.

2 mengharapkan

expect(self, msg: &str): T

Berperilaku seperti tidak terbungkus, kecuali bahwa ia mengeluarkan pesan khusus sebelum panik di samping konten kesalahan.

membuka()

Fungsi unwrap () mengembalikan hasil aktual operasi yang berhasil. Ini mengembalikan kepanikan dengan pesan kesalahan default jika operasi gagal. Fungsi ini adalah singkatan dari pernyataan kecocokan. Ini ditunjukkan pada contoh di bawah -

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

Ubah kode di atas untuk meneruskan angka ganjil ke is_even() fungsi.

Fungsi unwrap () akan panik dan mengembalikan pesan kesalahan default seperti yang ditunjukkan di bawah ini

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

mengharapkan()

Program dapat mengembalikan pesan kesalahan kustom jika terjadi kepanikan. Ini ditunjukkan pada contoh berikut -

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");
}

Fungsi ekspektasi () mirip dengan unwrap (). Satu-satunya perbedaan adalah bahwa pesan kesalahan kustom dapat ditampilkan menggunakan ekspektasi.

Keluaran

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.

Generik adalah fasilitas untuk menulis kode untuk berbagai konteks dengan tipe yang berbeda. Di Rust, generik mengacu pada parameterisasi tipe data dan sifat. Generik memungkinkan untuk menulis kode yang lebih ringkas dan bersih dengan mengurangi duplikasi kode dan memberikan keamanan tipe. Konsep Generik dapat diterapkan pada metode, fungsi, struktur, enumerasi, koleksi, dan ciri.

Itu <T> syntaxdikenal sebagai parameter tipe, digunakan untuk mendeklarasikan konstruksi generik. T mewakili tipe data apa pun.

Ilustrasi: Koleksi Generik

Contoh berikut mendeklarasikan vektor yang hanya dapat menyimpan bilangan bulat.

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

Keluaran

[20, 30, 40]

Pertimbangkan cuplikan berikut -

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);
}

Contoh di atas menunjukkan bahwa vektor tipe integer hanya dapat menyimpan nilai integer. Jadi, jika kita mencoba memasukkan nilai string ke dalam koleksi, kompilator akan mengembalikan kesalahan. Generik membuat koleksi menjadi lebih aman.

Ilustrasi: Struktur Generik

Parameter type merepresentasikan sebuah tipe, yang akan diisi kompilator nanti.

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);
}

Contoh di atas mendeklarasikan struktur umum bernama Data . Tipe <T> menunjukkan beberapa tipe data. Fungsi main () membuat dua instance - instance integer dan instance string, dari struktur.

Keluaran

value is :350
value is :Tom

Sifat

Sifat dapat digunakan untuk mengimplementasikan seperangkat perilaku (metode) standar di berbagai struktur. Sifat itu sepertiinterfacesdalam Pemrograman Berorientasi Objek. Sintaks sifatnya seperti yang ditunjukkan di bawah ini -

Nyatakan suatu Sifat

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
   }
}

Sifat dapat berisi metode konkret (metode dengan tubuh) atau metode abstrak (metode tanpa tubuh). Gunakan metode konkret jika definisi metode akan dibagikan oleh semua struktur yang menerapkan Ciri tersebut. Namun, struktur dapat memilih untuk mengganti fungsi yang ditentukan oleh sifat tersebut.

Gunakan metode abstrak jika definisi metode bervariasi untuk struktur pelaksana.

Sintaks - Menerapkan Sifat

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

Contoh-contoh berikut mendefinisikan sifat cetak dengan metode cetak () , yang dilaksanakan oleh struktur buku .

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)
   }
}

Keluaran

Printing book with id:1001 and name Rust in Action

Fungsi Generik

Contoh ini mendefinisikan fungsi generik yang menampilkan parameter yang diteruskan padanya. Parameternya bisa jenis apa saja. Jenis parameter harus mengimplementasikan sifat Tampilan sehingga nilainya dapat dicetak oleh println! makro.

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);
}

Keluaran

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

Bab ini membahas cara menerima nilai dari input standar (keyboard) dan nilai tampilan ke output standar (konsol). Dalam bab ini, kita juga akan membahas argumen baris perintah yang lewat.

Jenis Pembaca dan Penulis

Fitur pustaka standar Rust untuk input dan output diatur di sekitar dua ciri -

  • Read
  • Write
Sr Tidak Sifat & Deskripsi Contoh
1

Read

Jenis yang mengimplementasikan Baca memiliki metode untuk input berorientasi byte. Mereka disebut pembaca

Stdin, Berkas
2

Write

Jenis yang mengimplementasikan Write mendukung keluaran teks berorientasi byte dan UTF-8. Mereka disebut penulis.

Stdout, File

Baca Sifat

Readersadalah komponen tempat program Anda dapat membaca byte. Contohnya termasuk membaca input dari keyboard, file, dllread_line() metode sifat ini dapat digunakan untuk membaca data, satu baris pada satu waktu, dari file atau aliran input standar.

Sr Tidak Sifat Metode & Deskripsi
1 Baca

read_line(&mut line)->Result

Membaca sebaris teks dan menambahkannya ke baris, yang merupakan String. Nilai yang dikembalikan adalah io :: Result, jumlah byte yang dibaca.

Ilustrasi - Membaca dari Konsol - stdin ()

Program Rust mungkin harus menerima nilai dari pengguna saat runtime. Contoh berikut membaca nilai dari input standar (Keyboard) dan mencetaknya ke konsol.

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);
}

Fungsi stdin () mengembalikan pegangan ke aliran input standar dari proses saat ini, di mana fungsi read_line dapat diterapkan. Fungsi ini mencoba membaca semua karakter yang ada di buffer input ketika bertemu dengan karakter akhir baris.

Keluaran

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

Tulis Sifat

Writersadalah komponen yang program Anda dapat menulis byte. Contohnya termasuk nilai pencetakan ke konsol, menulis ke file, dll. Metode write () dari sifat ini dapat digunakan untuk menulis data ke file atau aliran keluaran standar.

Sr Tidak Sifat Metode & Deskripsi
1 Menulis

write(&buf)->Result

Menulis beberapa byte dalam slice buf ke aliran yang mendasarinya. Ia mengembalikan io :: Result, jumlah byte yang ditulis.

Ilustrasi - Menulis ke Konsol - stdout ()

The cetak! atau println! makro dapat digunakan untuk menampilkan teks di konsol. Namun, Anda juga bisa menggunakan fungsi pustaka standar write () untuk menampilkan beberapa teks ke keluaran standar.

Mari kita pertimbangkan contoh untuk memahami ini.

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();
}

Keluaran

Tutorials Point
bytes written 15

The stdout () fungsi perpustakaan standar kembali pegangan untuk standar keluaran dari proses saat ini, dimanawritefungsi dapat diterapkan. Metode write () mengembalikan enum, Result. The unwrap () adalah metode pembantu untuk mengekstrak hasil aktual dari enumerasi. Metode buka bungkus akan membuat panik jika terjadi kesalahan.

NOTE - File IO dibahas di bab berikutnya.

Argumen CommandLine

Argumen CommandLine diteruskan ke program sebelum menjalankannya. Mereka seperti parameter yang diteruskan ke fungsi. Parameter CommandLine dapat digunakan untuk meneruskan nilai ke fungsi main () . Itustd::env::args() mengembalikan argumen baris perintah.

Ilustrasi

Contoh berikut meneruskan nilai sebagai argumen commandLine ke fungsi main (). Program ini dibuat dalam nama file 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
   }
}

Program ini akan menghasilkan file main.exe setelah dikompilasi. Beberapa parameter baris perintah harus dipisahkan dengan spasi. Jalankan main.exe dari terminal sebagai main.exe halo tutorialspoint .

NOTE- halo dan tutorialspoint adalah argumen baris perintah.

Keluaran

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

Output menunjukkan 3 argumen karena main.exe adalah argumen pertama.

Ilustrasi

Program berikut menghitung jumlah nilai yang diteruskan sebagai argumen baris perintah. Sebuah daftar nilai integer dipisahkan oleh spasi diteruskan ke 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);
}

Saat menjalankan program sebagai main.exe 1 2 3 4, hasilnya adalah -

No of elements in arguments is :5
sum is 10

Selain membaca dan menulis ke konsol, Rust memungkinkan membaca dan menulis ke file.

File struct merepresentasikan sebuah file. Ini memungkinkan program untuk melakukan operasi baca-tulis pada file. Semua metode dalam file struct mengembalikan varian dari pencacahan io :: Result.

Metode yang umum digunakan dari file struct tercantum dalam tabel di bawah ini -

Sr Tidak Modul metode Tanda tangan Deskripsi
1 std :: fs :: File Buka() pub fn open <P: AsRef> (jalur: P) -> Hasil Metode statis terbuka dapat digunakan untuk membuka file dalam mode hanya-baca.
2 std :: fs :: File membuat() pub fn create <P: AsRef> (jalur: P) -> Hasil Metode statis membuka file dalam mode hanya tulis. Jika file sudah ada, konten lama dihancurkan. Jika tidak, file baru dibuat.
3 std :: fs :: remove_file remove_file () pub fn remove_file <P: AsRef> (jalur: P) -> Hasil <()> Menghapus file dari sistem file. Tidak ada jaminan bahwa file tersebut akan segera dihapus.
4 std :: fs :: OpenOptions menambahkan() pub fn tambahkan (& mut self, tambahkan: bool) -> & mut OpenOptions Menetapkan opsi untuk mode penambahan file.
5 std :: io :: Menulis write_all () fn write_all (& mut self, buf: & [u8]) -> Result <()> Mencoba menulis seluruh buffer ke dalam penulisan ini.
6 std :: io :: Baca read_to_string () fn read_to_string (& mut self, buf: & mut String) -> Hasil Membaca semua byte hingga EOF di sumber ini, menambahkannya ke buf.

Tulis ke File

Mari kita lihat contoh untuk memahami cara menulis file.

Program berikut membuat file 'data.txt'. Metode create () digunakan untuk membuat file. Metode mengembalikan pegangan file jika file berhasil dibuat. Fungsi write_all baris terakhir akan menulis byte dalam file yang baru dibuat. Jika salah satu operasi gagal, fungsi ekspektasi () mengembalikan pesan kesalahan.

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" );
}

Keluaran

data written to file

Membaca dari File

Program berikut membaca konten dalam file data.txt dan mencetaknya ke konsol. Fungsi "terbuka" digunakan untuk membuka file yang sudah ada. Path absolut atau relatif ke file diteruskan ke fungsi open () sebagai parameter. Fungsi open () melontarkan pengecualian jika file tidak ada, atau jika tidak dapat diakses karena alasan apa pun. Jika berhasil, pegangan file ke file tersebut ditugaskan ke variabel "file".

Fungsi "read_to_string" dari pegangan "file" digunakan untuk membaca konten file itu ke dalam variabel string.

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);
}

Keluaran

Hello World
TutorialsPoint

Hapus file

Contoh berikut menggunakan fungsi remove_file () untuk menghapus file. Fungsi expect () mengembalikan pesan khusus jika terjadi kesalahan.

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

Keluaran

file is removed

Tambahkan data ke file

Fungsi append () menulis data di akhir file. Ini ditunjukkan pada contoh yang diberikan di bawah ini -

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");
}

Keluaran

file append success

Salin file

Contoh berikut menyalin konten dalam file ke file baru.

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; }
   }
}

Jalankan program di atas sebagai main.exe data.txt datacopy.txt . Dua argumen baris perintah dilewatkan saat menjalankan file -

  • jalur ke file sumber
  • file tujuan

Cargo adalah manajer paket untuk RUST. Ini berfungsi seperti alat dan mengelola proyek Rust.

Beberapa perintah kargo yang umum digunakan tercantum dalam tabel di bawah ini -

Sr Tidak Perintah & Deskripsi
1

cargo build

Mengompilasi proyek saat ini.

2

cargo check

Menganalisis proyek saat ini dan melaporkan kesalahan, tetapi jangan membuat file objek.

3

cargo run

Membangun dan menjalankan src / main.rs.

4

cargo clean

Menghapus direktori target.

5

cargo update

Dependensi pembaruan yang tercantum di Cargo.lock.

6

cargo new

Membuat proyek kargo baru.

Cargo membantu mengunduh perpustakaan pihak ketiga. Oleh karena itu, ia bertindak seperti pengelola paket. Anda juga dapat membangun perpustakaan Anda sendiri. Cargo diinstal secara default saat Anda menginstal Rust.

Untuk membuat proyek kargo baru, kita dapat menggunakan perintah yang diberikan di bawah ini.

Buat peti biner

cargo new project_name --bin

Buat peti perpustakaan

cargo new project_name --lib

Untuk memeriksa versi kargo saat ini, jalankan perintah berikut -

cargo --version

Ilustrasi - Buat proyek Binary Cargo

Gim ini menghasilkan nomor acak dan meminta pengguna menebak nomor tersebut.

Langkah 1 - Buat folder proyek

Buka terminal dan ketik perintah berikut cargo new guess-game-app --bin .

Ini akan membuat struktur folder berikut.

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

Perintah kargo baru digunakan untuk membuat peti. The --bin bendera menunjukkan bahwa peti yang dibuat adalah peti biner. Krat umum disimpan di repositori pusat yang disebut crates.iohttps://crates.io/.

Langkah 2 - Sertakan referensi ke perpustakaan eksternal

Contoh ini perlu menghasilkan nomor acak. Karena pustaka standar internal tidak menyediakan logika pembuatan bilangan acak, kita perlu melihat pustaka atau peti eksternal. Mari kita gunakanrandcrate yang tersedia di situs crates.io crates.io

Itu https://crates.io/crates/randadalah perpustakaan karat untuk pembuatan nomor acak. Rand menyediakan utilitas untuk menghasilkan bilangan acak, untuk mengubahnya menjadi jenis dan distribusi yang berguna, dan beberapa algoritme terkait keacakan.

Diagram berikut menunjukkan situs crate.io dan hasil pencarian untuk rand crate.

Salin versi rand crate ke file Cargo.toml rand = "0.5.5" .

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

[dependencies]
rand = "0.5.5"

Langkah 3: Kompilasi Proyek

Arahkan ke folder proyek. Jalankan perintahnyacargo build di jendela terminal -

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

Rand crate dan semua dependensi transitif (dependensi dalam rand) akan diunduh secara otomatis.

Langkah 4 - Memahami Logika Bisnis

Sekarang mari kita lihat bagaimana logika bisnis bekerja untuk permainan menebak angka -

  • Game awalnya menghasilkan nomor acak.

  • Seorang pengguna diminta untuk memasukkan input dan menebak angkanya.

  • Jika nomor lebih kecil dari nomor yang dihasilkan, pesan "Terlalu rendah" dicetak.

  • Jika angka lebih besar dari angka yang dihasilkan, pesan "Terlalu tinggi" akan dicetak.

  • Jika pengguna memasukkan nomor yang dihasilkan oleh program, permainan keluar.

Langkah 5 - Edit file main.rs

Tambahkan logika bisnis ke file 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;
      }
   }
}

Langkah 6 - Kompilasi dan Jalankan Proyek

Jalankan kargo perintah yang dijalankan di terminal. Pastikan terminal menunjuk ke direktori Proyek.

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 ..

Di bab ini, kita akan mempelajari cara kerja iterator dan closure di RUST.

Iterator

Sebuah iterator membantu untuk melakukan iterasi atas kumpulan nilai seperti array, vektor, peta, dll. Iterator mengimplementasikan sifat Iterator yang didefinisikan di pustaka standar Rust. Metode iter () mengembalikan objek iterator dari koleksi. Nilai dalam objek iterator disebut item. Metode next () dari iterator dapat digunakan untuk melintasi item. Metode next () mengembalikan nilai Tidak Ada saat mencapai akhir pengumpulan.

Contoh berikut menggunakan iterator untuk membaca nilai dari array.

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());
}

Keluaran

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

Jika koleksi seperti array atau Vector mengimplementasikan sifat Iterator maka itu dapat dilalui menggunakan for ... dalam sintaks seperti yang ditunjukkan di bawah ini-

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

Keluaran

10 20 30

3 metode berikut mengembalikan objek iterator dari koleksi, di mana T mewakili elemen dalam koleksi.

Sr Tidak Metode & Deskripsi
1

iter()

memberikan iterator di atas & T (referensi ke T)

2

into_iter()

memberikan iterator di atas T

3

iter_mut()

memberikan iterator & mut T

Ilustrasi: iter ()

Fungsi iter () menggunakan konsep peminjaman. Ini mengembalikan referensi ke setiap elemen koleksi, membiarkan koleksi tidak tersentuh dan tersedia untuk digunakan kembali setelah pengulangan.

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
}

Keluaran

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

Ilustrasi - into_iter ()

Fungsi ini menggunakan konsep kepemilikan. Ini memindahkan nilai dalam koleksi ke objek iter, yaitu, koleksi dipakai dan tidak lagi tersedia untuk digunakan kembali.

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
}

Keluaran

Hello Kannan
There is a rustacean among us!
Hello Kiran

Ilustrasi - untuk dan iter_mut ()

Fungsi ini seperti fungsi iter () . Namun, fungsi ini dapat mengubah elemen di dalam koleksi.

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
}

Keluaran

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

Penutupan

Penutupan mengacu pada fungsi dalam fungsi lain. Ini adalah fungsi anonim - fungsi tanpa nama. Penutupan dapat digunakan untuk menetapkan fungsi ke variabel. Ini memungkinkan program untuk melewatkan fungsi sebagai parameter ke fungsi lain. Penutupan juga dikenal sebagai fungsi sebaris. Variabel di fungsi luar dapat diakses dengan fungsi sebaris.

Sintaks: Mendefinisikan Penutupan

Definisi penutupan secara opsional dapat memiliki parameter. Parameter diapit dalam dua batang vertikal.

let closure_function = |parameter| {
   //logic
}

Sintaks yang meminta penerapan Penutupan Fnsifat. Jadi, bisa dipanggil dengan() sintaksis.

closure_function(parameter);    //invoking

Ilustrasi

Contoh berikut mendefinisikan closure is_even dalam fungsi main () . Closure mengembalikan nilai true jika sebuah bilangan genap dan mengembalikan false jika jumlahnya ganjil.

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

Keluaran

13 is even ? false

Ilustrasi

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

Fungsi main () mendeklarasikan val variabel dan penutupan. Closure mengakses variabel yang dideklarasikan di fungsi luar main () .

Keluaran

12

Rust mengalokasikan semua yang ada di tumpukan secara default. Anda dapat menyimpan berbagai hal di heap dengan membungkusnya dengan petunjuk cerdas seperti Box . Jenis seperti Vec dan String secara implisit membantu alokasi heap. Pointer cerdas menerapkan ciri-ciri yang tercantum dalam tabel di bawah ini. Ciri-ciri dari petunjuk pintar ini membedakan mereka dari struct biasa -

Sr Tidak Nama sifat Paket & Deskripsi
1 Deref

std::ops::Deref

Digunakan untuk operasi dereferensi yang tidak dapat diubah, seperti * v.

2 Penurunan

std::ops::Drop

Digunakan untuk menjalankan beberapa kode ketika nilai keluar dari ruang lingkup. Ini kadang-kadang disebut destruktor

Dalam bab ini, kita akan mempelajari tentang Boxpenunjuk cerdas. Kami juga akan belajar cara membuat penunjuk pintar khusus seperti Box.

Kotak

Pointer cerdas Box yang juga disebut kotak memungkinkan Anda menyimpan data di heap, bukan di stack. Tumpukan berisi penunjuk ke data heap. Box tidak memiliki overhead performa, selain menyimpan datanya di heap.

Mari kita lihat cara menggunakan kotak untuk menyimpan nilai i32 di heap.

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

Keluaran

b = 5

Untuk mengakses nilai yang ditunjukkan oleh variabel, gunakan dereferencing. * Digunakan sebagai operator dereferensi. Mari kita lihat bagaimana menggunakan dereferensi dengan 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
}

Variabel x adalah tipe nilai dengan nilai 5. Jadi, ekspresi 5 == x akan mengembalikan true. Variabel y menunjuk ke heap. Untuk mengakses nilai di heap, kita perlu melakukan dereferensi menggunakan * y. * y mengembalikan nilai 5. Jadi, ekspresi 5 == * y mengembalikan nilai true.

Keluaran

true
true

Ilustrasi - Sifat Deref

Sifat Deref, disediakan oleh pustaka standar, mengharuskan kita untuk mengimplementasikan satu metode bernama deref , yang meminjam diri dan mengembalikan referensi ke data dalam. Contoh berikut membuat struktur MyBox , yang merupakan tipe generik. Ini mengimplementasikan sifat Deref . Sifat ini membantu kita mengakses nilai heap yang dibungkus oleh y menggunakan * 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
}

Keluaran

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

Ilustrasi - Jatuhkan Sifat

Sifat Drop berisi metode drop () . Metode ini dipanggil ketika struktur yang menerapkan sifat ini keluar dari ruang lingkup. Dalam beberapa bahasa, pemrogram harus memanggil kode untuk mengosongkan memori atau sumber daya setiap kali mereka selesai menggunakan contoh penunjuk cerdas. Di Rust, Anda dapat mencapai deallokasi memori otomatis menggunakan sifat Jatuhkan.

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");
}

Dalam contoh di atas, metode drop akan dipanggil dua kali saat kita membuat dua objek di heap.

dropping MyBox object from memory
dropping MyBox object from memory

Dalam pemrograman bersamaan, berbagai bagian program dijalankan secara independen. Di sisi lain, dalam pemrograman paralel, berbagai bagian program dijalankan pada waktu yang bersamaan. Kedua model sama pentingnya karena lebih banyak komputer memanfaatkan beberapa prosesor mereka.

Benang

Kita dapat menggunakan utas untuk menjalankan kode secara bersamaan. Dalam sistem operasi saat ini, kode program yang dijalankan dijalankan dalam suatu proses, dan sistem operasi mengelola banyak proses sekaligus. Dalam program Anda, Anda juga dapat memiliki bagian independen yang berjalan secara bersamaan. Fitur yang menjalankan bagian independen ini disebut utas.

Membuat Thread

Itu thread::spawnfungsi digunakan untuk membuat utas baru. Fungsi spawn mengambil closure sebagai parameter. Penutupan mendefinisikan kode yang harus dijalankan oleh utas. Contoh berikut mencetak beberapa teks dari utas utama dan teks lain dari utas baru.

//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));
   }
}

Keluaran

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!

Utas utama mencetak nilai dari 1 hingga 4.

NOTE- Utas baru akan dihentikan saat utas utama berakhir. Output dari program ini mungkin akan sedikit berbeda setiap saat.

Itu thread::sleepfungsi memaksa utas menghentikan eksekusinya untuk durasi singkat, memungkinkan utas lain untuk berjalan. Utas mungkin akan bergiliran, tetapi itu tidak dijamin - itu tergantung pada bagaimana sistem operasi menjadwalkan utas. Dalam proses ini, utas utama dicetak terlebih dahulu, meskipun pernyataan cetak dari utas yang muncul muncul pertama kali dalam kode. Selain itu, bahkan jika utas yang muncul diprogram untuk mencetak nilai hingga 9, itu hanya sampai 5 sebelum utas utama ditutup.

Bergabunglah dengan Handles

Utas yang muncul mungkin tidak memiliki kesempatan untuk dijalankan atau dijalankan sepenuhnya. Ini karena utas utama selesai dengan cepat. Fungsi menelurkan <F, T> (f: F) -> JoinHandlelt; T> mengembalikan JoinHandle. Metode join () di JoinHandle menunggu utas terkait selesai.

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();
}

Keluaran

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!

Utas utama dan utas yang muncul terus beralih.

NOTE - Utas utama menunggu utas yang muncul selesai karena panggilan ke join() metode.