जंग - त्वरित गाइड

रस्ट एक सिस्टम स्तर की प्रोग्रामिंग भाषा है, जिसे ग्रेडॉन होरे द्वारा विकसित किया गया है। मोज़िला लैब्स ने बाद में कार्यक्रम का अधिग्रहण किया।

अनुप्रयोग v / s सिस्टम प्रोग्रामिंग भाषाएँ

जावा / सी # जैसी एप्लिकेशन प्रोग्रामिंग भाषाएं सॉफ्टवेयर बनाने के लिए उपयोग की जाती हैं, जो सीधे उपयोगकर्ता को सेवाएं प्रदान करती हैं। वे स्प्रेडशीट, वर्ड प्रोसेसर, वेब एप्लिकेशन या मोबाइल एप्लिकेशन जैसे व्यावसायिक एप्लिकेशन बनाने में हमारी सहायता करते हैं।

सॉफ्टवेयर / सॉफ्टवेयर प्लेटफॉर्म बनाने के लिए C / C ++ जैसी सिस्टम प्रोग्रामिंग भाषाओं का उपयोग किया जाता है। उनका उपयोग ऑपरेटिंग सिस्टम, गेम इंजन, कंपाइलर इत्यादि के निर्माण के लिए किया जा सकता है। इन प्रोग्रामिंग लैंग्वेज में हार्डवेयर इंटरैक्शन की एक बड़ी डिग्री की आवश्यकता होती है।

सिस्टम और एप्लिकेशन प्रोग्रामिंग भाषाएं दो प्रमुख समस्याओं का सामना करती हैं -

  • सुरक्षित कोड लिखना मुश्किल है।
  • मल्टी-थ्रेडेड कोड लिखना मुश्किल है।

जंग क्यों?

जंग तीन लक्ष्यों पर केंद्रित है -

  • Safety
  • Speed
  • Concurrency

भाषा को एक सरल तरीके से अत्यधिक विश्वसनीय और तेज़ सॉफ़्टवेयर विकसित करने के लिए डिज़ाइन किया गया था। हार्डवेयर-विशिष्ट कार्यक्रमों के लिए उच्च-स्तरीय कार्यक्रमों को लिखने के लिए जंग का उपयोग किया जा सकता है।

प्रदर्शन

रस्ट प्रोग्रामिंग लैंग्वेज में डिज़ाइन द्वारा गारबेज कलेक्टर (GC) नहीं है। यह रनटाइम पर प्रदर्शन में सुधार करता है।

संकलन के समय मेमोरी सुरक्षा

रस्ट का उपयोग करके बनाया गया सॉफ्टवेयर मेमोरी पॉइंट्स जैसे डैंगलिंग पॉइंटर्स, बफर ओवररन और मेमोरी लीक से सुरक्षित है।

बहु थ्रेडेड अनुप्रयोग

रस्ट के स्वामित्व और मेमोरी सुरक्षा नियम डेटा दौड़ के बिना संगामिति प्रदान करते हैं।

वेब असेंबली के लिए समर्थन (WASM)

वेब असेंबली ब्राउज़र में उच्च संगणना गहन एल्गोरिदम को अंजाम देने में मदद करता है, एम्बेडेड उपकरणों पर, या कहीं और। यह देशी कोड की गति से चलता है। तेजी से, विश्वसनीय निष्पादन के लिए वेब असेंबली में जंग को संकलित किया जा सकता है।

जंग की स्थापना के माध्यम से आसान बना दिया है rustup, जंग संस्करणों और संबंधित उपकरणों के प्रबंधन के लिए एक कंसोल-आधारित उपकरण।

विंडोज पर स्थापना

आइए जानें कि विंडोज पर RUST कैसे इंस्टॉल करें।

  • विजुअल स्टूडियो की स्थापना 2013 में या C ++ टूल के साथ विंडोज़ पर रस्ट प्रोग्राम चलाने के लिए अनिवार्य है। सबसे पहले, वीएस 2013 एक्सप्रेस यहां से विजुअल स्टूडियो डाउनलोड करें

  • डाउनलोड करो और इंस्टॉल करो rustup विंडोज़ के लिए उपकरण। rustup-init.exeयहाँ डाउनलोड के लिए उपलब्ध है - जंग लैंग

  • डबल क्लिक करें rustup-init.exeफ़ाइल। क्लिक करने पर, निम्न स्क्रीन दिखाई देगी।

  • डिफ़ॉल्ट स्थापना के लिए एंटर दबाएं । एक बार इंस्टॉलेशन पूरा हो जाने के बाद, निम्न स्क्रीन दिखाई देती है।

  • स्थापना स्क्रीन से, यह स्पष्ट है कि रस्ट संबंधित फाइलें फ़ोल्डर में संग्रहीत हैं -

    सी:। \ उपयोगकर्ता \ {पीसी} \ कार्गो \ बिन

फ़ोल्डर की सामग्री हैं -

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
  • CargoRust के लिए पैकेज मैनेजर है। सत्यापित करने के लिए यदिcargo स्थापित किया गया है, निम्नलिखित कमांड निष्पादित करें -

C:\Users\Admin>cargo -V
cargo 1.29.0 (524a578d7 2018-08-05)
  • जंग के लिए संकलक है rustc। संकलक संस्करण को सत्यापित करने के लिए, निम्नलिखित कमांड निष्पादित करें -

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

लिनक्स / मैक पर स्थापना

स्थापित करने के लिए rustup लिनक्स या मैकओएस पर, एक टर्मिनल खोलें और निम्न कमांड दर्ज करें।

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

कमांड एक स्क्रिप्ट डाउनलोड करता है और की स्थापना शुरू करता है rustupउपकरण, जो रस्ट के नवीनतम स्थिर संस्करण को स्थापित करता है। आपको अपने पासवर्ड के लिए संकेत दिया जा सकता है। यदि स्थापना सफल है, तो निम्न पंक्ति दिखाई देगी -

Rust is installed now. Great!

इंस्टॉलेशन स्क्रिप्ट स्वचालित रूप से आपके अगले लॉगिन के बाद आपके सिस्टम PATH में Rust जोड़ती है। अपने टर्मिनल को फिर से शुरू करने के बजाय तुरंत रस्ट का उपयोग करना शुरू करने के लिए, अपने सिस्टम में Rust जोड़ने के लिए अपने शेल में निम्न कमांड चलाएँ: PHH -

$ source $HOME/.cargo/env

वैकल्पिक रूप से, आप निम्न लाइन को अपने ~ / .bash_profile पर जोड़ सकते हैं -

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

NOTE - जब आप किसी रस्ट प्रोग्राम को संकलित करने का प्रयास करते हैं और त्रुटियां प्राप्त करते हैं, यह दर्शाता है कि एक लिंकर निष्पादित नहीं कर सकता है, तो इसका मतलब है कि आपके सिस्टम पर एक लिंकर स्थापित नहीं है और आपको मैन्युअल रूप से एक स्थापित करने की आवश्यकता होगी।

RUST के लिए ट्यूटोरियल पॉइंट कोडिंग ग्राउंड का उपयोग करना

एक पढ़ें-मूल्यांकन-प्रिंट लूप (आरईपीएल) कंप्यूटर प्रोग्राम को संकलित करने और निष्पादित करने के लिए इंटरैक्टिव शेल का उपयोग करना आसान है। यदि आप ब्राउज़र के भीतर रस्ट कार्यक्रमों को ऑनलाइन संकलित और निष्पादित करना चाहते हैं, तो ट्यूटोरियलस्पॉटिंग कोडिंग ग्राउंड का उपयोग करें ।

यह अध्याय मूल भाषा के मूल वाक्य को एक के माध्यम से समझाता है HelloWorld उदाहरण।

  • बनाओ HelloWorld-App फ़ोल्डर और टर्मिनल पर उस फ़ोल्डर में नेविगेट करें

C:\Users\Admin>mkdir HelloWorld-App
C:\Users\Admin>cd HelloWorld-App
C:\Users\Admin\HelloWorld-App>
  • रस्ट फ़ाइल बनाने के लिए, निम्न कमांड निष्पादित करें -

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

जंग कार्यक्रम फाइलों में एक एक्सटेंशन .rs है। उपरोक्त कमांड एक खाली फाइल बनाता हैHello.rsऔर इसे नोटपैड में खोलता है। इस फ़ाइल के नीचे दिया गया कोड जोड़ें -

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

उपरोक्त कार्यक्रम एक फ़ंक्शन मुख्य fn मुख्य () को परिभाषित करता है । Fn कीवर्ड एक समारोह को परिभाषित करने के लिए किया जाता है। मुख्य () एक पूर्वनिर्धारित समारोह है कि कार्यक्रम के लिए एक प्रवेश बिंदु के रूप में कार्य करता है। println! जंग में एक पूर्वनिर्धारित मैक्रो है। इसका उपयोग कंसोल को एक स्ट्रिंग (यहां हैलो) प्रिंट करने के लिए किया जाता है। मैक्रो कॉल हमेशा एक विस्मयादिबोधक चिह्न के साथ चिह्नित होते हैं - !

  • संकलन करें Hello.rs फ़ाइल का उपयोग कर rustc

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

कार्यक्रम के सफल संकलन पर, एक निष्पादन योग्य फ़ाइल ( file_name.exe ) उत्पन्न होती है। यदि .exe फ़ाइल जनरेट किया गया है, तो सत्यापित करने के लिए , निम्न आदेश निष्पादित करें।

C:\Users\Admin\HelloWorld-App>dir
//lists the files in folder
Hello.exe
Hello.pdb
Hello.rs
  • Hello.exe फ़ाइल को निष्पादित करें और आउटपुट को सत्यापित करें।

स्थूल क्या है?

जंग एक शक्तिशाली मैक्रो सिस्टम प्रदान करता है जो मेटा-प्रोग्रामिंग की अनुमति देता है। जैसा कि आपने पिछले उदाहरण में देखा है, मैक्रोज़ फ़ंक्शंस की तरह दिखते हैं, सिवाय इसके कि उनका नाम धमाकेदार (!) के साथ समाप्त होता है, लेकिन एक फ़ंक्शन कॉल उत्पन्न करने के बजाय, मैक्रोज़ को स्रोत कोड में विस्तारित किया जाता है जो कि बाकी प्रोग्राम के साथ संकलित हो जाता है। इसलिए, वे फ़ंक्शन के विपरीत प्रोग्राम के लिए अधिक रनटाइम सुविधाएँ प्रदान करते हैं। मैक्रोज़ फ़ंक्शंस का एक विस्तारित संस्करण है।

Println का उपयोग करना! मैक्रो - सिंटैक्स

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

रस्ट में टिप्पणियाँ

टिप्पणियाँ एक कार्यक्रम की पठनीयता में सुधार करने का एक तरीका है। टिप्पणियों का उपयोग किसी प्रोग्राम के बारे में अतिरिक्त जानकारी शामिल करने के लिए किया जा सकता है जैसे कोड के लेखक, किसी फ़ंक्शन / निर्माण के बारे में संकेत आदि। संकलक टिप्पणियों की उपेक्षा करता है।

रूस्त निम्नलिखित प्रकार की टिप्पणियों का समर्थन करता है -

  • एकल-पंक्ति टिप्पणियाँ (//) - एक पंक्ति के अंत और / के बीच के किसी भी पाठ को एक टिप्पणी के रूप में माना जाता है

  • मल्टी-लाइन टिप्पणियां (/ * * /) - इन टिप्पणियों में कई लाइनें हो सकती हैं।

उदाहरण

//this is single line comment

/* This is a
   Multi-line comment
*/

ऑनलाइन परीक्षा दें

ट्यूटोरियलस्पॉटिंग कोडिंग ग्राउंड के माध्यम से रस्ट कार्यक्रमों को ऑनलाइन निष्पादित किया जा सकता है । संपादक टैब में HelloWorld प्रोग्राम लिखें और परिणाम देखने के लिए निष्पादन पर क्लिक करें।

टाइप सिस्टम भाषा द्वारा समर्थित विभिन्न प्रकार के मूल्यों का प्रतिनिधित्व करता है। टाइप सिस्टम आपूर्ति किए गए मूल्यों की वैधता की जांच करता है, इससे पहले कि वे प्रोग्राम द्वारा संग्रहीत या हेरफेर किए जाते हैं। यह सुनिश्चित करता है कि कोड अपेक्षित रूप से व्यवहार करता है। टाइप सिस्टम आगे अमीर कोड संकेत और स्वचालित प्रलेखन के लिए भी अनुमति देता है।

जंग एक स्टेटिकली टाइप्ड भाषा है। Rust में प्रत्येक मान एक निश्चित डेटा प्रकार का होता है। कंपाइलर स्वचालित रूप से इसके द्वारा दिए गए मूल्य के आधार पर चर के डेटा प्रकार का अनुमान लगा सकता है।

एक चर घोषित करें

उपयोग let एक चर घोषित करने के लिए कीवर्ड।

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

उपरोक्त उदाहरण में, चर के डेटा प्रकार को उनके द्वारा निर्दिष्ट मूल्यों से अनुमान लगाया जाएगा। उदाहरण के लिए, रूस्ट वैरिएबल company_string को स्ट्रिंग डेटा टाइप , रेटिंग डेटा टाइप को रेटिंग_फ्लोट , आदि के लिए असाइन करेगा

Println! स्थूल में दो तर्क होते हैं -

  • एक विशेष वाक्यविन्यास {} , जो प्लेसहोल्डर है
  • चर नाम या एक स्थिर

प्लेसहोल्डर को वैरिएबल के मान से बदल दिया जाएगा

उपरोक्त कोड स्निपेट का उत्पादन होगा -

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

स्केलर प्रकार

एक स्केलर प्रकार एक एकल मान का प्रतिनिधित्व करता है। उदाहरण के लिए, 10,3.14, 'सी'। जंग में चार प्राथमिक स्केलर प्रकार होते हैं।

  • Integer
  • Floating-point
  • Booleans
  • Characters

हम अपने बाद के अनुभागों में प्रत्येक प्रकार के बारे में जानेंगे।

पूर्णांक

पूर्णांक एक अंश के बिना एक संख्या है। सीधे शब्दों में कहें, पूर्णांक डेटा प्रकार का उपयोग संपूर्ण संख्याओं को दर्शाने के लिए किया जाता है।

इंटेगर को आगे हस्ताक्षरित और अनसाइनड के रूप में वर्गीकृत किया जा सकता है। हस्ताक्षर किए गए पूर्णांक नकारात्मक और सकारात्मक दोनों मूल्यों को संग्रहीत कर सकते हैं। अहस्ताक्षरित पूर्णांक केवल सकारात्मक मान संग्रहीत कर सकते हैं। पूर्णांक प्रकार नीचे दिए गए हैं, तो एक विस्तृत विवरण -

अनु क्रमांक। आकार पर हस्ताक्षर किए अहस्ताक्षरित
1 8 बिट i8 u8
2 16 बिट i16 यू 16
3 32 बिट i32 u32
4 64 बिट i64 u64
5 128 बिट I128 u128
6 मेहराब isize usize

पूर्णांक का आकार आर्च हो सकता है । इसका अर्थ है कि डेटा प्रकार का आकार मशीन की वास्तुकला से लिया जाएगा । एक पूर्णांक जिसका आकार आर्क है, एक x86 मशीन पर 32 बिट्स और 64 बिट्स x64 मशीन पर होगा। एक आर्क पूर्णांक मुख्य रूप से किसी प्रकार के संग्रह को अनुक्रमित करते समय उपयोग किया जाता है।

चित्रण

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

आउटपुट नीचे दिया जाएगा -

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

यदि आप आयु के मान को फ़्लोटिंग-पॉइंट मान से बदलते हैं तो उपरोक्त कोड एक संकलन त्रुटि देगा ।

इंटीजर रेंज

प्रत्येक हस्ताक्षरित संस्करण संख्याओं को स्टोर कर सकते हैं - (2 ^ (n-1) से 2 ^ (n-1) -1 , जहाँ n बिट्स की संख्या है जिसका भिन्न प्रकार उपयोग करता है। उदाहरण के लिए, i8 से संख्याएँ संग्रहीत कर सकते हैं - (2 ^) 7) से 2 ^ 7 -1 - यहां हमने n को 8 से बदल दिया ।

प्रत्येक अहस्ताक्षरित संस्करण 0 से (2 ^ n) -1 तक संख्याओं को संग्रहीत कर सकता है । उदाहरण के लिए, u8 0 से 2 ^ 7 तक संख्याओं को संग्रहीत कर सकता है , जो 0 से 255 के बराबर है।

पूर्णांक अतिप्रवाह

पूर्णांक ओवरफ़्लो तब होता है जब पूर्णांक चर के लिए दिया गया मान डेटा प्रकार के लिए जंग परिभाषित सीमा से अधिक होता है। आइए इसे एक उदाहरण से समझते हैं -

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

अहस्ताक्षरित u8 चर की मान्य सीमा 0 से 255 है। उपरोक्त उदाहरण में, चर 255 से अधिक मान दिए गए हैं (रस्ट में पूर्णांक चर के लिए ऊपरी सीमा)। निष्पादन पर, उपरोक्त कोड एक चेतावनी लौटाएगा -warning − literal out of range for u8वजन, ऊंचाई और स्कोर चर के लिए। 255 के बाद अतिप्रवाह मान 0, 1, 2, आदि से शुरू होगा। चेतावनी के बिना अंतिम आउटपुट नीचे दिखाया गया है -

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

फ्लोट

जंग में फ्लोट डेटा प्रकार को वर्गीकृत किया जा सकता है f32 तथा f64। F32 प्रकार एक एकल-सटीक फ्लोट है, और F64 में दोहरी परिशुद्धता है। डिफ़ॉल्ट प्रकार f64 है। फ्लोट डेटा प्रकार के बारे में अधिक समझने के लिए निम्न उदाहरण पर विचार करें।

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

आउटपुट नीचे दिखाया गया है -

interest is 8.35
cost is 15000.6

स्वचालित प्रकार की कास्टिंग

रस्ट में स्वचालित प्रकार की कास्टिंग की अनुमति नहीं है। निम्नलिखित कोड स्निपेट पर विचार करें। एक पूर्णांक मान फ़्लोट चर को सौंपा गया हैinterest

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

संकलक एक फेंकता है mismatched types error जैसा कि नीचे दिया गया है।

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)

नंबर सेपरेटर

बड़ी संख्या की आसान पठनीयता के लिए, हम अलग-अलग अंकों के लिए एक दृश्य विभाजक _ अंडरस्कोर का उपयोग कर सकते हैं। कि 50,000 को 50_000 लिखा जा सकता है। यह नीचे दिए गए उदाहरण में दिखाया गया है।

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

आउटपुट नीचे दिया गया है -

float value 11000.555001
int value 50000

बूलियन

बूलियन प्रकार के दो संभावित मूल्य हैं - सच या गलत । उपयोगbool बूलियन चर घोषित करने के लिए कीवर्ड।

चित्रण

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

उपरोक्त कोड का आउटपुट होगा -

Is Rust Programming Fun ? true

चरित्र

Rust में वर्ण डेटा प्रकार संख्याओं, वर्णमालाओं, यूनिकोड और विशेष वर्णों का समर्थन करता है। उपयोगcharकीवर्ड को वर्ण डेटा प्रकार का एक चर घोषित करने के लिए। रस्ट का चार प्रकार एक यूनिकोड स्केलर मान का प्रतिनिधित्व करता है, जिसका अर्थ है कि यह सिर्फ ASCII से बहुत अधिक प्रतिनिधित्व कर सकता है। यूनिकोड स्केलर वैल्यू से लेकरU+0000 सेवा U+D7FF तथा U+E000 सेवा U+10FFFF समावेशी।

हमें चरित्र डेटा प्रकार के बारे में अधिक समझने के लिए एक उदाहरण पर विचार करें।

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

उपरोक्त कोड का आउटपुट होगा -

special character is @
alphabet is A
emoji is

एक वैरिएबल एक नामित भंडारण है जिसे प्रोग्राम हेरफेर कर सकते हैं। सीधे शब्दों में कहें तो एक वैरिएबल प्रोग्राम को वैल्यू स्टोर करने में मदद करता है। जंग में चर एक विशिष्ट डेटा प्रकार के साथ जुड़े हुए हैं। डेटा प्रकार चर की मेमोरी के आकार और लेआउट को निर्धारित करता है, मूल्यों की श्रेणी जो उस मेमोरी के भीतर संग्रहीत की जा सकती है और संचालन का सेट जो चर पर किया जा सकता है।

एक परिवर्तनीय नामकरण के लिए नियम

इस खंड में, हम एक चर के नामकरण के विभिन्न नियमों के बारे में जानेंगे।

  • एक चर का नाम अक्षरों, अंकों और अंडरस्कोर वर्ण से बना हो सकता है।

  • यह या तो एक पत्र या एक अंडरस्कोर से शुरू होना चाहिए।

  • ऊपरी और निचले अक्षर अलग-अलग हैं क्योंकि जंग केस-संवेदी है।

वाक्य - विन्यास

Rust में वैरिएबल घोषित करते समय डेटा प्रकार वैकल्पिक है। डेटा प्रकार वैरिएबल को दिए गए मान से अनुमानित है।

एक चर घोषित करने का सिंटैक्स नीचे दिया गया है।

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

चित्रण

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

उपरोक्त कोड का आउटपुट होगा fees is 25000 and salary is 35000

अडिग

डिफ़ॉल्ट रूप से, चर अपरिवर्तनीय हैं - केवल जंग में पढ़ें। दूसरे शब्दों में, चर का मान एक चर नाम से बंधे होने के बाद बदला नहीं जा सकता।

इसे एक उदाहरण से समझते हैं।

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

आउटपुट नीचे दिखाया गया है -

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)

त्रुटि संदेश त्रुटि के कारण को इंगित करता है - आप अपरिवर्तनीय चर शुल्क के लिए दो बार मान असाइन नहीं कर सकते हैं। यह उन कई तरीकों में से एक है जो रस्ट प्रोग्रामर को कोड लिखने की अनुमति देता है और सुरक्षा और आसान संगामिति का लाभ उठाता है।

परिवर्तनशील

चर डिफ़ॉल्ट रूप से अपरिवर्तनीय हैं। साथ चर नाम उपसर्गmutखोजशब्द इसे परिवर्तनशील बनाने के लिए। एक परिवर्तनशील परिवर्तनशील के मान को बदला जा सकता है।

एक परिवर्तनशील चर घोषित करने के लिए सिंटैक्स निम्नानुसार है -

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

स्निपेट का आउटपुट नीचे दिया गया है -

fees is 25000
fees changed is 35000

निरंतरता उन मूल्यों का प्रतिनिधित्व करती है जिन्हें बदला नहीं जा सकता। यदि आप एक स्थिरांक घोषित करते हैं तो इसके मूल्य में परिवर्तन का कोई तरीका नहीं है। स्थिरांक का उपयोग करने के लिए कीवर्ड हैconst। लगातार स्पष्ट रूप से टाइप किया जाना चाहिए। स्थिरांक घोषित करने के लिए सिंटैक्स निम्नलिखित है।

const VARIABLE_NAME:dataType = value;

जंग लगातार नामकरण कन्वेंशन

कॉन्स्टेंट के लिए नामकरण सम्मेलन चर के समान है। एक स्थिर नाम के सभी वर्ण आमतौर पर अपरकेस में हैं। चर घोषित करने के विपरीत,let कीवर्ड का उपयोग निरंतर घोषित करने के लिए नहीं किया जाता है।

हमने नीचे उदाहरण में जंग में स्थिरांक का उपयोग किया है -

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
}

लगातार v / s चर

इस खंड में, हम स्थिरांक और चर के बीच के विभेदक कारकों के बारे में जानेंगे।

  • का उपयोग करते हुए लगातार घोषित किए जाते हैं const कीवर्ड जब चर का उपयोग कर घोषित किए जाते हैं let कीवर्ड।

  • एक परिवर्तनीय घोषणा में वैकल्पिक रूप से एक डेटा प्रकार हो सकता है जबकि एक निरंतर घोषणा में डेटा प्रकार निर्दिष्ट करना चाहिए। इसका मतलब यह है कि USER_LIMIT = 100 के परिणाम में त्रुटि होगी।

  • का उपयोग कर घोषित एक चर letकीवर्ड डिफ़ॉल्ट रूप से अपरिवर्तनीय है। हालाँकि, आपके पास विकल्प का उपयोग करके इसे म्यूट करने का विकल्प हैmutकीवर्ड। स्थिरांक अपरिवर्तनीय हैं।

  • स्थिरांक केवल एक स्थिर अभिव्यक्ति पर सेट किया जा सकता है और फ़ंक्शन कॉल या किसी अन्य मूल्य के परिणाम के लिए नहीं जो क्रम में गणना की जाएगी।

  • ग्लोबल स्कोप सहित किसी भी दायरे में स्थिरांक घोषित किए जा सकते हैं, जो उन्हें उन मूल्यों के लिए उपयोगी बनाता है जिनके बारे में कोड के कई हिस्सों को जानना आवश्यक है।

चर और स्थिरांक का छायांकन

जंग प्रोग्रामर को एक ही नाम के साथ चर घोषित करने की अनुमति देती है। ऐसे मामले में, नया चर पिछले चर को ओवरराइड करता है।

इसे एक उदाहरण से समझते हैं।

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

उपरोक्त कोड नाम वेतन द्वारा दो चर घोषित करता है। पहली घोषणा को एक एसओबी सौंपा गया है जबकि दूसरी घोषणा को 1.50 का मूल्य दिया गया है। दूसरा चर छाया या आउटपुट प्रदर्शित करते समय पहले चर को छुपाता है।

उत्पादन

The value of salary is :1.50

जंग छाया करते समय विभिन्न डेटा प्रकारों के साथ चर का समर्थन करता है।

निम्नलिखित उदाहरण पर विचार करें।

कोड नाम से दो चर घोषित करता है uname। पहली घोषणा को एक स्ट्रिंग मान दिया जाता है, जबकि दूसरी घोषणा को एक पूर्णांक सौंपा जाता है। लेन फ़ंक्शन एक स्ट्रिंग मान में वर्णों की कुल संख्या लौटाता है।

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

उत्पादन

name changed to integer: 9

चरों के विपरीत, स्थिरांक को छायांकित नहीं किया जा सकता है। यदि उपरोक्त प्रोग्राम में चर को स्थिरांक के साथ बदल दिया जाता है, तो संकलक एक त्रुटि फेंक देगा।

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

जंग में स्ट्रिंग डेटा प्रकार को निम्नलिखित में वर्गीकृत किया जा सकता है -

  • शाब्दिक स्ट्रिंग(&str)

  • स्ट्रिंग वस्तु(String)

शाब्दिक स्ट्रिंग

स्ट्रिंग शाब्दिक (और str) का उपयोग तब किया जाता है जब एक स्ट्रिंग का मूल्य संकलन समय पर जाना जाता है। स्ट्रिंग शाब्दिक वर्णों का एक सेट है, जो एक चर में हार्डकोड किया जाता है। उदाहरण के लिए, कंपनी = "ट्यूटोरियल पॉइंट" दें । स्ट्रिंग शाब्दिक मॉड्यूल std :: str में पाए जाते हैं। स्ट्रिंग शाब्दिक भी स्ट्रिंग स्लाइस के रूप में जाना जाता है।

निम्नलिखित उदाहरण दो स्ट्रिंग शाब्दिक घोषित करता है - कंपनी और स्थान

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

स्ट्रिंग शाब्दिक डिफ़ॉल्ट रूप से स्थिर होते हैं। इसका मतलब है कि पूरे कार्यक्रम की अवधि के लिए स्ट्रिंग शाब्दिक मान्य हैं। हम चर को स्पष्ट रूप से स्थिर के रूप में निर्दिष्ट कर सकते हैं जैसा कि नीचे दिखाया गया है -

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

उपरोक्त कार्यक्रम निम्न आउटपुट उत्पन्न करेगा -

company is : TutorialsPoint location :Hyderabad

स्ट्रिंग वस्तु

स्ट्रिंग ऑब्जेक्ट प्रकार मानक लाइब्रेरी में प्रदान किया गया है। स्ट्रिंग शाब्दिक के विपरीत, स्ट्रिंग ऑब्जेक्ट प्रकार कोर भाषा का एक हिस्सा नहीं है। इसे मानक लाइब्रेरी पब स्ट्रक्चर स्ट्रिंग में सार्वजनिक संरचना के रूप में परिभाषित किया गया है । स्ट्रिंग एक संग्रहणीय संग्रह है। यह उत्परिवर्तनीय और UTF-8 एन्कोडेड प्रकार है। Stringऑब्जेक्ट प्रकार का उपयोग स्ट्रिंग मानों का प्रतिनिधित्व करने के लिए किया जा सकता है जो रनटाइम पर प्रदान किए जाते हैं। स्ट्रिंग में स्टिंग ऑब्जेक्ट आवंटित किया गया है।

वाक्य - विन्यास

स्ट्रिंग ऑब्जेक्ट बनाने के लिए, हम निम्नलिखित सिंटैक्स में से किसी का उपयोग कर सकते हैं -

String::new()

उपरोक्त सिंटैक्स एक रिक्त स्ट्रिंग बनाता है

String::from()

यह पैरामीटर के रूप में पारित कुछ डिफ़ॉल्ट मान के साथ एक स्ट्रिंग बनाता है from() तरीका।

निम्न उदाहरण एक स्ट्रिंग ऑब्जेक्ट के उपयोग को दिखाता है।

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

कोई रिक्त स्ट्रिंग का उपयोग कर वस्तु - उपरोक्त उदाहरण में दो तार बनाता है नई विधि और का उपयोग कर स्ट्रिंग शाब्दिक से एक स्ट्रिंग वस्तु से विधि।

आउटपुट नीचे दिखाया गया है -

length is 0
length is 14

सामान्य तरीके - स्ट्रिंग ऑब्जेक्ट

अनु क्रमांक। तरीका हस्ताक्षर विवरण
1 नया() pub const fn new () → स्ट्रिंग एक नया खाली स्ट्रिंग बनाता है।
2 तार() fn to_string (और स्वयं) → स्ट्रिंग दिए गए मान को स्ट्रिंग में परिवर्तित करता है।
3 बदलने के() पब fn की जगह <'a, P> (&' a self, from: P, to: & str) → स्ट्रिंग पैटर्न के सभी मैचों को दूसरी स्ट्रिंग के साथ बदल देता है।
4 as_str () pub fn as_str (& self) → & str एक स्ट्रिंग स्लाइस निकालता है जिसमें पूरी स्ट्रिंग होती है।
5 धक्का दें() पब एफएन पुश (और म्यूट सेल्फ, ch: char) इस स्ट्रिंग के अंत में दिए गए चार को जोड़ता है।
6 push_str () pub fn push_str (और mut self, string: & str) इस स्ट्रिंग के अंत में दिए गए स्ट्रिंग स्लाइस को जोड़ता है।
7 लेन () पब एफएन लेन (और स्व) → usize इस स्ट्रिंग की लंबाई, बाइट्स में देता है।
8 ट्रिम () पब fn ट्रिम (और स्व) → & str प्रमुख और पीछे वाले व्हाट्सएप के साथ एक स्ट्रिंग टुकड़ा लौटाता है।
9 split_whitespace () पब fn स्प्लिट_व्हाट्सएप (& स्वयं) → स्प्लिटविट्सस्पेस व्हाट्सएप द्वारा एक स्ट्रिंग स्लाइस को विभाजित करता है और एक पुनरावृत्ति देता है।
10 विभाजित करें() pub fn विभाजित <'a, P> (&' a self, pat: P) → स्प्लिट <'a, P>, जहां P पैटर्न हो सकता है और str, char, या एक क्लोजर जो विभाजन को निर्धारित करता है। इस स्ट्रिंग स्लाइस के सबस्टिट्यूटिंग पर एक इटरेटर लौटाता है, जो एक पैटर्न द्वारा मिलान किए गए वर्णों द्वारा अलग किया जाता है।
1 1 वर्ण () पब fn वर्ण (और स्व) → चार्ट एक स्ट्रिंग स्लाइस के वर्ण पर एक पुनरावृत्ति देता है।

चित्रण: नया ()

रिक्त स्ट्रिंग ऑब्जेक्ट का उपयोग करके बनाया गया है new()विधि और उसका मान हैलो में सेट है ।

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

उत्पादन

उपरोक्त कार्यक्रम निम्नलिखित आउटपुट उत्पन्न करता है -

hello

चित्रण: to_string ()

स्ट्रिंग ऑब्जेक्ट के सभी तरीकों तक पहुँचने के लिए, स्ट्रिंग स्ट्रिंग को ऑब्जेक्ट के प्रकार का उपयोग करके कन्वर्ट करें to_string() समारोह।

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

उत्पादन

उपरोक्त कार्यक्रम निम्नलिखित आउटपुट उत्पन्न करता है -

Hello TutorialsPoint , Hello!

चित्रण: बदलें ()

replace()फ़ंक्शन दो पैरामीटर लेता है - पहला पैरामीटर खोज करने के लिए एक स्ट्रिंग पैटर्न है और दूसरा पैरामीटर प्रतिस्थापित किया जाने वाला नया मान है। उपरोक्त उदाहरण में, नमस्कार name1 स्ट्रिंग में दो बार दिखाई देता है ।

प्रतिस्थापित फ़ंक्शन स्ट्रिंग की सभी घटनाओं को प्रतिस्थापित करता है Hello साथ में Howdy

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

उत्पादन

उपरोक्त कार्यक्रम निम्नलिखित आउटपुट उत्पन्न करता है -

Howdy TutorialsPoint , Howdy!

चित्रण: as_str ()

as_str() फ़ंक्शन एक स्ट्रिंग टुकड़ा निकालता है जिसमें पूरे स्ट्रिंग होते हैं।

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

उत्पादन

उपरोक्त कार्यक्रम निम्नलिखित आउटपुट उत्पन्न करता है -

displaying string literal example_string

चित्रण: धक्का ()

push() फ़ंक्शन इस स्ट्रिंग के अंत में दिए गए चार को जोड़ता है।

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

उत्पादन

उपरोक्त कार्यक्रम निम्नलिखित आउटपुट उत्पन्न करता है -

Tutorials

चित्रण: push_str ()

push_str() फ़ंक्शन स्ट्रिंग के अंत में दिए गए स्ट्रिंग स्लाइस को जोड़ता है।

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

उत्पादन

उपरोक्त कार्यक्रम निम्नलिखित आउटपुट उत्पन्न करता है -

Tutorials Point

चित्रण: लेन ()

len() फ़ंक्शन एक स्ट्रिंग (रिक्त स्थान सहित) में वर्णों की कुल संख्या देता है।

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

उत्पादन

उपरोक्त कार्यक्रम निम्नलिखित आउटपुट उत्पन्न करता है -

length is 20

चित्रण: ट्रिम ()

ट्रिम () फ़ंक्शन एक स्ट्रिंग में अग्रणी और अनुगामी रिक्त स्थान को निकालता है। ध्यान दें कि यह फ़ंक्शन इनलाइन रिक्त स्थान को नहीं हटाएगा।

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

उत्पादन

उपरोक्त कार्यक्रम निम्नलिखित आउटपुट उत्पन्न करता है -

Before trim
length is 24

After trim
length is 15

उदाहरण: split_whitespace ()

split_whitespace()इनपुट स्ट्रिंग को अलग-अलग तारों में विभाजित करता है। यह एक पुनरावृत्ति देता है ताकि हम नीचे दिखाए गए टोकन के माध्यम से पुनरावृत्ति कर रहे हैं -

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

उत्पादन

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

चित्रण: विभाजन () स्ट्रिंग

split() stringविधि एक स्ट्रिंग स्लाइस के सबस्ट्रिंग पर एक पुनरावृत्ति देता है, एक पैटर्न द्वारा मिलान किए गए वर्णों द्वारा अलग किया जाता है। विभाजन () विधि की सीमा यह है कि परिणाम बाद के उपयोग के लिए संग्रहीत नहीं किया जा सकता है। collect एक वेक्टर के रूप में विभाजित () द्वारा लौटाए गए परिणाम को संग्रहीत करने के लिए विधि का उपयोग किया जा सकता है।

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

उपरोक्त उदाहरण स्ट्रिंग को विभाजित करता है fullname, जब भी यह एक अल्पविराम का सामना करता है (,)

उत्पादन

token is Kannan
token is Sudhakaran
token is Tutorialspoint

firstName is Kannan
lastname is Sudhakaran
company is Tutorialspoint

चित्रण: आकर्षण ()

एक स्ट्रिंग में अलग-अलग वर्णों को वर्ण विधि का उपयोग करके एक्सेस किया जा सकता है। इसे समझने के लिए एक उदाहरण पर विचार करते हैं।

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

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

उत्पादन

T
u
t
o
r
i
a
l
s

संचालक के साथ स्ट्रिंग्स का सामंजस्य

एक स्ट्रिंग मान को दूसरे स्ट्रिंग में जोड़ा जा सकता है। इसे संघनन या प्रक्षेप कहा जाता है। स्ट्रिंग संघनन का परिणाम एक नया स्ट्रिंग ऑब्जेक्ट है। + ऑपरेटर आंतरिक रूप से एक ऐड विधि का उपयोग करता है । ऐड फ़ंक्शन का सिंटैक्स दो पैरामीटर लेता है। पहला पैरामीटर स्वयं है - स्ट्रिंग ऑब्जेक्ट स्वयं और दूसरा पैरामीटर दूसरी स्ट्रिंग ऑब्जेक्ट का एक संदर्भ है। यह नीचे दिखाया गया है -

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

चित्रण: स्ट्रिंग कॉनटेनटेशन

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

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

आउटपुट नीचे दिया जाएगा

TutorialsPoint

चित्रण: टाइप कास्टिंग

निम्न उदाहरण एक संख्या को एक स्ट्रिंग ऑब्जेक्ट में परिवर्तित करने का चित्रण करता है -

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

आउटपुट नीचे दिया जाएगा

2020
true

चित्रण: स्वरूप! मैक्रो

स्ट्रिंग वस्तुओं को एक साथ जोड़ने का एक अन्य तरीका एक मैक्रो फ़ंक्शन का उपयोग करना है जिसे प्रारूप कहा जाता है। प्रारूप का उपयोग! जैसा कि नीचे दिखाया गया है।

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

आउटपुट नीचे दिया जाएगा

Tutorials Point

एक ऑपरेटर कुछ फ़ंक्शन को परिभाषित करता है जो डेटा पर किया जाएगा। जिस डेटा पर ऑपरेटर काम करते हैं उसे ऑपरेंड कहा जाता है। निम्नलिखित अभिव्यक्ति पर विचार करें -

7 + 5 = 12

यहां, मान 7, 5 और 12 ऑपरेंड हैं, जबकि + और = ऑपरेटर हैं।

रस्ट में प्रमुख ऑपरेटरों को निम्नानुसार वर्गीकृत किया जा सकता है -

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

अंकगणितीय आपरेटर

मान लें कि चर a और b क्रमशः 10 और 5 हैं।

उदाहरण दिखाएं

अनु क्रमांक ऑपरेटर विवरण उदाहरण
1 + (जोड़) ऑपरेंड का योग देता है a + b 15 है
2 - (घटाव) मूल्यों का अंतर लौटाता है एब 5 है
3 * (गुणा) मूल्यों का उत्पाद लौटाता है a * b 50 है
4 / (विभाजन) विभाजन ऑपरेशन करता है और भागफल लौटाता है a / b 2 है
5 % (मापांक) विभाजन ऑपरेशन करता है और शेष को लौटाता है अ% ख 0 र

NOTE - + + और - ऑपरेटरों को जंग में समर्थित नहीं हैं।

संबंधपरक संकारक

रिलेशनल ऑपरेटर्स दो संस्थाओं के बीच संबंधों के प्रकार का परीक्षण या परिभाषित करते हैं। रिलेशनल ऑपरेटरों का उपयोग दो या अधिक मूल्यों की तुलना करने के लिए किया जाता है। रिलेशनल ऑपरेटर्स एक बूलियन मान लौटाते हैं - सही या गलत।

मान लें कि A का मान 10 है और B का मान 20 है।

उदाहरण दिखाएं

अनु क्रमांक ऑपरेटर विवरण उदाहरण
1 > से अधिक (ए> बी) गलत है
2 < के मुकाबले कम (ए <बी) सच है
3 > = इससे बड़ा या इसके बराबर (ए> = बी) गलत है
4 <= से कम या उसके बराबर (ए <= बी) सच है
5 == समानता (ए == बी) झूठा है
6 ! = बराबर नहीं (ए! = बी) सच है

लॉजिकल ऑपरेटर्स

लॉजिकल ऑपरेटर्स का उपयोग दो या दो से अधिक स्थितियों को मिलाने के लिए किया जाता है। लॉजिकल ऑपरेटर भी, एक बूलियन मान लौटाते हैं। चर A का मान 10 है और B का मान 20 है।

उदाहरण दिखाएं

अनु क्रमांक ऑपरेटर विवरण उदाहरण
1 && (तथा) ऑपरेटर तभी सही लौटाता है, जब सभी अभिव्यक्तियाँ सही हो (A> 10 && B> 10) गलत है
2 || (या) यदि ऑपरेटर निर्दिष्ट रिटर्न में से कम से कम एक सही है तो ऑपरेटर सही लौटाता है (ए> 10 || बी> 10) सच है
3 ! (नहीं) ऑपरेटर अभिव्यक्ति के परिणाम का प्रतिलोम लौटाता है। उदाहरण के लिए:! (> 5) गलत है ! (A> 10) सत्य है

बिटवाइज ऑपरेटर्स

चर ए = 2 और बी = 3 मान लें।

उदाहरण दिखाएं

अनु क्रमांक ऑपरेटर विवरण उदाहरण
1 और (बिटवाइज़ और) यह अपने पूर्णांक तर्कों के प्रत्येक बिट पर एक बूलियन और ऑपरेशन करता है। (A & B) 2 है
2 | (बिट वाइज OR) यह अपने पूर्णांक तर्कों के प्रत्येक बिट पर एक बूलियन या ऑपरेशन करता है। (ए | बी) 3 है
3 ^ (बिट वाइज XOR) यह अपने पूर्णांक तर्कों के प्रत्येक बिट पर एक बूलियन अनन्य या ऑपरेशन करता है। एक्सक्लूसिव OR का मतलब है कि या तो ऑपरेंड एक सच है या ऑपरेंड दो सही है, लेकिन दोनों नहीं। (ए ^ बी) 1 है
4 ! (बिट वाइज नहीं) यह एक अपरिपक्व ऑपरेटर है और ऑपरेंड में सभी बिट्स को उलट कर संचालित करता है। (B) -4 है
5 << (लेफ्ट शिफ्ट) यह दूसरे ऑपरेंड में निर्दिष्ट स्थानों की संख्या से अपने पहले ऑपरेंड में सभी बिट्स को बाईं ओर ले जाता है। नए बिट्स शून्य से भरे हुए हैं। एक स्थिति द्वारा छोड़े गए मूल्य को 2 से गुणा करने के बराबर है, दो पदों को स्थानांतरित करना 4 से गुणा करने के बराबर है, और इसी तरह। (ए << 1) 4 है
6 >> (राइट शिफ्ट) बाइनरी राइट शिफ्ट ऑपरेटर। बाएं ऑपरेंड का मान दाएं ऑपरेंड द्वारा निर्दिष्ट बिट्स की संख्या द्वारा दाईं ओर ले जाया जाता है। (ए >> 1) 1 है
7 >>> (शून्य के साथ सही बदलाव) यह ऑपरेटर सिर्फ >> ऑपरेटर की तरह है, सिवाय इसके कि बाईं ओर स्थानांतरित बिट्स हमेशा शून्य हैं। (ए >>> 1) 1 है

निर्णय लेने की संरचनाओं के लिए आवश्यक है कि प्रोग्रामर एक या एक से अधिक शर्तों को कार्यक्रम द्वारा मूल्यांकन या परीक्षण करने के लिए निर्दिष्ट करता है, साथ ही एक बयान या बयानों के साथ निष्पादित किया जाता है यदि स्थिति सही होने के लिए निर्धारित की जाती है, और वैकल्पिक रूप से, निष्पादित किए जाने वाले अन्य बयान। हालत झूठी होने के लिए निर्धारित है।

नीचे दिखाया गया है कि अधिकांश प्रोग्रामिंग भाषाओं में पाया जाने वाला एक विशिष्ट निर्णय लेने की संरचना का सामान्य रूप है -

अनु क्रमांक विवरण और विवरण
1

if statement

यदि एक बयान में बूलियन अभिव्यक्ति होती है, जिसके बाद एक या अधिक कथन होते हैं।

2

if...else statement

एक अगर बयान एक वैकल्पिक द्वारा पीछा किया जा सकता है और कुछ बयान है, जो कार्यान्वित जब बूलियन अभिव्यक्ति गलत है।

3

else...if and nested ifstatement

आप उपयोग कर सकते हैं , तो या किसी और अगर दूसरे के अंदर बयान करता है, तो या किसी और अगर बयान (रों)।

4

match statement

एक मैच स्टेटमेंट एक वैरिएबल को मानों की सूची के खिलाफ परीक्षण करने की अनुमति देता है।

अगर स्टेटमेंट

अगर ... किसी और निर्माण मूल्यांकन करता है कोड का एक खंड से पहले एक शर्त है निष्पादित किया जाता है।

वाक्य - विन्यास

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

यदि बूलियन अभिव्यक्ति सच का मूल्यांकन करती है, तो यदि कथन के अंदर कोड का ब्लॉक निष्पादित किया जाएगा। यदि बूलियन अभिव्यक्ति झूठी का मूल्यांकन करती है, तो कोड का पहला सेट यदि विवरण के अंत के बाद (करी घुंघराले ब्रेस के बाद) निष्पादित किया जाएगा।

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

उपरोक्त उदाहरण प्रिंट होगा number is positive यदि ब्लॉक द्वारा निर्दिष्ट की गई स्थिति सत्य है।

अगर कोई और बयान

एक if एक वैकल्पिक द्वारा पीछा किया जा सकता है elseखंड मैथा। यदि कथन द्वारा गलत का मूल्यांकन करने पर बूलियन अभिव्यक्ति का परीक्षण किया जाता है, तो दूसरा ब्लॉक निष्पादित करेगा।

वाक्य - विन्यास

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

प्रवाह संचित्र

ifब्लॉक सशर्त अभिव्यक्ति की रक्षा करता है। यदि बूलियन अभिव्यक्ति सही का मूल्यांकन करती है, तो इफ स्टेटमेंट से जुड़े ब्लॉक को निष्पादित किया जाता है।

यदि किसी अन्य वैकल्पिक कथन द्वारा ब्लॉक का पालन किया जा सकता है। दूसरे ब्लॉक से जुड़े इंस्ट्रक्शन ब्लॉक को निष्पादित किया जाता है यदि अभिव्यक्ति गलत का मूल्यांकन करती है।

इलस्ट्रेशन - सरल अगर ... और

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

उपरोक्त उदाहरण प्रिंट करता है कि क्या एक चर में मूल्य समान या विषम है। यदि ब्लॉक समान को निर्धारित करने के लिए 2 से मान की विभाज्यता की जांच करता है। यहाँ उपरोक्त कोड का उत्पादन है -

Even

नेस्टेड इफ

else…ifसीढ़ी कई स्थितियों का परीक्षण करने के लिए उपयोगी है। सिंटैक्स नीचे दिखाया गया है -

वाक्य - विन्यास

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
}

जब ... और ... का उपयोग करते हैं, तो और कुछ कथन, ध्यान में रखने के लिए कुछ बिंदु हैं।

  • यदि शून्य हो सकता है या किसी और का हो सकता है और इसे किसी और के बाद आना चाहिए ...
  • यदि किसी व्यक्ति के पास शून्य हो सकता है..तो और उन्हें दूसरे से पहले आना चाहिए।
  • एक बार और..एक बार सफल होने के बाद, शेष कोई भी नहीं .. या किसी और का परीक्षण किया जाएगा।

उदाहरण: और ... अगर सीढ़ी

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

स्निपेट प्रदर्शित करता है कि मूल्य सकारात्मक, नकारात्मक या शून्य है या नहीं।

उत्पादन

2 is positive

मैच स्टेटमेंट

मैच स्टेटमेंट यह जांचता है कि क्या वर्तमान मान मानों की सूची से मेल खा रहा है, यह C भाषा में स्विच स्टेटमेंट के समान है। पहली बार में, ध्यान दें कि मैच कीवर्ड के बाद की अभिव्यक्ति को कोष्ठकों में संलग्न करने की आवश्यकता नहीं है।

सिंटैक्स नीचे दिखाया गया है।

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

नीचे दिए गए उदाहरण में, state_code मूल्यों की सूची के साथ मेल खाता है MH, KL, KA, GA- यदि कोई मैच पाया जाता है, तो एक स्ट्रिंग मान को चर स्थिति में लौटा दिया जाता है । यदि कोई मिलान नहीं पाया जाता है, तो डिफ़ॉल्ट केस _ मैच और मान Unkown वापस आ जाता है।

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

उत्पादन

Found match for MH
State name is Maharashtra

ऐसे उदाहरण हो सकते हैं, जहां कोड के एक ब्लॉक को बार-बार निष्पादित करने की आवश्यकता होती है। सामान्य तौर पर, प्रोग्रामिंग निर्देशों को क्रमिक रूप से निष्पादित किया जाता है: एक फ़ंक्शन में पहला स्टेटमेंट पहले निष्पादित किया जाता है, उसके बाद दूसरा, और इसी तरह।

प्रोग्रामिंग भाषाएँ विभिन्न नियंत्रण संरचनाएं प्रदान करती हैं जो अधिक जटिल निष्पादन पथों के लिए अनुमति देती हैं।

एक लूप स्टेटमेंट हमें कई बार स्टेटमेंट या स्टेटमेंट ऑफ स्टेट को निष्पादित करने की अनुमति देता है। नीचे दिए गए प्रोग्रामिंग भाषाओं में से अधिकांश में एक लूप स्टेटमेंट का सामान्य रूप है।

जंग लूपिंग आवश्यकताओं को संभालने के लिए विभिन्न प्रकार के लूप प्रदान करता है -

  • while
  • loop
  • for

निश्चित लूप

एक लूप की पुनरावृत्तियों की संख्या निश्चित है / निश्चित है जिसे एक निश्चित लूप कहा जाता है। for लूप एक निश्चित लूप का कार्यान्वयन है।

पाश के लिए

लूप के लिए निर्दिष्ट समय के लिए कोड ब्लॉक निष्पादित करता है। इसका उपयोग मानों के एक निश्चित समूह पर सरणी के रूप में पुनरावृत्त करने के लिए किया जा सकता है। लूप के लिए सिंटैक्स नीचे दिया गया है

वाक्य - विन्यास

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

लूप के लिए एक उदाहरण नीचे दिखाया गया है

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

NOTE: कि चर x केवल ब्लॉक के लिए सुलभ है।

उत्पादन

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

अनिश्चित लूप

अनिश्चित लूप का उपयोग तब किया जाता है जब एक लूप में पुनरावृत्तियों की संख्या अनिश्चित या अज्ञात होती है।

अनिश्चित लूप का उपयोग करके लागू किया जा सकता है -

अनु क्रमांक नाम और विवरण
1

While

जबकि पाश निर्देश हर बार हालत सही का आकलन निर्दिष्ट कार्यान्वित

2

Loop

पाश थोड़ी देर के (सही) अनिश्चितकालीन पाश है

चित्रण - कुछ समय के लिए

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

आउटपुट नीचे दिखाया गया है -

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

चित्रण ustloop

fn main(){
   //while true

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

      if x==15 {
         break;
      }
   }
}

breakस्टेटमेंट का उपयोग किसी निर्माण से नियंत्रण हटाने के लिए किया जाता है। एक लूप में ब्रेक का उपयोग करने से प्रोग्राम लूप से बाहर निकल जाता है।

उत्पादन

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

जारी बयान

जारी बयान वर्तमान पुनरावृत्ति में बाद के बयानों को छोड़ देता है और नियंत्रण को लूप की शुरुआत में वापस ले जाता है। ब्रेक स्टेटमेंट के विपरीत, जारी लूप से बाहर नहीं निकलता है। यह वर्तमान पुनरावृत्ति को समाप्त करता है और बाद में पुनरावृत्ति शुरू करता है।

जारी बयान का एक उदाहरण नीचे दिया गया है।

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
}

उपरोक्त उदाहरण 0 और 20 के बीच भी मानों की संख्या प्रदर्शित करता है। लूप वर्तमान पुनरावृत्ति को बाहर निकालता है यदि संख्या समान है। यह जारी स्टेटमेंट का उपयोग करके प्राप्त किया जाता है।

0 और 20 के बीच विषम मानों की गिनती 10 है

कार्य पठनीय, बनाए रखने योग्य और पुन: प्रयोज्य कोड के निर्माण खंड हैं। एक फ़ंक्शन किसी विशिष्ट कार्य को करने के लिए कथनों का एक समूह है। फ़ंक्शंस प्रोग्राम को कोड के तार्किक ब्लॉकों में व्यवस्थित करते हैं। एक बार परिभाषित करने के बाद, फ़ंक्शंस को एक्सेस कोड कहा जा सकता है। यह कोड को पुन: प्रयोज्य बनाता है। इसके अलावा, फ़ंक्शन प्रोग्राम के कोड को पढ़ना और बनाए रखना आसान बनाते हैं।

एक फ़ंक्शन घोषणापत्र एक फ़ंक्शन के नाम, रिटर्न प्रकार और मापदंडों के बारे में संकलक को बताता है। एक फ़ंक्शन परिभाषा फ़ंक्शन का वास्तविक निकाय प्रदान करती है।

अनु क्रमांक समारोह विवरण
1

Defining a function

TA फ़ंक्शन परिभाषा निर्दिष्ट करती है कि एक विशिष्ट कार्य क्या और कैसे किया जाएगा।

2

Calling or invoking a Function

एक फ़ंक्शन को इसे निष्पादित करने के लिए बुलाया जाना चाहिए।

3

Returning Functions

नियंत्रण वापस कॉल करने वाले के साथ-साथ मूल्य भी वापस कर सकते हैं।

4

Parameterized Function

पैरामीटर फ़ंक्शन के मानों को पारित करने के लिए एक तंत्र है।

एक कार्य को परिभाषित करना

एक फ़ंक्शन परिभाषा निर्दिष्ट करती है कि एक विशिष्ट कार्य क्या और कैसे किया जाएगा। किसी फ़ंक्शन का उपयोग करने से पहले, इसे परिभाषित किया जाना चाहिए। फ़ंक्शन बॉडी में कोड होता है जिसे फ़ंक्शन द्वारा निष्पादित किया जाना चाहिए। किसी फ़ंक्शन के नामकरण के नियम एक चर के समान हैं। कार्यों का उपयोग करके परिभाषित किया गया हैfnकीवर्ड। मानक फ़ंक्शन को परिभाषित करने का सिंटैक्स नीचे दिया गया है

वाक्य - विन्यास

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

एक फ़ंक्शन घोषणा में वैकल्पिक रूप से पैरामीटर / तर्क हो सकते हैं। मान का उपयोग फ़ंक्शंस को मान देने के लिए किया जाता है।

उदाहरण - सरल कार्य परिभाषा

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

एक समारोह का आह्वान

एक फ़ंक्शन को इसे निष्पादित करने के लिए बुलाया जाना चाहिए। इस प्रक्रिया को कहा जाता हैfunction invocation। किसी फ़ंक्शन को लागू करने पर पैरामीटर के लिए मान पारित किए जाने चाहिए। किसी अन्य फ़ंक्शन को आमंत्रित करने वाले फ़ंक्शन को कहा जाता हैcaller function.

वाक्य - विन्यास

function_name(val1,val2,valN)

उदाहरण: एक फ़ंक्शन को लागू करना

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

यहां, मुख्य () कॉलर फ़ंक्शन है।

चित्रण

निम्न उदाहरण एक फ़ंक्शन को परिभाषित करता है fn_hello()। फ़ंक्शन कंसोल को संदेश प्रिंट करता है। main()फ़ंक्शन fn_hello () फ़ंक्शन को आमंत्रित करता है।

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

उत्पादन

hello from function fn_hello

एक फंक्शन से रिटर्निंग वैल्यू

नियंत्रण वापस कॉल करने वाले के साथ-साथ एक मान भी लौटा सकता है। ऐसे कार्यों को रिटर्निंग फ़ंक्शन कहा जाता है।

वाक्य - विन्यास

किसी भी प्रकार के सिंटैक्स का उपयोग रिटर्न प्रकार के साथ फ़ंक्शन को परिभाषित करने के लिए किया जा सकता है।

वापसी विवरण के साथ

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

रिटर्न स्टेटमेंट के बिना शॉर्टहैंड सिंटैक्स

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

lllustration

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

उत्पादन

pi value is 3.142857142857143

पैरामीटर के साथ फ़ंक्शन

पैरामीटर फ़ंक्शन के मानों को पारित करने के लिए एक तंत्र है। पैरामीटर फ़ंक्शन के हस्ताक्षर का एक हिस्सा बनाते हैं। पैरामीटर मान को इसके मंगलाचरण के दौरान फ़ंक्शन में पास किया जाता है। जब तक स्पष्ट रूप से निर्दिष्ट नहीं किया जाता है, तब तक फ़ंक्शन के लिए दिए गए मानों की संख्या निर्धारित मापदंडों की संख्या से मेल खाना चाहिए।

पैरामीटर को निम्नलिखित तकनीकों में से एक का उपयोग करके फ़ंक्शन में पास किया जा सकता है -

मान से पास करें

जब कोई विधि लागू होती है, तो प्रत्येक मान पैरामीटर के लिए एक नया संग्रहण स्थान बनाया जाता है। वास्तविक मापदंडों के मूल्यों को उनमें कॉपी किया जाता है। इसलिए, लागू विधि के अंदर पैरामीटर में किए गए परिवर्तनों का तर्क पर कोई प्रभाव नहीं पड़ता है।

निम्नलिखित उदाहरण एक चर संख्या की घोषणा करता है, जो शुरू में है। चर को पैरामीटर (मान से) के रूप में पारित किया जाता है mutate_no_to_zero()फ़ंक्शनन्यूशन, जो मान को शून्य में बदलता है। फ़ंक्शन कॉल के बाद जब नियंत्रण मुख्य विधि पर वापस लौटता है तो मूल्य समान होगा।

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

उत्पादन

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

संदर्भ द्वारा पास करें

जब आप संदर्भ द्वारा पैरामीटर पास करते हैं, तो मूल्य मापदंडों के विपरीत, इन मापदंडों के लिए एक नया भंडारण स्थान नहीं बनाया जाता है। संदर्भ पैरामीटर वास्तविक मेमोरी के समान स्मृति स्थान का प्रतिनिधित्व करते हैं जो विधि को आपूर्ति की जाती है। पैरामीटर मानों को चर नाम के साथ उपसर्ग द्वारा संदर्भ द्वारा पारित किया जा सकता है&

नीचे दिए गए उदाहरण में, हमारे पास एक चर संख्या है , जो शुरू में है 5. चर संख्या का एक संदर्भ पास नहीं हैmutate_no_to_zero()समारोह। फ़ंक्शन मूल चर पर काम करता है। फ़ंक्शन कॉल के बाद, जब नियंत्रण मुख्य विधि पर वापस लौटता है, तो मूल चर का मूल्य शून्य होगा।

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
}

* ऑपरेटर का उपयोग मेमोरी स्थान में संग्रहीत मान तक पहुंचने के लिए किया जाता है जो चर है param_noइशारा करना। इसे डेरीफेरिंग के रूप में भी जाना जाता है।

उत्पादन होगा -

The value of no is 0.

एक समारोह में स्ट्रिंग पासिंग

Main () समारोह के लिए एक स्ट्रिंग वस्तु गुजरता प्रदर्शन () समारोह।

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

उत्पादन

param_name value is :TutorialsPoint

ट्यूपल एक यौगिक डेटा प्रकार है। एक स्केलर प्रकार केवल एक प्रकार का डेटा संग्रहीत कर सकता है। उदाहरण के लिए, एक i32 चर केवल एक पूर्णांक मान को संग्रहीत कर सकता है। यौगिक प्रकारों में, हम एक समय में एक से अधिक मूल्य संग्रहीत कर सकते हैं और यह विभिन्न प्रकारों का हो सकता है।

टुपल्स की एक निश्चित लंबाई होती है - एक बार घोषित होने के बाद वे आकार में बढ़ या सिकुड़ नहीं सकते। टपल इंडेक्स से शुरू होता है0

वाक्य - विन्यास

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

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

चित्रण

निम्न उदाहरण टुप में मान प्रदर्शित करता है।

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

Println! ( "{}", टपल) वाक्य रचना एक टपल में मूल्यों को प्रदर्शित करने के लिए इस्तेमाल नहीं किया जा सकता। ऐसा इसलिए है क्योंकि टुप्ल एक यौगिक प्रकार है। का प्रयोग करें println! ( "{:?}", tuple_name) वाक्य रचना एक टपल में मूल्यों मुद्रित करने के लिए।

उत्पादन

(-325, 4.9, 22)

चित्रण

निम्न उदाहरण एक टपल में व्यक्तिगत मूल्यों को प्रिंट करता है।

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

उत्पादन

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

चित्रण

निम्न उदाहरण एक कार्य के पैरामीटर के रूप में एक गुच्छे से गुजरता है। कार्यों के लिए टुपल्स पास किए जाते हैं।

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

उत्पादन

Inside print method
(110, true, 10.9)

destructing

विनाशकारी असाइनमेंट जंग की एक विशेषता है जिसमें हम एक टपल के मूल्यों को अनपैक करते हैं। यह अलग-अलग चर के लिए एक टपल सौंपकर हासिल किया जाता है।

निम्नलिखित उदाहरण पर विचार करें -

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

चर एक्स एक टपल है जिसे लेट स्टेटमेंट को सौंपा गया है। प्रत्येक चर - आयु, is_male और cgpa में टुपल में संबंधित मान होंगे।

उत्पादन

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

इस अध्याय में, हम एक सरणी और उससे जुड़ी विभिन्न विशेषताओं के बारे में जानेंगे। इससे पहले कि हम सरणियों के बारे में जानें, आइए देखें कि एक सरणी एक चर से अलग कैसे है।

चर की निम्नलिखित सीमाएँ हैं -

  • चर प्रकृति में स्केलर हैं। दूसरे शब्दों में, एक चर घोषणा में केवल एक बार में एक ही मूल्य हो सकता है। इसका मतलब है कि एक प्रोग्राम में एन वैल्यू को स्टोर करने के लिए एन वेरिएबल डिक्लेरेशन की जरूरत होगी। इसलिए, जब मूल्यों का एक बड़ा संग्रह संग्रहीत करने की आवश्यकता होती है, तो चर का उपयोग संभव नहीं होता है।

  • एक कार्यक्रम में चर को यादृच्छिक क्रम में मेमोरी आवंटित की जाती है, जिससे उनकी घोषणा के क्रम में मूल्यों को पुनः प्राप्त / पढ़ना मुश्किल हो जाता है।

एक सरणी मानों का एक सजातीय संग्रह है। सीधे शब्दों में कहें, एक सरणी समान डेटा प्रकार के मूल्यों का एक संग्रह है।

एक ऐरे की विशेषताएं

एक सरणी की विशेषताएं नीचे सूचीबद्ध हैं -

  • एक सरणी घोषणा अनुक्रमिक मेमोरी ब्लॉकों को आवंटित करती है।

  • ऐरे स्थिर हैं। इसका मतलब यह है कि एक बार आरंभीकृत होने के बाद सरणी को आकार नहीं दिया जा सकता।

  • प्रत्येक मेमोरी ब्लॉक एक सरणी तत्व का प्रतिनिधित्व करता है।

  • सरणी तत्वों की पहचान एक अद्वितीय पूर्णांक द्वारा की जाती है जिसे तत्व का उपप्रकाश / सूचकांक कहा जाता है।

  • सरणी तत्वों को आबाद करना सरणी आरंभीकरण के रूप में जाना जाता है।

  • एरे तत्व मूल्यों को अद्यतन या संशोधित किया जा सकता है, लेकिन हटाया नहीं जा सकता।

घोषणा और प्रारंभिक सारणी

नीचे दिए गए सिंटैक्स का उपयोग करें और Rust में एक सरणी को घोषित करने के लिए।

वाक्य - विन्यास

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

पहले सिंटैक्स में, सरणी का प्रकार आरंभीकरण के दौरान सरणी के पहले तत्व के डेटा प्रकार से अनुमानित है।

चित्रण: सरल ऐरे

निम्न उदाहरण स्पष्ट रूप से सरणी के आकार और डेटा प्रकार को निर्दिष्ट करता है। {:}! Println की वाक्य रचना () समारोह सरणी में सभी मान मुद्रित करने के लिए प्रयोग किया जाता है। LEN () समारोह सरणी के आकार की गणना करने के लिए इस्तेमाल किया जाता है।

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

उत्पादन

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

चित्रण: डेटा प्रकार के बिना सरणी

निम्नलिखित कार्यक्रम 4 तत्वों की एक सरणी घोषित करता है। चर घोषणा के दौरान डेटाटाइप स्पष्ट रूप से निर्दिष्ट नहीं है। इस स्थिति में, सरणी पूर्णांक प्रकार की होगी। LEN () समारोह सरणी के आकार की गणना करने के लिए इस्तेमाल किया जाता है।

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

उत्पादन

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

चित्रण: डिफ़ॉल्ट मान

निम्न उदाहरण एक सरणी बनाता है और -1 के डिफ़ॉल्ट मान के साथ उसके सभी तत्वों को इनिशियलाइज़ करता है ।

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

उत्पादन

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

चित्रण: लूप के लिए सरणी

निम्न उदाहरण एक सरणी के माध्यम से पुनरावृत्त करता है और अनुक्रमित और उनके संबंधित मूल्यों को प्रिंट करता है। लूप इंडेक्स 0 से 4 (अंतिम सरणी तत्व का इंडेक्स) से मान प्राप्त करता है।

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

उत्पादन

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

चित्रण: iter () फ़ंक्शन का उपयोग करना

Iter () फ़ंक्शन किसी सरणी में सभी तत्वों के मूल्यों को प्राप्त करता है।

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

उत्पादन

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

चित्रण: परस्पर सारणी

Mut कीवर्ड एक परिवर्तनशील सरणी घोषित करने के लिए इस्तेमाल किया जा सकता। निम्नलिखित उदाहरण एक उत्परिवर्तित सरणी की घोषणा करता है और दूसरे सरणी तत्व के मूल्य को संशोधित करता है।

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

उत्पादन

[10, 0, 30, 40]

कार्य के लिए पैरामीटर्स के रूप में एरे को पास करना

एक सरणी को मानों या फ़ंक्शन के संदर्भ में पारित किया जा सकता है।

चित्रण: मूल्य द्वारा पास

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

उत्पादन

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

चित्रण: संदर्भ से गुजारें

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

उत्पादन

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

ऐलान घोषणा और लगातार

आइए सरणी घोषणा और स्थिरांक को समझने के लिए नीचे दिए गए एक उदाहरण पर विचार करें।

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

कंपाइलर एक अपवाद के रूप में परिणाम देगा। ऐसा इसलिए है क्योंकि संकलन समय पर एक सरणी की लंबाई ज्ञात होनी चाहिए। यहां, रनटाइम पर चर "एन" का मूल्य निर्धारित किया जाएगा। दूसरे शब्दों में, किसी सरणी के आकार को परिभाषित करने के लिए चर का उपयोग नहीं किया जा सकता है।

हालाँकि, निम्न प्रोग्राम मान्य है -

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

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

कास्ट कीवर्ड के साथ पहले से पहचाने गए पहचानकर्ता का मान संकलन समय पर परिभाषित किया गया है और रनटाइम में नहीं बदला जा सकता है। usize पॉइंटर-आकार है, इस प्रकार इसका वास्तविक आकार उस आर्किटेक्चर पर निर्भर करता है जिसके लिए आप अपना प्रोग्राम संकलित कर रहे हैं।

एक कार्यक्रम के लिए मेमोरी निम्नलिखित में आवंटित की जा सकती है -

  • Stack
  • Heap

ढेर

एक स्टैक पहले आउट ऑर्डर में एक अंतिम का अनुसरण करता है। स्टैक डेटा मूल्यों को संग्रहीत करता है जिसके लिए आकार को संकलन समय पर जाना जाता है। उदाहरण के लिए, स्थिर आकार i32 का एक चर ढेर आवंटन के लिए एक उम्मीदवार है। इसका आकार संकलन समय पर जाना जाता है। आकार निर्धारित होने के कारण सभी स्केलर प्रकार स्टैक में संग्रहीत किए जा सकते हैं।

एक स्ट्रिंग के उदाहरण पर विचार करें, जिसे रनटाइम पर एक मान असाइन किया गया है। ऐसे स्ट्रिंग का सटीक आकार संकलन समय पर निर्धारित नहीं किया जा सकता है। तो यह स्टैक आवंटन के लिए एक उम्मीदवार नहीं है, लेकिन ढेर आवंटन के लिए है।

ढेर

ढेर मेमोरी डेटा को संचित करता है जिसका आकार संकलन समय पर अज्ञात है। इसका उपयोग डायनेमिक डेटा को स्टोर करने के लिए किया जाता है। सीधे शब्दों में कहें, एक ढेर मेमोरी को उन डेटा मानों के लिए आवंटित किया जाता है जो कार्यक्रम के पूरे जीवन चक्र में बदल सकते हैं। ढेर स्मृति में एक क्षेत्र है जो स्टैक की तुलना में कम व्यवस्थित होता है।

स्वामित्व क्या है?

रस्ट के प्रत्येक मान में एक वैरिएबल होता है जिसे कहा जाता है ownerमूल्य का। रस्ट में संग्रहीत प्रत्येक डेटा के साथ एक मालिक जुड़ा होगा। उदाहरण के लिए, वाक्यविन्यास में - चलो उम्र = 30, आयु मूल्य 30 का मालिक है ।

  • प्रत्येक डेटा में एक बार में केवल एक ही स्वामी हो सकता है।

  • दो चर एक ही मेमोरी लोकेशन की ओर इशारा नहीं कर सकते। चर हमेशा अलग-अलग मेमोरी स्थानों की ओर इशारा करेंगे।

स्वामित्व हस्तांतरित करना

मूल्य का स्वामित्व इसके द्वारा हस्तांतरित किया जा सकता है -

  • एक चर का दूसरे चर पर मान देना।

  • फ़ंक्शन के लिए पासिंग मान।

  • किसी फ़ंक्शन से मान वापस करना।

एक चर का दूसरे चर पर मान देना

एक भाषा के रूप में रस्ट का प्रमुख विक्रय बिंदु इसकी मेमोरी सुरक्षा है। स्मृति सुरक्षा कड़ी नियंत्रण द्वारा हासिल की जाती है जो प्रतिबंधों का क्या और कब उपयोग कर सकता है।

निम्नलिखित स्निपेट पर विचार करें -

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

उपरोक्त उदाहरण एक वेक्टर v घोषित करता है। स्वामित्व का विचार यह है कि केवल एक चर एक संसाधन को बांधता है, या तो v संसाधन के लिए बांधता है या v2संसाधन के लिए बाध्य करता है। ऊपर दिए गए उदाहरण में एक त्रुटि है - स्थानांतरित मूल्य का उपयोग: `v` । ऐसा इसलिए है क्योंकि संसाधन का स्वामित्व v2 में स्थानांतरित किया गया है। इसका अर्थ है कि स्वामित्व को v से v2 (v2 = v) में स्थानांतरित कर दिया गया है और v को स्थानांतरित करने के बाद अमान्य कर दिया गया है।

फ़ंक्शन के लिए पासिंग मान

जब हम एक वस्तु को बंद या कार्य के लिए पास करते हैं, तो एक मूल्य का स्वामित्व भी बदल जाता है।

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

किसी फ़ंक्शन से मान वापस करना

फ़ंक्शन के पास दिया गया स्वामित्व फ़ंक्शन निष्पादन पूर्ण होने पर अमान्य हो जाएगा। इसके लिए चारों ओर एक काम यह है कि फ़ंक्शन कॉलर के पास स्वामित्व वाली वस्तु वापस लौटा दे।

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

स्वामित्व और आदिम प्रकार

आदिम प्रकारों के मामले में, एक चर से सामग्री को दूसरे में कॉपी किया जाता है। इसलिए, कोई मालिकाना कदम नहीं हो रहा है। ऐसा इसलिए है क्योंकि एक आदिम चर को किसी वस्तु की तुलना में कम संसाधनों की आवश्यकता होती है। निम्नलिखित उदाहरण पर विचार करें -

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

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

आउटपुट होगा - 10।

एक चर के स्वामित्व को किसी अन्य फ़ंक्शन में पास करना और फिर स्वामित्व वापस करना बहुत असुविधाजनक है। जंग एक अवधारणा का समर्थन करता है, उधार, जहां एक मूल्य का स्वामित्व अस्थायी रूप से एक इकाई में स्थानांतरित किया जाता है और फिर मूल मालिक इकाई में वापस आ जाता है।

निम्नलिखित पर विचार करें -

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

मुख्य फ़ंक्शन एक फ़ंक्शन print_vector () को आमंत्रित करता है । एक वेक्टर इस फ़ंक्शन के पैरामीटर के रूप में पारित किया गया है। वेक्टर का स्वामित्व मुख्य से प्रिंट_वेक्टर () फ़ंक्शन से भी पारित किया गया है । उपरोक्त कोड के परिणामस्वरूप त्रुटि दिखाई देगी जब मुख्य () फ़ंक्शन वेक्टर v तक पहुंचने का प्रयास करता है ।

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

ऐसा इसलिए है क्योंकि एक चर या मान का उपयोग उस फ़ंक्शन द्वारा नहीं किया जा सकता है जो स्वामित्व के किसी अन्य फ़ंक्शन में स्थानांतरित होने के बाद मूल रूप से उसका स्वामित्व रखता है।

उधार क्या है?

जब कोई फ़ंक्शन अस्थायी रूप से कुछ समय के लिए एक चर / मान पर अपना नियंत्रण स्थानांतरित करता है, तो इसे उधार कहा जाता है। यह चर के संदर्भ को पास करके प्राप्त किया जाता है(& var_name)चर / मान को फ़ंक्शन में पास करने के बजाय। चर / मान का स्वामित्व उस फ़ंक्शन के चर के मूल स्वामी को हस्तांतरित किया जाता है जिसके बाद नियंत्रण पारित किया गया था जो निष्पादन को पूरा करता है।

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

उत्पादन

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

परस्पर संदर्भ

एक फ़ंक्शन ऐसे संसाधन के एक परस्पर संदर्भ का उपयोग करके एक उधार संसाधन को संशोधित कर सकता है । एक परस्पर संदर्भ के साथ उपसर्ग है&mut। उत्परिवर्तित संदर्भ केवल परिवर्तनशील चरों पर काम कर सकते हैं।

चित्रण: पूर्णांक संदर्भ को म्यूट करना

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

Main () समारोह एक परिवर्तनशील पूर्णांक चर वाणी मैं और करने के लिए मैं के एक परिवर्तनशील संदर्भ गुजरताadd_one()। Add_one () एक से चर के मान को बढ़ाता है।

चित्रण: एक स्ट्रिंग संदर्भ म्यूट करना

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
}

Main () समारोह चर का एक परिवर्तनशील संदर्भ गुजरता नाम करने के लिए प्रदर्शन () समारोह। प्रदर्शन फ़ंक्शन मूल नाम चर के लिए एक अतिरिक्त स्ट्रिंग जोड़ता है ।

उत्पादन

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

एक टुकड़ा स्मृति के एक ब्लॉक के लिए एक संकेतक है। स्लाइस का उपयोग सन्निहित स्मृति ब्लॉकों में संग्रहीत डेटा के अंश तक पहुंचने के लिए किया जा सकता है। इसका उपयोग सरणियों, वैक्टर और तारों जैसी डेटा संरचनाओं के साथ किया जा सकता है। स्लाइस डेटा के कुछ हिस्सों को एक्सेस करने के लिए इंडेक्स नंबरों का उपयोग करते हैं। एक स्लाइस का आकार रनटाइम पर निर्धारित किया जाता है।

स्लाइस वास्तविक डेटा के संकेत हैं। उन्हें कार्यों के संदर्भ में पारित किया जाता है, जिसे उधार के रूप में भी जाना जाता है।

उदाहरण के लिए, स्लाइस का उपयोग स्ट्रिंग मान के एक हिस्से को लाने के लिए किया जा सकता है। एक कटा हुआ स्ट्रिंग वास्तविक स्ट्रिंग ऑब्जेक्ट का एक संकेतक है। इसलिए, हमें एक स्ट्रिंग के शुरुआती और समाप्ति सूचकांक को निर्दिष्ट करने की आवश्यकता है। इंडेक्स 0 की तरह ही शुरू होता है।

वाक्य - विन्यास

let sliced_value = &data_structure[start_index..end_index]

न्यूनतम सूचकांक मूल्य 0 है और अधिकतम सूचकांक मूल्य डेटा संरचना का आकार है। ध्यान दें कि end_index अंतिम स्ट्रिंग में शामिल नहीं किया जाएगा।

नीचे दिए गए आरेख में एक नमूना स्ट्रिंग ट्यूटोरियल दिखाया गया है , जिसमें 9 वर्ण हैं। पहले वर्ण का सूचकांक 0 है और अंतिम वर्ण का 8 है।

निम्न कोड स्ट्रिंग से 5 वर्णों को प्राप्त करता है (सूचकांक 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);
}

उत्पादन

length of string is 9
rials

चित्रण - एक पूर्णांक सरणी को जोड़ना

मुख्य () फ़ंक्शन 5 तत्वों के साथ एक सरणी घोषित करता है। यह आह्वान करता हैuse_slice()फ़ंक्शन और इसमें तीन तत्वों का एक टुकड़ा (डेटा सरणी को इंगित करता है) गुजरता है। स्लाइस को संदर्भ द्वारा पारित किया जाता है। Use_slice () फ़ंक्शन स्लाइस और उसकी लंबाई के मूल्य को प्रिंट करता है।

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

उत्पादन

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

म्यूटेबल स्लाइस

&mut कीवर्ड का उपयोग म्यूट के रूप में एक स्लाइस को चिह्नित करने के लिए किया जा सकता है।

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
}

उत्पादन

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

उपरोक्त कोड use_slice () फ़ंक्शन के लिए एक परस्पर स्लाइस पास करता है। फ़ंक्शन मूल सरणी के दूसरे तत्व को संशोधित करता है।

मानों के एक सजातीय संग्रह का प्रतिनिधित्व करने के लिए Arrays का उपयोग किया जाता है। इसी प्रकार, एक संरचना रस्ट में उपलब्ध एक अन्य उपयोगकर्ता परिभाषित डेटा प्रकार है जो हमें एक अन्य संरचना सहित विभिन्न प्रकारों के डेटा आइटम को संयोजित करने की अनुमति देता है। एक संरचना डेटा को कुंजी-मूल्य जोड़ी के रूप में परिभाषित करती है।

सिंटेक्स - एक संरचना की घोषणा

Struct कीवर्ड एक संरचना घोषित करने के लिए प्रयोग किया जाता है। चूंकि संरचनाएं सांख्यिकीय रूप से टाइप की जाती हैं, इसलिए संरचना के प्रत्येक क्षेत्र को डेटा प्रकार के साथ जोड़ा जाना चाहिए। एक संरचना के लिए नामकरण नियम और कन्वेंशन एक चर की तरह है। संरचना ब्लॉक को अर्धविराम के साथ समाप्त होना चाहिए।

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

सिंटेक्स - एक संरचना की शुरुआत

एक संरचना की घोषणा करने के बाद, प्रत्येक क्षेत्र को एक मूल्य सौंपा जाना चाहिए। इसे आरंभीकरण के रूप में जाना जाता है।

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

उपरोक्त उदाहरण तीन क्षेत्रों के साथ एक संरचना कर्मचारी की घोषणा करता है - नाम, कंपनी और प्रकार की आयु। मुख्य () संरचना को इनिशियलाइज़ करता है। यह Println का उपयोग करता है! संरचना में परिभाषित क्षेत्रों के मूल्यों को प्रिंट करने के लिए मैक्रो।

उत्पादन

Name is :Mohtashim company is TutorialsPoint age is 50

एक संरचनात्मक उदाहरण को संशोधित करना

एक उदाहरण को संशोधित करने के लिए, उदाहरण चर को उत्परिवर्तित चिह्नित किया जाना चाहिए। नीचे दिए गए उदाहरण ने कर्मचारी नामक एक संरचना की घोषणा की और शुरू की और बाद में आयु क्षेत्र के मूल्य को 50 से 40 तक संशोधित किया ।

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

उत्पादन

Name is :Mohtashim company is TutorialsPoint age is 40

एक समारोह में एक संरचना पारित करना

निम्न उदाहरण दिखाता है कि पैरामीटर के रूप में संरचना का उदाहरण कैसे पारित किया जाए। प्रदर्शन विधि पैरामीटर के रूप में एक कर्मचारी उदाहरण लेती है और विवरण प्रिंट करती है।

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

उत्पादन

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

एक फ़ंक्शन से रिटर्निंग स्ट्रक्चर

आइए एक फ़ंक्शन पर विचार करें who_is_elder () , जो दो कर्मचारियों की उम्र की तुलना करता है और बड़े को लौटाता है।

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

ये रहा पूरा कार्यक्रम -

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
}

उत्पादन

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

संरचना में विधि

कार्य की तरह तरीके हैं। वे प्रोग्रामिंग निर्देशों का एक तार्किक समूह हैं। विधियों के साथ घोषित किया जाता हैfnकीवर्ड। एक विधि का दायरा संरचना ब्लॉक के भीतर है।

संरचना ब्लॉक के बाहर तरीके घोषित किए जाते हैं। implकिसी संरचना के संदर्भ में एक विधि को परिभाषित करने के लिए कीवर्ड का उपयोग किया जाता है। एक विधि का पहला पैरामीटर हमेशा रहेगाself, जो संरचना के कॉलिंग उदाहरण का प्रतिनिधित्व करता है। एक संरचना के डेटा सदस्यों पर तरीके संचालित होते हैं।

एक विधि का आह्वान करने के लिए, हमें पहले संरचना को तत्काल तैयार करने की आवश्यकता है। संरचना के उदाहरण का उपयोग करके विधि को बुलाया जा सकता है।

वाक्य - विन्यास

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

चित्रण

निम्न उदाहरण एक संरचना को परिभाषित करता है आयत - साथ क्षेत्रों चौड़ाई और ऊंचाई । संरचना के संदर्भ में एक विधि क्षेत्र को परिभाषित किया गया है। क्षेत्र विधि स्वयं कीवर्ड के माध्यम से संरचना के क्षेत्रों तक पहुंचती है और एक आयत के क्षेत्र की गणना करती है।

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

उत्पादन

width is 10 height is 20 area of Rectangle is 200

संरचना में स्थैतिक विधि

स्थैतिक विधियों का उपयोग उपयोगिता विधियों के रूप में किया जा सकता है। संरचना के त्वरित होने से पहले ही ये विधियाँ मौजूद हैं। संरचना के नाम का उपयोग करके स्थैतिक तरीकों को लागू किया जाता है और इसे बिना उदाहरण के एक्सेस किया जा सकता है। सामान्य तरीकों के विपरीत, एक स्थैतिक विधि & self पैरामीटर नहीं लेगा ।

सिंटेक्स - एक स्थिर विधि की घोषणा

कार्यों और अन्य विधियों की तरह एक स्थिर विधि में वैकल्पिक रूप से पैरामीटर हो सकते हैं।

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

सिंटैक्स - एक स्थिर विधि का आह्वान

Structure_name :: वाक्य रचना एक स्थिर विधि का उपयोग किया जाता है।

structure_name::method_name(v1,v2)

चित्रण

निम्न उदाहरण एक फैक्ट्री क्लास के रूप में getInstance पद्धति का उपयोग करता है जो संरचना बिंदु के उदाहरण बनाता है और लौटाता है ।

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

उत्पादन

x =10 y=20

रस्ट प्रोग्रामिंग में, जब हमें संभावित वेरिएंट की सूची से एक मूल्य का चयन करना होता है तो हम एन्यूमरेशन डेटा प्रकारों का उपयोग करते हैं। Enumerated type को enum keyword का उपयोग करके घोषित किया जाता है । एनुम का वाक्य विन्यास निम्नलिखित है -

enum enum_name {
   variant1,
   variant2,
   variant3
}

चित्रण: एक गणना का उपयोग करना

उदाहरण एक Enum की घोषणा करता है - GenderCategory , जिसमें पुरुष और महिला के रूप में वेरिएंट हैं। प्रिंट! मैक्रो एनम का मान प्रदर्शित करता है। कंपाइलर एक त्रुटि को ट्रेट स्टैड में फेंक देगा :: fmt :: डिबग को GenderCategory के लिए लागू नहीं किया जाता है । इस त्रुटि को दबाने के लिए विशेषता # [व्युत्पन्न (डीबग)] का उपयोग किया जाता है।

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

उत्पादन

Male
Female

संरचना और Enum

निम्न उदाहरण एक व्यक्ति को संरचना को परिभाषित करता है। फ़ील्ड लिंग , GenderCategory (जो एक एनम है) के प्रकार का है और इसे पुरुष या महिला के मान के रूप में सौंपा जा सकता है ।

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

उदाहरण प्रकार के व्यक्ति के ऑब्जेक्ट p1 और P2 बनाता है और इनमें से प्रत्येक ऑब्जेक्ट के लिए विशेषताओं, नाम और लिंग को इनिशियलाइज़ करता है।

उत्पादन

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

विकल्प Enum

विकल्प रुस्ट मानक पुस्तकालय में एक पूर्वनिर्धारित एनम है। इस एनम के दो मूल्य हैं - कुछ (डेटा) और कोई नहीं।

वाक्य - विन्यास

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

यहाँ, T टाइप किसी भी प्रकार का मान दर्शाता है।

जंग अशक्त कीवर्ड का समर्थन नहीं करता है । मूल्य कोई नहीं , में enumOption , एक समारोह से इस्तेमाल किया जा सकता एक अशक्त मान देने के लिए। यदि लौटने के लिए डेटा है, तो फ़ंक्शन कुछ (डेटा) वापस कर सकता है ।

आइए इसे एक उदाहरण से समझते हैं -

कार्यक्रम एक प्रकार को परिभाषित करता है एक रिटर्न प्रकार विकल्प के साथ, is_even () है । फ़ंक्शन सत्यापित करता है कि मान दिया गया मान एक सम संख्या है। यदि इनपुट समान है, तो एक मान सत्य लौटाया जाता है, और फ़ंक्शन कोई भी नहीं लौटाता है ।

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

उत्पादन

None
Some(true)

मैच स्टेटमेंट और एनम

मैच बयान एक enum में संग्रहीत मूल्यों की तुलना करने के लिए इस्तेमाल किया जा सकता है। निम्न उदाहरण एक फ़ंक्शन को परिभाषित करता है, print_size , जो पैरामीटर के रूप में CarType Enum लेता है । फ़ंक्शन स्थिरांक के पूर्व-निर्धारित सेट के साथ पैरामीटर मानों की तुलना करता है और उपयुक्त संदेश प्रदर्शित करता है।

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

उत्पादन

Large sized Sports Utility car
Small sized car
medium sized car

विकल्प के साथ मिलान करें

Is_even फ़ंक्शन का उदाहरण , जो विकल्प प्रकार देता है, को नीचे दिए गए विवरण के साथ मैच स्टेटमेंट के साथ भी लागू किया जा सकता है -

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

उत्पादन

not even

डेटा प्रकार के साथ मैच और एनम

एक एनम के प्रत्येक संस्करण में डेटा प्रकार जोड़ना संभव है। निम्नलिखित उदाहरण में, enum के नाम और Usr_ID संस्करण क्रमशः स्ट्रिंग और पूर्णांक प्रकार के हैं। निम्न उदाहरण एक डेटा प्रकार वाले एनम के साथ मैच स्टेटमेंट के उपयोग को दर्शाता है।

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

उत्पादन

Name("Mohtashim")
Usr_ID(100)
Mohtashim

कोड के एक तार्किक समूह को एक मॉड्यूल कहा जाता है। एकाधिक मॉड्यूल को एक इकाई में संकलित किया जाता हैcrate। जंग कार्यक्रमों में एक बाइनरी टोकरा या एक पुस्तकालय टोकरा हो सकता है। एक बाइनरी क्रेट एक निष्पादन योग्य परियोजना है जिसमें एक मुख्य () विधि है। एक पुस्तकालय टोकरा घटकों का एक समूह है जिसे अन्य परियोजनाओं में पुन: उपयोग किया जा सकता है। एक बाइनरी क्रेट के विपरीत, एक लाइब्रेरी क्रेट में एक प्रवेश बिंदु (मुख्य () विधि) नहीं है। कार्गो टूल का उपयोग रस्ट में बक्से को प्रबंधित करने के लिए किया जाता है। उदाहरण के लिए, नेटवर्क मॉड्यूल में नेटवर्क से संबंधित फ़ंक्शन होते हैं और ग्राफिक्स मॉड्यूल में ड्राइंग-संबंधित फ़ंक्शन होते हैं। मॉड्यूल अन्य प्रोग्रामिंग भाषाओं में नामस्थान के समान हैं। थर्ड-पार्टी क्रेट को crates.io से कार्गो का उपयोग करके डाउनलोड किया जा सकता है ।

अनु क्रमांक शब्द और विवरण
1

crate

जंग में एक संकलन इकाई है; क्रेट को बाइनरी या लाइब्रेरी के लिए संकलित किया गया है।

2

cargo

टोकरा के लिए आधिकारिक जंग पैकेज प्रबंधन उपकरण।

3

module

तार्किक रूप से समूह एक टोकरा के भीतर कोड।

4

crates.io

आधिकारिक जंग पैकेज रजिस्ट्री।

वाक्य - विन्यास

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

मॉड्यूल सार्वजनिक या निजी हो सकते हैं। एक निजी मॉड्यूल में घटकों को अन्य मॉड्यूल द्वारा एक्सेस नहीं किया जा सकता है। डिफ़ॉल्ट रूप से जंग में मॉड्यूल निजी हैं। इसके विपरीत, एक सार्वजनिक मॉड्यूल में कार्यों को अन्य मॉड्यूल द्वारा एक्सेस किया जा सकता है। मॉड्यूल के साथ उपसर्ग किया जाना चाहिएpubकीवर्ड इसे सार्वजनिक करने के लिए। सार्वजनिक मॉड्यूल के भीतर कार्य को भी सार्वजनिक किया जाना चाहिए।

चित्रण: एक मॉड्यूल को परिभाषित करना

उदाहरण एक सार्वजनिक मॉड्यूल को परिभाषित करता है - फिल्में । मॉड्यूल में एक फ़ंक्शन प्ले () होता है जो एक पैरामीटर को स्वीकार करता है और इसके मूल्य को प्रिंट करता है।

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

उत्पादन

Playing movie Herold and Kumar

कीवर्ड का उपयोग करें

उपयोग कीवर्ड एक सार्वजनिक मॉड्यूल आयात करने के लिए मदद करता है।

वाक्य - विन्यास

use public_module_name::function_name;

चित्रण

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

उत्पादन

Playing movie Herold and Kumar

नेस्टेड मॉड्यूल

मॉड्यूल भी घोंसले के शिकार हो सकते हैं। कॉमेडी मॉड्यूल के भीतर नीडिंत है अंग्रेज़ी मॉड्यूल है, जो आगे में नीडिंत है फिल्में मॉड्यूल। नीचे दिया गया उदाहरण फिल्मों / अंग्रेजी / कॉमेडी मॉड्यूल के अंदर एक फंक्शन प्ले को परिभाषित करता है ।

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

उत्पादन

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

चित्रण - एक बायनरी क्रेट में लाइब्रेरी क्रेट और कंस्यूम बनाएं

आइए नाम से एक लाइब्रेरी क्रेट बनाएं movie_lib, जिसमें एक मॉड्यूल होता है movies। बनाने के लिएmovie_lib पुस्तकालय टोकरा, हम उपकरण का उपयोग करेंगे cargo

चरण 1 - प्रोजेक्ट फ़ोल्डर बनाएँ

एक फोल्डर मूवी-ऐप बनाएं, उसके बाद सब-फोल्डर मूवी-लिब । फ़ोल्डर और उप-फ़ोल्डर बनाए जाने के बाद, एक बनाएंsrcइस डायरेक्टरी में फोल्डर और एक कार्गो.टोमल फाइल। स्रोत कोड को src फ़ोल्डर में जाना चाहिए । Src फ़ोल्डर में lib.rs और movies.rs फ़ाइल बनाएँ। Cargo.toml फ़ाइल संस्करण संख्या, लेखक का नाम, आदि जैसे परियोजना के मेटाडाटा में शामिल होंगे

परियोजना निर्देशिका संरचना निम्नानुसार दिखाई जाएगी -

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

चरण 2 - प्रोजेक्ट मेटाडेटा जोड़ने के लिए Cargo.toml फ़ाइल को संपादित करें

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

चरण 3 - lib.rs फ़ाइल को संपादित करें।

इस फ़ाइल में निम्न मॉड्यूल परिभाषा जोड़ें।

pub mod movies;

उपरोक्त पंक्ति एक सार्वजनिक मॉड्यूल बनाती है - movies

चरण 4 - movies.rs फ़ाइल को संपादित करें

यह फ़ाइल मूवी मॉड्यूल के लिए सभी फ़ंक्शन को परिभाषित करेगी।

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

उपरोक्त कोड एक फ़ंक्शन को परिभाषित करता है play() यह एक पैरामीटर को स्वीकार करता है और इसे कंसोल पर प्रिंट करता है।

चरण 5 - पुस्तकालय टोकरा बनाएँ

का उपयोग कर एप्लिकेशन बनाएँ cargo buildयह सत्यापित करने के लिए कि लाइब्रेरी टोकरा ठीक से संरचित है या नहीं। सुनिश्चित करें कि आप प्रोजेक्ट के मूल में हैं - मूवी-ऐप फ़ोल्डर। बिल्ड के सफल होने पर निम्न संदेश टर्मिनल में प्रदर्शित होगा।

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

चरण 6 - एक परीक्षण एप्लिकेशन बनाएं

एक और फ़ोल्डर बनाएँ movie-lib-testकार्गो-फ़ाइल और src फ़ोल्डर के बाद मूवी-ऐप फ़ोल्डर में। इस परियोजना में मुख्य विधि होनी चाहिए क्योंकि यह एक द्विआधारी टोकरा है, जो पहले बनाए गए पुस्तकालय टोकरे का उपभोग करेगा। Src फ़ोल्डर में एक main.rs फ़ाइल बनाएँ। फ़ोल्डर संरचना के रूप में दिखाया जाएगा।

movie-app
   movie-lib 
   // already completed

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

चरण 7 - कार्गो.टॉम फाइल में निम्नलिखित जोड़ें

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

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

NOTE- लाइब्रेरी फ़ोल्डर का मार्ग निर्भरता के रूप में सेट किया गया है। निम्नलिखित आरेख दोनों परियोजनाओं की सामग्री को दर्शाता है।

चरण 8 - निम्नलिखित को main.rs फ़ाइल में जोड़ें

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

उपरोक्त कोड मूवीज_लिब नामक एक बाहरी पैकेज का आयात करता है। टोकरे के नाम को सत्यापित करने के लिए वर्तमान परियोजना के Cargo.toml की जांच करें।

चरण 9 - कार्गो बिल्ड और कार्गो रन का उपयोग

हम बाइनरी प्रोजेक्ट के निर्माण के लिए कार्गो बिल्ड और कार्गो रन का उपयोग करेंगे और इसे नीचे दिखाए अनुसार निष्पादित करेंगे -

रस्ट का मानक संग्रह पुस्तकालय सबसे सामान्य सामान्य प्रयोजन प्रोग्रामिंग डेटा संरचनाओं का कुशल कार्यान्वयन प्रदान करता है। इस अध्याय में आमतौर पर उपयोग किए जाने वाले संग्रहों के कार्यान्वयन पर चर्चा की गई है - वेक्टर, हैशपैप और हैशसेट।

वेक्टर

एक वेक्टर एक resizable सरणी है। यह सन्निहित स्मृति ब्लॉकों में मूल्यों को संग्रहीत करता है। वैक्टर बनाने के लिए पूर्वनिर्धारित संरचना Vec का उपयोग किया जा सकता है। एक वेक्टर की कुछ महत्वपूर्ण विशेषताएं हैं -

  • एक वेक्टर रनटाइम पर बढ़ या सिकुड़ सकता है।

  • एक वेक्टर एक सजातीय संग्रह है।

  • एक वेक्टर एक विशेष क्रम में तत्वों के अनुक्रम के रूप में डेटा संग्रहीत करता है। वेक्टर में प्रत्येक तत्व को एक अद्वितीय सूचकांक संख्या दी गई है। सूचकांक 0 से शुरू होता है और n-1 तक जाता है जहां, n संग्रह का आकार है। उदाहरण के लिए, 5 तत्वों के संग्रह में, पहला तत्व सूचकांक 0 पर होगा और अंतिम तत्व सूचकांक 4 पर होगा।

  • एक वेक्टर केवल (या पास) के अंत में मूल्यों को जोड़ देगा। दूसरे शब्दों में, स्टैक को लागू करने के लिए एक वेक्टर का उपयोग किया जा सकता है।

  • एक वेक्टर के लिए मेमोरी ढेर में आवंटित की जाती है।

सिंटेक्स - एक वेक्टर बनाना

let mut instance_name = Vec::new();

Vec संरचना की स्थिर विधि नई () का उपयोग वेक्टर उदाहरण बनाने के लिए किया जाता है।

वैकल्पिक रूप से, वेक्टर का उपयोग करके एक वेक्टर भी बनाया जा सकता है! मैक्रो। सिंटैक्स नीचे दिया गया है -

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

निम्न तालिका Vec संरचना के कुछ सामान्यतः उपयोग किए जाने वाले कार्यों को सूचीबद्ध करती है।

अनु क्रमांक तरीका हस्ताक्षर और विवरण
1 नया()

pub fn new()->Vect

एक नया, खाली वीईसी का निर्माण करता है। जब तक तत्वों को उस पर धकेल नहीं दिया जाएगा तब तक वेक्टर आवंटित नहीं होगा।

2 धक्का दें()

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

एक संग्रह के पीछे एक तत्व जोड़ता है।

3 हटाना()

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

वेक्टर के भीतर स्थिति सूचकांक में तत्व को निकालता है और छोड़ता है, इसके बाद सभी तत्वों को बाईं ओर स्थानांतरित करता है।

4 शामिल ()

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

सही है अगर टुकड़ा में दिए गए मान के साथ एक तत्व होता है।

5 लेन ()

pub fn len(&self) -> usize

वेक्टर में तत्वों की संख्या लौटाता है, जिसे इसकी 'लंबाई' भी कहा जाता है।

चित्रण: एक वेक्टर बनाना - नया ()

एक सदिश बनाने के लिए, हम स्थिर विधि का उपयोग नई -

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

उपर्युक्त उदाहरण स्टैटिक मेथड न्यू () का उपयोग करके एक वेक्टर बनाता है जिसे संरचना Vec में परिभाषित किया गया है । धक्का (वैल) समारोह संग्रह करने के लिए पैरामीटर के रूप में पारित कर दिया मूल्य जोड़ देता है। लेन () फ़ंक्शन वेक्टर की लंबाई लौटाता है।

उत्पादन

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

चित्रण: एक वेक्टर बनाना - vec! मैक्रो

निम्न कोड vec का उपयोग करके एक वेक्टर बनाता है! मैक्रो। वेक्टर का डेटा प्रकार पहला मान है जो इसे सौंपा गया है।

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

उत्पादन

[1, 2, 3]

जैसा कि पहले उल्लेख किया गया है, एक वेक्टर में केवल एक ही डेटा प्रकार के मान हो सकते हैं। निम्नलिखित स्निपेट एक त्रुटि फेंक देगा [E0308]: बेमेल प्रकार की त्रुटि।

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

चित्रण: धक्का ()

एक संग्रह के अंत में एक तत्व जोड़ता है।

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

उत्पादन

[20, 30, 40]

चित्रण: हटाएं ()

वेक्टर के भीतर स्थिति सूचकांक में तत्व को निकालता है और छोड़ता है, इसके बाद सभी तत्वों को बाईं ओर स्थानांतरित करता है।

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

उत्पादन

[10, 30]

चित्रण - शामिल ()

सही है अगर टुकड़ा में दिए गए मान के साथ एक तत्व है -

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

उत्पादन

found 10
[10, 20, 30]

चित्रण: लेन ()

वेक्टर में तत्वों की संख्या लौटाता है, जिसे इसकी 'लंबाई' भी कहा जाता है।

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

उत्पादन

size of vector is :3

एक वेक्टर से मानों तक पहुंचना

एक वेक्टर में अलग-अलग तत्वों को उनके संबंधित इंडेक्स नंबरों का उपयोग करके एक्सेस किया जा सकता है। निम्नलिखित उदाहरण एक वेक्टर विज्ञापन बनाता है जो पहले तत्व के मूल्य को प्रिंट करता है।

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

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

एक सदिश में मान भी संग्रह के संदर्भ का उपयोग करके प्राप्त किया जा सकता है।

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

उत्पादन

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

हैश मैप

एक मानचित्र कुंजी-मूल्य जोड़े (प्रविष्टियों कहा जाता है) का एक संग्रह है। किसी मानचित्र में दो प्रविष्टियों में एक ही कुंजी नहीं हो सकती है। संक्षेप में, एक नक्शा एक लुकअप टेबल है। HashMap हैश तालिका में कुंजी और मान संग्रहीत करता है। प्रविष्टियों को एक अनियंत्रित क्रम में संग्रहीत किया जाता है। कुंजी का उपयोग हाशप में मानों को खोजने के लिए किया जाता है। हाशप संरचना में परिभाषित किया गया हैstd::collectionsमापांक। यह मॉड्यूल HashMap संरचना तक पहुंचने के लिए स्पष्ट रूप से आयात किया जाना चाहिए।

सिंटेक्स: एक हैशपैप बनाना

let mut instance_name = HashMap::new();

HashMap संरचना की स्थिर विधि नई () का उपयोग HashMap ऑब्जेक्ट बनाने के लिए किया जाता है। यह विधि एक खाली HashMap बनाता है।

हाशप के आमतौर पर उपयोग किए जाने वाले कार्यों की चर्चा नीचे की गई है -

अनु क्रमांक तरीका हस्ताक्षर और विवरण
1 डालने ()

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

एक कुंजी / मूल्य जोड़ी सम्मिलित करता है, यदि कोई कुंजी नहीं है तो कोई भी वापस नहीं किया जाता है। अपडेट करने के बाद, पुराना मान वापस कर दिया जाता है।

2 लेन ()

pub fn len(&self) -> usize

नक्शे में तत्वों की संख्या लौटाता है।

3 प्राप्त()

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

कुंजी के अनुरूप मान का संदर्भ देता है।

4 आईटीईआर ()

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

एक पुनरावृत्त व्यक्ति सभी प्रमुख-मूल्य जोड़े का मनमाना क्रम में दौरा करता है। इटरेटर तत्व प्रकार (& a K, & 'a V) है।

5 contains_key

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

यदि मानचित्र में निर्दिष्ट कुंजी के लिए मान है तो यह सही है।

6 हटाना()

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

यदि नक्शे में कुंजी पहले थी, तो संग्रहीत कुंजी और मान लौटाते हुए मानचित्र से एक कुंजी निकालता है।

चित्रण: सम्मिलित करें ()

हाशपैप में एक कुंजी / मान युग्म सम्मिलित करता है।

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

उपरोक्त कार्यक्रम एक हैशपॉप बनाता है और इसे 2 कुंजी-मूल्य जोड़े के साथ आरंभ करता है।

उत्पादन

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

चित्रण: लेन ()

नक्शे में तत्वों की संख्या लौटाता है

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

उपरोक्त उदाहरण एक HashMap बनाता है और इसमें कुल तत्वों को प्रिंट करता है।

उत्पादन

size of map is 2

चित्रण - प्राप्त ()

कुंजी के अनुरूप मान का संदर्भ देता है। निम्न उदाहरण हाशप में कुंजी KL के लिए मान को पुनः प्राप्त करता है ।

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

उत्पादन

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

चित्रण - iter ()

एक पुनरावृत्त रिटर्न देता है जिसमें सभी कुंजी-मूल्य जोड़े के संदर्भ में एक मनमाना क्रम होता है।

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

उत्पादन

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

चित्रण: समाहित_की ()

यदि मानचित्र में निर्दिष्ट कुंजी के लिए मान है तो यह सही है।

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

उत्पादन

found key

चित्रण: हटाएं ()

नक्शे से एक कुंजी निकालता है।

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

उत्पादन

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

HashSet

HashSet T के अनूठे मूल्यों का एक समूह है। मानों को जोड़ना और हटाना तेजी से होता है, और यह पूछना तेजी से होता है कि सेट में कोई मान दिया गया है या नहीं। HashSet संरचना को std :: collection मॉड्यूल में परिभाषित किया गया है। इस मॉड्यूल को हैशसेट संरचना तक पहुंचने के लिए स्पष्ट रूप से आयात किया जाना चाहिए।

सिंटेक्स: एक हैशसेट बनाना

let mut hash_set_name = HashSet::new();

HashSet संरचना का नया , नया तरीका, HashSet बनाने के लिए उपयोग किया जाता है। यह विधि एक खाली HashSet बनाता है।

निम्न तालिका हस्सेट संरचना के कुछ सामान्य रूप से उपयोग किए गए तरीकों को सूचीबद्ध करती है।

अनु क्रमांक तरीका हस्ताक्षर और विवरण
1 डालने ()

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

सेट के लिए एक मान जोड़ता है। यदि सेट में यह मान मौजूद नहीं है, तो सत्य को और गलत माना जाता है।

2 लेन ()

pub fn len(&self) -> usize

सेट में तत्वों की संख्या लौटाता है।

3 प्राप्त()

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

दिए गए मान के बराबर होने पर, सेट में दिए गए मान का संदर्भ देता है।

4 आईटीईआर ()

pub fn iter(&self) -> Iter

मनमानी क्रम में सभी तत्वों पर जाकर एक पुनरावृति देता है। इटरेटर तत्व प्रकार है & 'a T।

5 contains_key

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

यदि सेट में कोई मान है तो यह सही है।

6 हटाना()

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

सेट से एक मान निकालता है। यदि सेट में मान मौजूद था, तो यह सच है।

चित्रण - सम्मिलित करें ()

सेट के लिए एक मान जोड़ता है। A HashSet संग्रह में डुप्लिकेट मान नहीं जोड़ता है।

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

उत्पादन

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

चित्रण: लेन ()

सेट में तत्वों की संख्या लौटाता है।

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

उत्पादन

size of the set is 3

चित्रण - iter ()

मनमानी क्रम में सभी तत्वों का दौरा करने वाले एक पुनरावृत्तिकर्ता को हटा देता है।

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

उत्पादन

TutorialsPoint
Mohtashim
Kannan

चित्रण: प्राप्त ()

सेट में मान का संदर्भ देता है, यदि कोई हो, जो दिए गए मूल्य के बराबर है।

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

उत्पादन

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

चित्रण - शामिल ()

यदि सेट में कोई मान है तो यह सही है।

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

उत्पादन

found name

चित्रण: हटाएं ()

सेट से एक मान निकालता है।

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

उत्पादन

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

जंग में, त्रुटियों को दो प्रमुख श्रेणियों में वर्गीकृत किया जा सकता है जैसा कि नीचे दी गई तालिका में दिखाया गया है।

अनु क्रमांक नाम और विवरण प्रयोग
1

Recoverable

त्रुटियां जिन्हें संभाला जा सकता है

नतीजा एनुम
2

UnRecoverable

त्रुटियां जिन्हें संभाला नहीं जा सकता है

घबराहट

पुनर्प्राप्त करने योग्य त्रुटि एक त्रुटि है जिसे ठीक किया जा सकता है। एक प्रोग्राम विफल ऑपरेशन को फिर से कर सकता है या एक वैकल्पिक पाठ्यक्रम निर्दिष्ट कर सकता है जब यह एक पुनर्प्राप्त करने योग्य त्रुटि का सामना करता है। पुनर्प्राप्त करने योग्य त्रुटियों के कारण अचानक प्रोग्राम विफल नहीं होता है। पुनर्प्राप्त करने योग्य त्रुटि का एक उदाहरण फ़ाइल नहीं मिली त्रुटि है।

अपरिवर्तनीय त्रुटियां एक कार्यक्रम को अचानक विफल करने का कारण बनती हैं। यदि कोई अपरिवर्तनीय त्रुटि होती है, तो एक कार्यक्रम अपनी सामान्य स्थिति में वापस नहीं आ सकता है। यह विफल ऑपरेशन को पुन: प्रयास नहीं कर सकता है या त्रुटि को पूर्ववत नहीं कर सकता है। एक अपरिवर्तनीय त्रुटि का एक उदाहरण एक सरणी के अंत से परे एक स्थान तक पहुंचने की कोशिश कर रहा है।

अन्य प्रोग्रामिंग भाषाओं के विपरीत, रस्ट के पास अपवाद नहीं हैं। यह पुनर्प्राप्त करने योग्य त्रुटियों के लिए एक enum परिणाम <T, E> देता है, जबकि यह कॉल करता हैpanicमैक्रो यदि प्रोग्राम एक अपरिवर्तनीय त्रुटि का सामना करता है। आतंक मैक्रो कार्यक्रम अचानक बाहर निकलने के लिए कारण बनता है।

आतंक मैक्रों और अपरिवर्तनीय त्रुटियां

घबड़ाहट! मैक्रो प्रोग्राम को तुरंत समाप्त करने और प्रोग्रामर के कॉलर को प्रतिक्रिया प्रदान करने की अनुमति देता है। इसका उपयोग तब किया जाना चाहिए जब कोई कार्यक्रम अपरिवर्तनीय स्थिति तक पहुंच जाए।

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

उपरोक्त उदाहरण में, यह कार्यक्रम तुरंत समाप्त हो जाएगा जब यह आतंक का सामना करेगा ! मैक्रो।

उत्पादन

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

चित्रण: घबराहट! मैक्रो

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

आउटपुट नीचे दिखाया गया है -

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.

एक कार्यक्रम आतंक को आह्वान कर सकता है! मैक्रो यदि व्यावसायिक नियमों का उल्लंघन किया जाता है जैसा कि नीचे दिए गए उदाहरण में दिखाया गया है -

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

उपरोक्त उदाहरण त्रुटि देता है यदि चर को दिया गया मान विषम है।

उत्पादन

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

परिणाम Enum और पुनर्प्राप्त करने योग्य त्रुटियाँ

Enum परिणाम - <T, E> का उपयोग पुनर्प्राप्त करने योग्य त्रुटियों को संभालने के लिए किया जा सकता है। इसके दो संस्करण हैं -OK तथा ErrT तथा E सामान्य प्रकार के पैरामीटर हैं। T उस मूल्य के प्रकार का प्रतिनिधित्व करता है जिसे ओके वेरिएंट के भीतर एक सफलता के मामले में वापस किया जाएगा, और E त्रुटि के प्रकार का प्रतिनिधित्व करता है जिसे एरर वेरिएंट में विफलता के मामले में वापस किया जाएगा।

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

आइए इसे एक उदाहरण की मदद से समझते हैं -

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

यदि फ़ाइल पहले से मौजूद है और फ़ाइल नहीं मिली तो Err (त्रुटि) ठीक होने पर प्रोग्राम ठीक है (फ़ाइल)

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

आइए अब देखते हैं कि एरर वेरिएंट को कैसे संभालना है।

निम्न उदाहरण फ़ाइल का उपयोग करते हुए फ़ाइल खोलते समय दी गई एक त्रुटि को संभालता है match बयान

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- कार्यक्रम मुख्य घटना के अंत प्रिंट करता है, हालांकि फ़ाइल नहीं मिली। इसका मतलब यह है कि कार्यक्रम को त्रुटिपूर्ण तरीके से नियंत्रित किया गया है।

उत्पादन

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

चित्रण

Is_even समारोह एक त्रुटि देता है यदि संख्या सम संख्या नहीं है। मुख्य () फ़ंक्शन इस त्रुटि को संभालता है।

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- के बाद से मुख्य कार्य हैंडल शान से त्रुटि, अंत के मुख्य बयान छपा है।

उत्पादन

Error msg is NOT_AN_EVEN
end of main

खोलना () और उम्मीद ()

मानक पुस्तकालय में सहायक विधियों की एक जोड़ी होती है, जो दोनों को लागू करती है - परिणाम <T, E> और विकल्प <T> लागू करें। आप उनका उपयोग उन त्रुटि मामलों को सरल बनाने में कर सकते हैं जहां आप वास्तव में चीजों के असफल होने की उम्मीद नहीं करते हैं। एक विधि से सफलता के मामले में, "परिणाम" को वास्तविक परिणाम निकालने के लिए प्रयोग किया जाता है।

अनु क्रमांक तरीका हस्ताक्षर और विवरण
1 खोलना

unwrap(self): T

स्वयं को ओके / कुछ होने की उम्मीद करता है और भीतर निहित मूल्य लौटाता है। अगर यह होता हैErr या None इसके बजाय, यह प्रदर्शित की गई त्रुटि की सामग्री से घबराहट पैदा करता है।

2 उम्मीद

expect(self, msg: &str): T

अनवाप्स की तरह व्यवहार करता है, सिवाय इसके कि यह त्रुटि की सामग्री के अलावा पैन करने से पहले एक कस्टम संदेश को आउटपुट करता है।

खोलना ()

एक्सट्रैप () फ़ंक्शन वास्तविक परिणाम देता है जो एक ऑपरेशन सफल होता है। यदि कोई कार्रवाई विफल हो जाती है तो यह डिफ़ॉल्ट त्रुटि संदेश के साथ घबराहट देता है। यह फ़ंक्शन मैच स्टेटमेंट के लिए एक शॉर्टहैंड है। यह नीचे दिए गए उदाहरण में दिखाया गया है -

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

एक विषम संख्या पास करने के लिए उपरोक्त कोड को संशोधित करें is_even() समारोह।

खोलना () समारोह आतंक और एक डिफ़ॉल्ट त्रुटि संदेश लौटने जैसा कि नीचे दिखाया जाएगा

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

उम्मीद ()

घबराहट की स्थिति में प्रोग्राम एक कस्टम त्रुटि संदेश लौटा सकता है। इसे निम्न उदाहरण में दिखाया गया है -

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

फ़ंक्शन की अपेक्षा () अनट्रैप () के समान है। एकमात्र अंतर यह है कि एक कस्टम त्रुटि संदेश उम्मीद का उपयोग करके प्रदर्शित किया जा सकता है।

उत्पादन

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.

जेनरिक विभिन्न प्रकारों के साथ कई संदर्भों के लिए कोड लिखने की सुविधा है। जंग में, जेनेरिक डेटा प्रकारों और लक्षणों के मानकीकरण का उल्लेख करते हैं। जेनरिक कोड दोहराव को कम करके और टाइप-सुरक्षा प्रदान करके अधिक संक्षिप्त और स्वच्छ कोड लिखने की अनुमति देता है। जेनरिक की अवधारणा को विधियों, कार्यों, संरचनाओं, गणना, संग्रह और लक्षणों पर लागू किया जा सकता है।

<T> syntaxप्रकार पैरामीटर के रूप में जाना जाता है, एक सामान्य निर्माण घोषित करने के लिए उपयोग किया जाता है। T किसी भी डेटा-प्रकार का प्रतिनिधित्व करता है।

चित्रण: सामान्य संग्रह

निम्नलिखित उदाहरण एक वेक्टर घोषित करता है जो केवल पूर्णांकों को संग्रहीत कर सकता है।

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

उत्पादन

[20, 30, 40]

निम्नलिखित स्निपेट पर विचार करें -

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

उपरोक्त उदाहरण से पता चलता है कि पूर्णांक प्रकार का वेक्टर केवल पूर्णांक मानों को संग्रहीत कर सकता है। इसलिए, यदि हम संग्रह में एक स्ट्रिंग मान को पुश करने का प्रयास करते हैं, तो संकलक एक त्रुटि लौटाएगा। जेनरिक संग्रह को अधिक सुरक्षित बनाते हैं।

चित्रण: सामान्य संरचना

प्रकार पैरामीटर एक प्रकार का प्रतिनिधित्व करता है, जिसे कंपाइलर बाद में भर देगा।

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

उपरोक्त उदाहरण डेटा नामक एक सामान्य संरचना की घोषणा करता है । <टी> प्रकार कुछ डेटा प्रकार इंगित करता है। मुख्य () संरचना के एक पूर्णांक उदाहरण और एक स्ट्रिंग उदाहरण - समारोह दो उदाहरणों पैदा करता है।

उत्पादन

value is :350
value is :Tom

लक्षण

कई संरचनाओं में व्यवहार (विधियों) के एक मानक सेट को लागू करने के लिए लक्षणों का उपयोग किया जा सकता है। लक्षण जैसे हैंinterfacesऑब्जेक्ट ओरिएंटेड प्रोग्रामिंग में। लक्षण का सिंटैक्स नीचे दिखाया गया है -

एक लक्षण घोषित करें

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

लक्षण में ठोस विधियाँ (शरीर के साथ विधियाँ) या सार विधियाँ (शरीर के बिना विधियाँ) हो सकती हैं। एक ठोस विधि का उपयोग करें यदि विधि परिभाषा को ट्रिट को लागू करने वाली सभी संरचनाओं द्वारा साझा किया जाएगा। हालांकि, एक संरचना विशेषता द्वारा परिभाषित फ़ंक्शन को ओवरराइड करने का विकल्प चुन सकती है।

अमूर्त विधियों का उपयोग करें यदि विधि की परिभाषा कार्यान्वयन संरचनाओं के लिए भिन्न होती है।

सिंटैक्स - एक गुण को लागू करें

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

निम्नलिखित उदाहरण एक विधि प्रिंट () के साथ एक मुद्रण योग्य को परिभाषित करता है , जिसे संरचना पुस्तक द्वारा कार्यान्वित किया जाता है ।

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

उत्पादन

Printing book with id:1001 and name Rust in Action

सामान्य कार्य

उदाहरण एक जेनेरिक फ़ंक्शन को परिभाषित करता है जो इसे पारित किए गए पैरामीटर को प्रदर्शित करता है। पैरामीटर किसी भी प्रकार का हो सकता है। पैरामीटर्स के प्रकार को डिस्प्ले विशेषता को लागू करना चाहिए ताकि इसका मूल्य प्रिंटलाइन द्वारा प्रिंट किया जा सके! मैक्रो।

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

उत्पादन

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

यह अध्याय चर्चा करता है कि मानक इनपुट (कीबोर्ड) से मूल्यों को कैसे स्वीकार किया जाए और मानों को मानक आउटपुट (कंसोल) पर प्रदर्शित किया जाए। इस अध्याय में, हम कमांड लाइन के तर्कों को पारित करने पर भी चर्चा करेंगे।

पाठक और लेखक प्रकार

इनपुट और आउटपुट के लिए रस्ट के मानक पुस्तकालय की विशेषताएं दो लक्षणों के आसपास आयोजित की जाती हैं -

  • Read
  • Write
अनु क्रमांक विशेषता और विवरण उदाहरण
1

Read

प्रकार जो कार्यान्वित करते हैं पढ़ें में बाइट-उन्मुख इनपुट के लिए विधियां हैं। वे पाठक कहलाते हैं

Stdin, फ़ाइल
2

Write

लागू करने वाले प्रकार बाइट-ओरिएंटेड और UTF-8 टेक्स्ट आउटपुट दोनों को समर्थन लिखें। वे लेखक कहलाते हैं।

Stdout, फ़ाइल

लक्षण पढ़ें

Readersऐसे घटक हैं जिनसे आपका प्रोग्राम बाइट्स पढ़ सकता है। उदाहरणों में कीबोर्ड, फाइल आदि से रीडिंग इनपुट शामिल हैंread_line() इस विशेषता का उपयोग फ़ाइल या मानक इनपुट स्ट्रीम से, डेटा को एक समय में एक पंक्ति पढ़ने के लिए किया जा सकता है।

अनु क्रमांक विशेषता विधि और विवरण
1 पढ़ें

read_line(&mut line)->Result

पाठ की एक पंक्ति पढ़ता है और इसे पंक्ति में जोड़ता है, जो एक स्ट्रिंग है। वापसी मूल्य एक io :: परिणाम है, बाइट्स की संख्या पढ़ी जाती है।

चित्रण - कंसोल से पढ़ना - स्टडिन ()

रस्ट प्रोग्राम को रनटाइम पर उपयोगकर्ता से मान स्वीकार करना पड़ सकता है। निम्न उदाहरण मानक इनपुट (कीबोर्ड) से मूल्यों को पढ़ता है और इसे कंसोल पर प्रिंट करता है।

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

Stdin () समारोह मौजूदा प्रक्रिया है, जो करने के लिए की मानक इनपुट स्ट्रीम में एक संभाल रिटर्न read_line समारोह लागू किया जा सकता। यह फ़ंक्शन इनपुट बफर में मौजूद सभी वर्णों को पढ़ने की कोशिश करता है जब यह एक अंत-पंक्ति चरित्र का सामना करता है।

उत्पादन

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

लक्षण लिखें

Writersऐसे घटक हैं जो आपके प्रोग्राम को बाइट्स लिख सकते हैं। उदाहरणों में कंसोल पर प्रिंटिंग मान, फ़ाइलों को लिखना, आदि शामिल हैं। इस विशेषता के लेखन () पद्धति का उपयोग किसी फ़ाइल या मानक आउटपुट स्ट्रीम में डेटा लिखने के लिए किया जा सकता है।

अनु क्रमांक विशेषता विधि और विवरण
1 लिखो

write(&buf)->Result

स्लाइस बफ़ में कुछ बाइट्स को अंतर्निहित धारा में लिखते हैं। यह एक io :: परिणाम, लिखित बाइट्स की संख्या देता है।

चित्रण - सांत्वना के लिए लेखन - stdout ()

प्रिंट! या प्रिन्टल! कंसोल पर टेक्स्ट प्रदर्शित करने के लिए मैक्रोज़ का उपयोग किया जा सकता है। हालाँकि, आप कुछ पाठ को मानक आउटपुट में प्रदर्शित करने के लिए लेखन () मानक लाइब्रेरी फ़ंक्शन का भी उपयोग कर सकते हैं ।

इसे समझने के लिए एक उदाहरण पर विचार करते हैं।

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

उत्पादन

Tutorials Point
bytes written 15

Stdout () मानक पुस्तकालय समारोह, मौजूदा प्रक्रिया के मानक आउटपुट स्ट्रीम में एक संभाल देता है जो करने के लिएwriteफ़ंक्शन लागू किया जा सकता है। लेखन () विधि एक एनुम, परिणाम देता है। उच्छ्वास () एन्यूमरेशन से वास्तविक परिणाम निकालने के लिए एक सहायक विधि है। यदि कोई त्रुटि होती है, तो अनचाहा तरीका आतंक भेज देगा।

NOTE - फाइल IO अगले अध्याय में चर्चा की गई है।

कमांडलाइन तर्क

कमांडलाइन तर्क को निष्पादित करने से पहले एक कार्यक्रम में पारित किया जाता है। वे कार्यों के लिए पारित मापदंडों की तरह हैं। कमांडलाइन मापदंडों का उपयोग मुख्य () फ़ंक्शन के मानों को पारित करने के लिए किया जा सकता है । std::env::args() कमांडलाइन तर्क देता है।

चित्रण

निम्न उदाहरण मुख्य () फ़ंक्शन के लिए कमांडलाइन तर्क के रूप में मानों को पास करता है। कार्यक्रम एक फ़ाइल नाम 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
   }
}

एक बार संकलित होने पर प्रोग्राम एक फ़ाइल main.exe उत्पन्न करेगा । एकाधिक कमांड लाइन पैरामीटर को अंतरिक्ष से अलग किया जाना चाहिए। टर्मिनल से main.exe को मुख्य ex.exe के रूप में निष्पादित करें ।

NOTE- हैलो और ट्यूटोरियलस्पेललाइन कमांडलाइन तर्क हैं।

उत्पादन

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

आउटपुट 3 तर्क दिखाता है क्योंकि main.exe पहला तर्क है।

चित्रण

निम्न प्रोग्राम कमांडलाइन तर्कों के रूप में पारित मूल्यों के योग की गणना करता है। अंतरिक्ष द्वारा अलग किए गए एक सूची पूर्णांक मान को प्रोग्राम में पास किया जाता है।

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

प्रोग्राम को main.exe 1 2 3 4 के रूप में निष्पादित करने पर, आउटपुट होगा -

No of elements in arguments is :5
sum is 10

कंसोल के लिए पढ़ने और लिखने के अलावा, रस्ट फाइलों को पढ़ने और लिखने की अनुमति देता है।

फ़ाइल संरचना एक फ़ाइल का प्रतिनिधित्व करती है। यह एक प्रोग्राम को फाइल पर रीड-राइट ऑपरेशन करने की अनुमति देता है। फ़ाइल संरचना में सभी विधियाँ io का एक प्रकार लौटाती हैं :: परिणाम गणना।

फ़ाइल संरचना के आमतौर पर उपयोग किए जाने वाले तरीके नीचे दी गई तालिका में सूचीबद्ध हैं -

अनु क्रमांक मापांक तरीका हस्ताक्षर विवरण
1 std :: FS :: फ़ाइल खुला हुआ() पब fn खुला <P: AsRef> (पथ: P) -> परिणाम रीड-ओनली मोड में किसी फ़ाइल को खोलने के लिए ओपन स्टैटिक विधि का उपयोग किया जा सकता है।
2 std :: FS :: फ़ाइल सृजन करना() pub fn create <P: AsRef> (पाथ: P) -> परिणाम स्टेटिक मेथड केवल-राइट मोड में एक फाइल खोलता है। यदि फ़ाइल पहले से मौजूद है, तो पुरानी सामग्री नष्ट हो गई है। अन्यथा, एक नई फ़ाइल बनाई जाती है।
3 std :: FS :: remove_file फ़ाइल को हटाएं() pub fn remove_file <P: AsRef> (पथ: P) -> परिणाम <()> फाइलसिस्टम से एक फाइल को हटाता है। इस बात की कोई गारंटी नहीं है कि फ़ाइल को तुरंत हटा दिया गया है।
4 std :: FS :: OpenOptions संलग्न () पब एफएन एपेंड (और म्यूट सेल्फ, एपेंड: बूल) -> और म्यूट ओपनऑक्शंस फ़ाइल के एपेंड मोड के लिए विकल्प सेट करता है।
5 std :: कब :: लिखता है write_all () fn write_all (और mut self, buf: & [u8]) -> परिणाम <()> इस लेखन में एक संपूर्ण बफ़र लिखने का प्रयास।
6 std :: कब :: पढ़ें read_to_string () fn read_to_string (और mut self, buf: & mut String) -> परिणाम इस स्रोत में EOF तक सभी बाइट्स पढ़ता है, उन्हें buf में जोड़ रहा है।

एक फ़ाइल में लिखें

फाइल कैसे लिखें, यह समझने के लिए एक उदाहरण देखते हैं।

निम्न प्रोग्राम एक फ़ाइल बनाता है 'data.txt'। फ़ाइल बनाने के लिए बनाने () विधि का उपयोग किया जाता है। यदि फ़ाइल सफलतापूर्वक बनाई गई है, तो विधि फ़ाइल हैंडल लौटाती है। अंतिम पंक्ति write_all फ़ंक्शन नई बनाई गई फ़ाइल में बाइट्स लिखेगा। यदि कोई भी ऑपरेशन विफल हो जाता है, तो उम्मीद () फ़ंक्शन एक त्रुटि संदेश देता है।

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

उत्पादन

data written to file

एक फ़ाइल से पढ़ें

निम्न प्रोग्राम एक फ़ाइल data.txt में सामग्री को पढ़ता है और इसे कंसोल पर प्रिंट करता है। "ओपन" फ़ंक्शन का उपयोग किसी मौजूदा फ़ाइल को खोलने के लिए किया जाता है। फ़ाइल के लिए एक निरपेक्ष या सापेक्ष पथ एक पैरामीटर के रूप में खुले () फ़ंक्शन को पास किया जाता है। यदि फ़ाइल मौजूद नहीं है, या यदि यह किसी भी कारण से सुलभ नहीं है, तो खुला () फ़ंक्शन अपवाद को फेंकता है। यदि यह सफल होता है, तो ऐसी फ़ाइल का एक फ़ाइल हैंडल "फ़ाइल" चर को सौंपा गया है।

"फ़ाइल" हैंडल के "read_to_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);
}

उत्पादन

Hello World
TutorialsPoint

एक फ़ाइल हटाएँ

निम्न उदाहरण फ़ाइल को हटाने के लिए remove_file () फ़ंक्शन का उपयोग करता है। यदि कोई त्रुटि होती है, तो उम्मीद () फ़ंक्शन कस्टम संदेश देता है।

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

उत्पादन

file is removed

डेटा को एक फ़ाइल में जोड़ें

एपेंड () फ़ंक्शन फ़ाइल के अंत में डेटा लिखता है। यह नीचे दिए गए उदाहरण में दिखाया गया है -

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

उत्पादन

file append success

एक फ़ाइल की प्रतिलिपि बनाएँ

निम्न उदाहरण एक फ़ाइल में सामग्री को एक नई फ़ाइल में कॉपी करता है।

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

उपरोक्त प्रोग्राम को main.exe data.txt datacopy.txt के रूप में निष्पादित करें । फ़ाइल को निष्पादित करते समय दो कमांड लाइन तर्क दिए गए हैं -

  • स्रोत फ़ाइल का पथ
  • गंतव्य फ़ाइल

RUST के लिए कार्गो पैकेज मैनेजर है। यह एक उपकरण की तरह काम करता है और जंग परियोजनाओं का प्रबंधन करता है।

कुछ सामान्य रूप से प्रयुक्त कार्गो कमांड नीचे दी गई तालिका में सूचीबद्ध हैं -

अनु क्रमांक कमांड और विवरण
1

cargo build

वर्तमान परियोजना संकलित करता है।

2

cargo check

वर्तमान परियोजना का विश्लेषण करता है और त्रुटियों की रिपोर्ट करता है, लेकिन ऑब्जेक्ट फ़ाइलों का निर्माण नहीं करता है।

3

cargo run

Src / main.rs बनाता और निष्पादित करता है।

4

cargo clean

लक्ष्य निर्देशिका को निकालता है।

5

cargo update

अपडेट निर्भरता कार्गो में सूचीबद्ध हैं।

6

cargo new

एक नया कार्गो प्रोजेक्ट बनाता है।

कार्गो थर्ड पार्टी लाइब्रेरी को डाउनलोड करने में मदद करता है। इसलिए, यह एक पैकेज मैनेजर की तरह काम करता है। आप अपनी खुद की लाइब्रेरी भी बना सकते हैं। जब आप Rust स्थापित करते हैं तो कार्गो डिफ़ॉल्ट रूप से स्थापित होता है।

एक नई कार्गो परियोजना बनाने के लिए, हम नीचे दिए गए आदेशों का उपयोग कर सकते हैं।

एक बाइनरी टोकरा बनाएँ

cargo new project_name --bin

एक पुस्तकालय टोकरा बनाएँ

cargo new project_name --lib

कार्गो के वर्तमान संस्करण की जांच करने के लिए, निम्नलिखित कमांड निष्पादित करें -

cargo --version

चित्रण - एक बाइनरी कार्गो प्रोजेक्ट बनाएं

गेम एक यादृच्छिक संख्या उत्पन्न करता है और उपयोगकर्ता को संख्या का अनुमान लगाने के लिए प्रेरित करता है।

चरण 1 - एक परियोजना फ़ोल्डर बनाएँ

टर्मिनल खोलें और निम्न कमांड कार्गो नया अनुमान-गेम-ऐप -बिन टाइप करें

यह निम्नलिखित फ़ोल्डर संरचना बनाएगा।

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

कार्गो नया आदेश एक टोकरे बनाने के लिए प्रयोग किया जाता है। --Bin ध्वज इंगित करता है कि टोकरा बनाया जा रहा एक द्विआधारी टोकरा है। सार्वजनिक बक्से को एक केंद्रीय भंडार में संग्रहीत किया जाता है जिसे crates.io कहा जाता हैhttps://crates.io/।

चरण 2 - बाहरी पुस्तकालयों के संदर्भ शामिल करें

इस उदाहरण को एक यादृच्छिक संख्या उत्पन्न करने की आवश्यकता है। चूंकि आंतरिक मानक पुस्तकालय यादृच्छिक संख्या पीढ़ी तर्क प्रदान नहीं करता है, हमें बाहरी पुस्तकालयों या बक्से को देखने की जरूरत है। हमें उपयोग करते हैंrandcrate जो crates.io वेबसाइट crates.io पर उपलब्ध है

https://crates.io/crates/randयादृच्छिक संख्या पीढ़ी के लिए एक जंग पुस्तकालय है। रैंड यादृच्छिक संख्याओं को उत्पन्न करने के लिए उपयोगिताओं को प्रदान करता है, उन्हें उपयोगी प्रकारों और वितरणों में बदलने के लिए, और कुछ यादृच्छिकता-संबंधी एल्गोरिदम।

निम्नलिखित आरेख crate.io वेबसाइट और रैंड क्रेट के लिए खोज परिणाम दिखाता है।

रैंड क्रेट के संस्करण को कॉपी करें। कार्गो फ़ाइल रैंड = "0.5.5" पर

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

[dependencies]
rand = "0.5.5"

चरण 3: परियोजना को संकलित करें

प्रोजेक्ट फ़ोल्डर पर नेविगेट करें। आदेश निष्पादित करेंcargo build टर्मिनल विंडो पर -

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

रैंड क्रेट और सभी सकर्मक निर्भरताएं (रैंड की आंतरिक निर्भरता) स्वचालित रूप से डाउनलोड हो जाएंगी।

चरण 4 - व्यापार तर्क को समझना

आइए अब देखते हैं कि संख्या अनुमान लगाने के खेल के लिए व्यावसायिक तर्क कैसे काम करता है -

  • गेम शुरू में एक यादृच्छिक संख्या उत्पन्न करता है।

  • एक उपयोगकर्ता को इनपुट दर्ज करने और संख्या का अनुमान लगाने के लिए कहा जाता है।

  • यदि संख्या उत्पन्न संख्या से कम है, तो एक संदेश "बहुत कम" मुद्रित होता है।

  • यदि संख्या उत्पन्न संख्या से अधिक है, तो एक संदेश "बहुत अधिक" मुद्रित होता है।

  • यदि उपयोगकर्ता प्रोग्राम द्वारा उत्पन्न संख्या में प्रवेश करता है, तो गेम से बाहर निकलता है।

चरण 5 - main.rs फ़ाइल को संपादित करें

व्यापार तर्क को 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;
      }
   }
}

चरण 6 - परियोजना को संकलित और निष्पादित करें

टर्मिनल पर कमांड कार्गो रन निष्पादित करें । सुनिश्चित करें कि टर्मिनल प्रोजेक्ट डायरेक्टरी को इंगित करता है।

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

इस अध्याय में, हम यह जानेंगे कि RUST में काम करने वाले और बंद करने वाले कैसे काम करते हैं।

iterators

एक पुनरावृत्त, मानों के संग्रह जैसे कि सरणियों, वैक्टरों, नक्शों इत्यादि पर पुनरावृति करने में मदद करता है Iterators रूस्टर मानक पुस्तकालय में परिभाषित Iterator विशेषता को कार्यान्वित करते हैं। आईटीईआर () विधि संग्रह का एक इटरेटर वस्तु देता है। इट्रेटर ऑब्जेक्ट में मानों को आइटम कहा जाता है। आइटमर की अगली () विधि का उपयोग वस्तुओं के माध्यम से पार करने के लिए किया जा सकता है। अगले () विधि एक मूल्य कोई नहीं देता है जब यह संग्रह के अंत तक पहुँचता है।

निम्न उदाहरण सरणी से मान पढ़ने के लिए एक पुनरावृत्ति का उपयोग करता है।

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

उत्पादन

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

यदि सरणी या वेक्टर इलेक्टर इटरेटर गुण जैसे संग्रह करता है, तो इसके लिए इसका उपयोग करके ट्रैवर्स किया जा सकता है ... सिंटैक्स में इसे नीचे दिखाया गया है-

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

उत्पादन

10 20 30

निम्नलिखित 3 विधियां एक संग्रह से एक पुनरावृत्ति वस्तु लौटाती हैं, जहां T एक संग्रह में तत्वों का प्रतिनिधित्व करता है।

अनु क्रमांक तरीके और विवरण
1

iter()

& T पर एक पुनरावृत्ति देता है (T के सन्दर्भ)

2

into_iter()

टी पर एक पुनरावृत्ति देता है

3

iter_mut()

अधिक & म्यूट टी पर एक पुनरावृत्ति देता है

उदाहरण: आईटीईआर ()

Iter () फ़ंक्शन उधार की अवधारणा का उपयोग करता है। यह संग्रह के प्रत्येक तत्व के लिए एक संदर्भ देता है, संग्रह को अछूता छोड़कर लूप के बाद पुन: उपयोग के लिए उपलब्ध है।

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
}

उत्पादन

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

चित्रण - in_iter ()

यह फ़ंक्शन स्वामित्व की अवधारणा का उपयोग करता है। यह संग्रह में मान को एक ऑब्जेक्ट ऑब्जेक्ट में ले जाता है, अर्थात, संग्रह का उपभोग किया जाता है और यह अब पुन: उपयोग के लिए उपलब्ध नहीं है।

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
}

उत्पादन

Hello Kannan
There is a rustacean among us!
Hello Kiran

चित्रण - for and iter_mut ()

यह फ़ंक्शन iter () फ़ंक्शन की तरह है। हालांकि, यह फ़ंक्शन संग्रह के भीतर तत्वों को संशोधित कर सकता है।

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
}

उत्पादन

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

समापन

क्लोजर एक फ़ंक्शन को दूसरे फ़ंक्शन के भीतर संदर्भित करता है। ये अनाम कार्य हैं - एक नाम के बिना कार्य। किसी फंक्शन को वेरिएबल में असाइन करने के लिए क्लोजर का इस्तेमाल किया जा सकता है। यह एक प्रोग्राम को अन्य फ़ंक्शन के पैरामीटर के रूप में फ़ंक्शन पास करने की अनुमति देता है। क्लोजर को एक इनलाइन फ़ंक्शन के रूप में भी जाना जाता है। बाहरी फ़ंक्शन में चर को इनलाइन फ़ंक्शन द्वारा एक्सेस किया जा सकता है।

सिंटैक्स: एक क्लोजिंग को परिभाषित करना

एक बंद परिभाषा में वैकल्पिक रूप से पैरामीटर हो सकते हैं। पैरामीटर दो ऊर्ध्वाधर सलाखों के भीतर संलग्न हैं।

let closure_function = |parameter| {
   //logic
}

वाक्य-विन्यास एक क्लोजर लागू करता है Fnलक्षण। इसलिए, इसे लागू किया जा सकता है() वाक्य - विन्यास।

closure_function(parameter);    //invoking

चित्रण

निम्न उदाहरण फ़ंक्शन मुख्य () के भीतर एक बंद is_even को परिभाषित करता है । यदि संख्या समान है तो क्लोजर सही है और यदि संख्या विषम है तो गलत रिटर्न देता है।

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

उत्पादन

13 is even ? false

चित्रण

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

Main () समारोह में एक चर वाणी वैल और एक बंद। क्लोजर बाहरी फ़ंक्शन मुख्य () में घोषित चर तक पहुंचता है ।

उत्पादन

12

स्टैक डिफ़ॉल्ट रूप से स्टैक पर सब कुछ आवंटित करता है। आप बॉक्स जैसे स्मार्ट पॉइंटर्स में लपेटकर ढेर पर चीजों को स्टोर कर सकते हैं । वीईसी और स्ट्रिंग जैसे प्रकारों से स्पष्ट रूप से ढेर आवंटन में मदद मिलती है। स्मार्ट पॉइंटर्स नीचे दी गई तालिका में सूचीबद्ध लक्षणों को लागू करते हैं। स्मार्ट पॉइंटर्स के ये लक्षण उन्हें एक साधारण संरचना से अलग करते हैं -

अनु क्रमांक लक्षण नाम पैकेज और विवरण
1 Deref

std::ops::Deref

अपरिवर्तनीय डेरेफेरिंग संचालन के लिए उपयोग किया जाता है, जैसे * v।

2 ड्रॉप

std::ops::Drop

कुछ कोड चलाने के लिए उपयोग किया जाता है जब एक मान दायरे से बाहर चला जाता है। इसे कभी-कभी विध्वंसक भी कहा जाता है

इस अध्याय में, हम इसके बारे में जानेंगे Boxस्मार्ट सूचक। हम यह भी सीखेंगे कि बॉक्स जैसे कस्टम स्मार्ट पॉइंटर कैसे बनाएं।

डिब्बा

बॉक्स स्मार्ट पॉइंटर भी कहा जाता है एक बॉक्स आपको ढेर के बजाय ढेर पर डेटा संग्रहीत करने की अनुमति देता है। ढेर में ढेर डेटा के लिए सूचक होता है। एक बॉक्स में प्रदर्शन ओवरहेड नहीं होता है, ढेर पर अपने डेटा को संग्रहीत करने के अलावा।

आइए देखें कि ढेर पर i32 मान संग्रहीत करने के लिए बॉक्स का उपयोग कैसे करें।

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

उत्पादन

b = 5

वैरिएबल द्वारा बताए गए मान को एक्सेस करने के लिए, डेरेफेरिंग का उपयोग करें। * का उपयोग एक डिरेक्शन ऑपरेटर के रूप में किया जाता है। आइए देखते हैं कि कैसे बॉक्स के साथ डीरेंस का उपयोग करें।

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
}

चर x मान 5 के साथ एक मान-प्रकार है। इसलिए, अभिव्यक्ति 5 == x सही लौटेगी। चर y ढेर को इंगित करता है। हीप में मूल्य का उपयोग करने के लिए, हमें * y का उपयोग करके डीरेफेरेंस करने की आवश्यकता है * y रिटर्न मान 5. तो, अभिव्यक्ति 5 == * y सही है।

उत्पादन

true
true

चित्रण - डेरेफ़ ट्रेट

Deref विशेषता, मानक पुस्तकालय द्वारा प्रदान की जाती है, हमें deref नामक एक विधि को लागू करने की आवश्यकता होती है , जो स्वयं उधार लेती है और आंतरिक डेटा का संदर्भ देती है। निम्न उदाहरण एक संरचना MyBox बनाता है , जो एक सामान्य प्रकार है। यह विशेषता डेरेफ को लागू करता है । यह विशेषता हमें * y का उपयोग करके 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
}

उत्पादन

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

चित्रण - ड्रॉप ट्रेट

ड्रॉप विशेषता में ड्रॉप () विधि शामिल है। इस पद्धति को तब कहा जाता है जब इस संरचना को लागू करने वाली संरचना कार्यक्षेत्र से बाहर हो जाती है। कुछ भाषाओं में, प्रोग्रामर को प्रत्येक बार स्मार्ट पॉइंटर के उदाहरण का उपयोग करके समाप्त मेमोरी या संसाधनों को कोड को कॉल करना होगा। जंग में, आप ड्रॉप विशेषता का उपयोग करके स्वचालित मेमोरी डीलक्लेवेशन प्राप्त कर सकते हैं।

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

उपरोक्त उदाहरण में, ड्रॉप विधि को दो बार कहा जाएगा क्योंकि हम ढेर में दो ऑब्जेक्ट बना रहे हैं।

dropping MyBox object from memory
dropping MyBox object from memory

समवर्ती प्रोग्रामिंग में, प्रोग्राम के विभिन्न भाग स्वतंत्र रूप से निष्पादित होते हैं। दूसरी ओर, समानांतर प्रोग्रामिंग में, एक ही समय में एक कार्यक्रम के विभिन्न हिस्सों को निष्पादित किया जाता है। दोनों मॉडल समान रूप से महत्वपूर्ण हैं क्योंकि अधिक कंप्यूटर अपने कई प्रोसेसर का लाभ उठाते हैं।

धागे

हम एक साथ कोड चलाने के लिए थ्रेड्स का उपयोग कर सकते हैं। वर्तमान ऑपरेटिंग सिस्टम में, एक निष्पादित प्रोग्राम का कोड एक प्रक्रिया में चलाया जाता है, और ऑपरेटिंग सिस्टम एक साथ कई प्रक्रियाओं का प्रबंधन करता है। आपके कार्यक्रम के भीतर, आपके पास स्वतंत्र भाग भी हो सकते हैं जो एक साथ चलते हैं। इन स्वतंत्र भागों को चलाने वाली विशेषताओं को थ्रेड्स कहा जाता है।

एक धागा बनाना

thread::spawnफ़ंक्शन का उपयोग एक नया धागा बनाने के लिए किया जाता है। स्पॉन फ़ंक्शन पैरामीटर के रूप में एक क्लोजर लेता है। क्लोजर कोड को परिभाषित करता है जिसे थ्रेड द्वारा निष्पादित किया जाना चाहिए। निम्नलिखित उदाहरण एक मुख्य धागे से कुछ पाठ और एक नए धागे से अन्य पाठ प्रिंट करता है।

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

उत्पादन

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!

मुख्य सूत्र 1 से 4 तक के मानों को प्रिंट करता है।

NOTE- मुख्य धागा समाप्त होने पर नया धागा बंद कर दिया जाएगा। इस प्रोग्राम से आउटपुट हर बार थोड़ा अलग हो सकता है।

thread::sleepफ़ंक्शन थ्रेड को एक छोटी अवधि के लिए इसके निष्पादन को रोकने के लिए मजबूर करता है, जिससे एक अलग थ्रेड चल सकता है। धागे शायद मोड़ लेंगे, लेकिन इसकी गारंटी नहीं है - यह निर्भर करता है कि ऑपरेटिंग सिस्टम थ्रेड्स को कैसे शेड्यूल करता है। इस रन में, मुख्य थ्रेड पहले मुद्रित किया जाता है, भले ही स्पॉन्ड थ्रेड से प्रिंट स्टेटमेंट पहले कोड में दिखाई देता है। इसके अलावा, यहां तक ​​कि अगर स्पॉन्ड थ्रेड को 9 तक मान प्रिंट करने के लिए प्रोग्राम किया जाता है, तो यह केवल मुख्य थ्रेड के बंद होने से पहले 5 तक पहुंच गया।

हैंडल से जुड़ें

एक स्पॉन्डेड थ्रेड को चलाने या पूरी तरह से चलाने का मौका नहीं मिल सकता है। ऐसा इसलिए है क्योंकि मुख्य धागा जल्दी से पूरा होता है। फ़ंक्शन स्पैन <F, T> (f: F) -> JoinHandlelt; T> एक JoinHandle देता है। में शामिल होने के () JoinHandle पर विधि समाप्त करने के लिए संबद्ध थ्रेड के लिए इंतजार कर रहा है।

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

उत्पादन

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!

मुख्य थ्रेड और स्पॉन्डेड थ्रेड स्विच करना जारी रखते हैं।

NOTE - मुख्य धागा कॉल के कारण स्पॉन्ड थ्रेड के पूरा होने की प्रतीक्षा करता है join() तरीका।