अमृत ​​- त्वरित गाइड

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

एलिक्जिर एक कार्यात्मक, गतिशील भाषा है, जो एरलंग और एरलंग वीएम के शीर्ष पर बनी है। Erlang एक भाषा है जो मूल रूप से वितरण, दोष-सहिष्णुता और संगामिति जैसी टेलीफोनी समस्याओं को हल करने के लिए एरिक्सन द्वारा 1986 में लिखी गई थी। जोस वालिम द्वारा लिखित अमृत, एर्लैंग का विस्तार करता है और एरलंग वीएम में एक मित्रवत वाक्यविन्यास प्रदान करता है। यह Erlang के समान स्तर के प्रदर्शन को बनाए रखते हुए ऐसा करता है।

अमृत ​​की विशेषताएं

आइए अब हम अमृत की कुछ महत्वपूर्ण विशेषताओं पर चर्चा करते हैं -

  • Scalability - सभी अमृत कोड हल्की प्रक्रियाओं के अंदर होते हैं जो अलग-थलग होते हैं और संदेशों के माध्यम से सूचनाओं का आदान-प्रदान करते हैं।

  • Fault Tolerance- अमृत पर्यवेक्षक प्रदान करता है जो बताता है कि आपके सिस्टम के कुछ हिस्सों को कैसे फिर से शुरू करें जब चीजें गलत हो जाती हैं, तो एक ज्ञात प्रारंभिक स्थिति में वापस जाना जो काम करने की गारंटी है। यह सुनिश्चित करता है कि आपका एप्लिकेशन / प्लेटफ़ॉर्म कभी डाउन न हो।

  • Functional Programming - फंक्शनल प्रोग्रामिंग एक कोडिंग शैली को बढ़ावा देता है जो डेवलपर्स को कोड लिखने में मदद करता है जो संक्षिप्त, तेज और अनुरक्षणीय है।

  • Build tools- विकास उपकरण के एक सेट के साथ अमृत जहाज। मिक्स एक ऐसा उपकरण है जो परियोजनाओं को बनाना, कार्यों का प्रबंधन करना, परीक्षण चलाना आदि को आसान बनाता है, इसका अपना पैकेज मैनेजर भी है - हेक्स।

  • Erlang Compatibility - एलिक्जिर एरलांग वीएम पर चलता है, जिससे डेवलपर्स को एरलैंग के इकोसिस्टम का पूरा एक्सेस मिलता है।

अमृत ​​को चलाने के लिए, आपको इसे अपने सिस्टम पर स्थानीय रूप से सेट करने की आवश्यकता है।

अमृत ​​को स्थापित करने के लिए, आपको सबसे पहले Erlang की आवश्यकता होगी। कुछ प्लेटफार्मों पर, एलिक्जिर पैकेज उनमें एर्लांग के साथ आते हैं।

अमृत ​​की स्थापना

आइए अब हम विभिन्न ऑपरेटिंग सिस्टम में एलिक्सिर की स्थापना को समझते हैं।

विंडोज सेटअप

खिड़कियों पर अमृत स्थापित करने के लिए, से इंस्टॉलर डाउनलोड करें https://repo.hex.pm/elixirwebsetup.exe और बस क्लिक करें Nextसभी चरणों के माध्यम से आगे बढ़ने के लिए। आपके पास यह आपके स्थानीय सिस्टम पर होगा।

यदि आपको इसे स्थापित करते समय कोई समस्या है, तो आप अधिक जानकारी के लिए इस पृष्ठ की जांच कर सकते हैं ।

मैक सेटअप

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

brew update

अब, नीचे दिए गए कमांड का उपयोग करके अमृत स्थापित करें -

brew install elixir

उबंटू / डेबियन सेटअप

एक Ubuntu / डेबियन सेटअप में अमृत स्थापित करने के लिए कदम इस प्रकार है -

Erlang Solutions रेपो जोड़ें -

wget https://packages.erlang-solutions.com/erlang-solutions_1.0_all.deb && sudo 
dpkg -i erlang-solutions_1.0_all.deb 
sudo apt-get update

Erlang / OTP प्लेटफ़ॉर्म और इसके सभी अनुप्रयोगों को स्थापित करें -

sudo apt-get install esl-erlang

अमृत ​​स्थापित करें -

sudo apt-get install elixir

अन्य लिनक्स डिस्ट्रोस

यदि आपके पास कोई अन्य लिनक्स वितरण है, तो कृपया अपने स्थानीय सिस्टम पर अमृत स्थापित करने के लिए इस पृष्ठ पर जाएं ।

सेटअप का परीक्षण

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

Erlang/OTP 19 [erts-8.0] [source-6dc93c1] [64-bit] 
[smp:4:4] [async-threads:10] [hipe] [kernel-poll:false]  

Interactive Elixir (1.3.1) - press Ctrl+C to exit (type h() ENTER for help) 
iex(1)>

अमृत ​​अब आपके सिस्टम पर सफलतापूर्वक स्थापित हो गया है।

हम प्रथागत 'हैलो वर्ल्ड' कार्यक्रम के साथ शुरू करेंगे।

अमृत ​​इंटरेक्टिव शेल शुरू करने के लिए, निम्न कमांड दर्ज करें।

iex

शेल शुरू होने के बाद, का उपयोग करें IO.putsकंसोल आउटपुट पर स्ट्रिंग को "डाल" करने के लिए फ़ंक्शन। अपने अमृत शेल में निम्नलिखित दर्ज करें -

IO.puts "Hello world"

इस ट्यूटोरियल में, हम Elixir स्क्रिप्ट मोड का उपयोग करेंगे, जहाँ हम विस्तार के साथ फाइल में Elixir कोड रखेंगे .ex। अब हम उपरोक्त कोड को इसमें रखते हैंtest.exफ़ाइल। सफल चरण में, हम इसका उपयोग करके निष्पादित करेंगेelixirc-

IO.puts "Hello world"

आइये अब उपरोक्त कार्यक्रम को निम्न प्रकार से चलाने का प्रयास करते हैं -

$elixirc test.ex

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

Hello World

यहां हम एक फंक्शन कह रहे हैं IO.putsआउटपुट के रूप में हमारे कंसोल को एक स्ट्रिंग उत्पन्न करने के लिए। इस फंक्शन को हम C, C ++, Java इत्यादि में भी कर सकते हैं, जो कि फंक्शन के बाद कोष्ठक में तर्क प्रदान करते हैं -

IO.puts("Hello world")

टिप्पणियाँ

सिंगल लाइन टिप्पणियां '#' प्रतीक से शुरू होती हैं। कोई बहु-पंक्ति टिप्पणी नहीं है, लेकिन आप कई टिप्पणियों को ढेर कर सकते हैं। उदाहरण के लिए -

#This is a comment in Elixir

लाइन अंत

',' जैसी कोई आवश्यक पंक्ति अंत नहीं हैं; अमृत ​​में। हालाँकि, हम ',' का उपयोग करते हुए एक ही पंक्ति में कई कथन रख सकते हैं। उदाहरण के लिए,

IO.puts("Hello"); IO.puts("World!")

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

Hello 
World!

पहचानकर्ता

चर, फ़ंक्शन नाम आदि जैसे पहचानकर्ताओं का उपयोग किसी चर, फ़ंक्शन आदि की पहचान करने के लिए किया जाता है। अमृत में, आप अपने पहचानकर्ताओं को संख्या के साथ कम केस वर्णमाला से शुरू कर सकते हैं, उसके बाद अपर केस अक्षर। इस नामकरण सम्मेलन को आमतौर पर स्नेककेस के रूप में जाना जाता है। उदाहरण के लिए, अमृत में कुछ मान्य पहचानकर्ता निम्नलिखित हैं -

var1       variable_2      one_M0r3_variable

कृपया ध्यान दें कि चर को एक प्रमुख अंडरस्कोर के साथ भी नाम दिया जा सकता है। ऐसा मान जो उपयोग करने के लिए नहीं है, उसे _ या अंडरस्कोर से शुरू होने वाले चर को सौंपा जाना चाहिए -

_some_random_value = 42

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

अमृत ​​में फ़ंक्शन नामकरण से संबंधित कई और पेचीदगियां हैं, जिनके बारे में हम आने वाले अध्यायों में चर्चा करेंगे।

सुरक्षित शब्द

निम्नलिखित शब्द आरक्षित हैं और इन्हें चर, मॉड्यूल या फ़ंक्शन नामों के रूप में उपयोग नहीं किया जा सकता है।

after     and     catch     do     inbits     inlist     nil     else     end 
not     or     false     fn     in     rescue     true     when     xor 
__MODULE__    __FILE__    __DIR__    __ENV__    __CALLER__

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

संख्यात्मक प्रकार

किसी भी अन्य प्रोग्रामिंग भाषा की तरह, अमृत पूर्णांक और फ्लोट दोनों का समर्थन करता है। यदि आप अपना अमृत खोलते हैं और किसी भी पूर्णांक को इनपुट करते हैं या इनपुट के रूप में फ्लोट करते हैं, तो यह उसका मान लौटा देगा। उदाहरण के लिए,

42

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

42

आप ऑक्टल, हेक्स और बाइनरी बेस में संख्याओं को भी परिभाषित कर सकते हैं।

अष्टभुजाकार

अष्टक आधार में एक संख्या को परिभाषित करने के लिए, इसे '0o' के साथ उपसर्ग करें। उदाहरण के लिए, ऑक्टल में 0o52 दशमलव में 42 के बराबर है।

हेक्साडेसिमल

दशमलव आधार में एक संख्या को परिभाषित करने के लिए, इसे '0x' के साथ उपसर्ग करें। उदाहरण के लिए, हेक्स में 0xF1 दशमलव में 241 के बराबर है।

बायनरी

बाइनरी बेस में एक संख्या को परिभाषित करने के लिए, इसे '0 बी' के साथ उपसर्ग करें। उदाहरण के लिए, द्विआधारी में 0b1101 दशमलव में 13 के बराबर है।

एलिक्सिर फ्लोटिंग पॉइंट नंबरों के लिए 64 बिट डबल परिशुद्धता का समर्थन करता है। और उन्हें एक घातांक शैली का उपयोग करके भी परिभाषित किया जा सकता है। उदाहरण के लिए, 10145230000 को 1.014523e10 लिखा जा सकता है

परमाणुओं

परमाणु निरंतर हैं जिनके नाम का मूल्य है। उन्हें रंग (:) प्रतीक का उपयोग करके बनाया जा सकता है। उदाहरण के लिए,

:hello

बूलियन्स

अमृत ​​समर्थन करता है true तथा falseबूलियन के रूप में। ये दोनों मूल्य वास्तव में परमाणुओं से जुड़े हैं: क्रमशः सत्य और: असत्य।

स्ट्रिंग्स

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

"Hello world"

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

"""
Hello
World!
"""

हम स्ट्रिंग अध्याय में गहराई से तार, बायनेरिज़ और चार सूचियों (तार के समान) के बारे में सीखेंगे।

बाइनरी

बायनेरिज़ << में संलग्न बाइट्स के अनुक्रम हैं >> एक अल्पविराम के साथ अलग हो गए। उदाहरण के लिए,

<< 65, 68, 75>>

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

<<65, 255, 289::size(15)>>

सूचियों

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

[1, "Hello", :an_atom, true]

सूचियाँ एचडी और टीएल नाम की सूची के सिर और पूंछ के लिए इनबिल्ट फ़ंक्शंस के साथ आती हैं जो क्रमशः सूची के सिर और पूंछ को वापस करती हैं। कभी-कभी जब आप एक सूची बनाते हैं, तो यह एक चार सूची लौटाएगा। ऐसा इसलिए है क्योंकि जब अमृत मुद्रण योग्य ASCII वर्णों की सूची देखता है, तो वह इसे एक चार्ट सूची के रूप में छापता है। कृपया ध्यान दें कि तार और चार सूचियाँ समान नहीं हैं। हम बाद के अध्यायों में सूचियों पर चर्चा करेंगे।

tuples

अमृत ​​टुपल्स को परिभाषित करने के लिए घुंघराले कोष्ठक का उपयोग करता है। सूचियों की तरह, टुपल्स किसी भी मूल्य को पकड़ सकते हैं।

{ 1, "Hello", :an_atom, true

यहां एक सवाल उठता है, - क्यों दोनों प्रदान करते हैं lists तथा tuplesजब वे दोनों एक ही तरीके से काम करते हैं? वैसे उनके अलग-अलग कार्यान्वयन हैं।

  • सूचियों को वास्तव में लिंक की गई सूचियों के रूप में संग्रहीत किया जाता है, इसलिए प्रविष्टि में विलोपन, विलोपन बहुत तेजी से होते हैं।

  • दूसरी ओर टुपल्स, सन्निहित मेमोरी ब्लॉक में संग्रहीत होते हैं, जो उन्हें तेजी से एक्सेस करते हैं, लेकिन सम्मिलन और विलोपन पर एक अतिरिक्त लागत जोड़ते हैं।

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

चर के प्रकार

अमृत ​​निम्नलिखित मूल प्रकार के चर का समर्थन करता है।

पूर्णांक

इनका उपयोग इंटेगर के लिए किया जाता है। वे 32 बिट आर्किटेक्चर पर 32 बिट और 64-बिट आर्किटेक्चर पर 64 बिट्स के आकार के हैं। इंटेगर हमेशा अमृत में हस्ताक्षरित होते हैं। यदि एक पूर्णांक अपनी सीमा से ऊपर आकार में विस्तार करने के लिए शुरू होता है, तो अमृत इसे एक बिग इंटेगर में वार्तालाप करता है जो 3 से n शब्दों में स्मृति में ले जाता है जो भी इसे स्मृति में फिट कर सकता है।

तैरता

फ़्लोट्स में अमृत में 64-बिट परिशुद्धता है। वे मेमोरी के मामले में पूर्णांक की तरह भी हैं। फ्लोट को परिभाषित करते समय, घातीय संकेतन का उपयोग किया जा सकता है।

बूलियन

वे 2 मान ले सकते हैं जो या तो सही है या गलत है।

स्ट्रिंग्स

तार utf-8 अमृत में कूटबद्ध हैं। उनके पास एक स्ट्रिंग्स मॉड्यूल है जो स्ट्रिंगर को हेरफेर करने के लिए प्रोग्रामर को बहुत अधिक कार्यक्षमता प्रदान करता है।

बेनामी फ़ंक्शंस / लम्बदा

ये ऐसे कार्य हैं जिन्हें परिभाषित किया जा सकता है और एक चर को सौंपा जा सकता है, जिसे तब इस फ़ंक्शन को कॉल करने के लिए उपयोग किया जा सकता है।

संग्रह

अमृत ​​में बहुत सारे संग्रह प्रकार उपलब्ध हैं। उनमें से कुछ सूचियाँ, टुपल्स, मैप्स, बायनेरिज़ इत्यादि हैं, इन पर बाद के अध्यायों में चर्चा की जाएगी।

परिवर्तनीय घोषणा

एक चर घोषणा दुभाषिया को बताती है कि चर के लिए भंडारण कहां और कितना बनाना है। अमृत ​​हमें केवल एक चर घोषित करने की अनुमति नहीं देता है। एक चर को उसी समय एक मूल्य घोषित और नियत किया जाना चाहिए। उदाहरण के लिए, जीवन नाम का एक वैरिएबल बनाने के लिए और इसे 42 मान देने के लिए, हम निम्नलिखित करते हैं -

life = 42

यह परिवर्तनशील जीवन को 42 के मान से बाँध देगा । यदि हम इस चर को एक नया मान देना चाहते हैं, तो हम ऊपर के समान सिंटैक्स का उपयोग करके ऐसा कर सकते हैं, अर्थात

life = "Hello world"

चर नामकरण

नामकरण चर एक का पालन करें snake_caseएलिक्जिर में कन्वेंशन, यानी, सभी वेरिएबल्स को लोअरकेस लेटर से शुरू करना चाहिए, उसके बाद 0 या उससे ज्यादा लेटर्स (अपर और लोअर दोनों तरह के केस) के बाद, ऑप्शनल के बाद '?' या '!'।

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

मुद्रण योग्य चर

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

life = 42

और अपने शेल में 'लाइफ' डालें, आपको आउटपुट मिलेगा -

42

लेकिन अगर आप एक चर को कंसोल में आउटपुट करना चाहते हैं (जब एक फ़ाइल से बाहरी स्क्रिप्ट चल रहा है), तो आपको चर को इनपुट की तरह प्रदान करना होगा IO.puts कार्य -

life = 42  
IO.puts life

या

life = 42 
IO.puts(life)

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

42

एक ऑपरेटर एक प्रतीक है जो संकलक को विशिष्ट गणितीय या तार्किक जोड़तोड़ करने के लिए कहता है। अमृत ​​द्वारा प्रदान किए गए ऑपरेटरों के बहुत सारे हैं। वे निम्नलिखित श्रेणियों में विभाजित हैं -

  • अंकगणितीय आपरेटर
  • तुलना संचालक
  • बूलियन ऑपरेटर्स
  • विविध संचालक

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

निम्न तालिका में अमृत भाषा द्वारा समर्थित सभी अंकगणितीय ऑपरेटरों को दिखाया गया है। चर मान लेंA 10 और चर रखता है B 20 रखती है, तो -

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

ऑपरेटर विवरण उदाहरण
+ 2 नंबर जोड़ता है। A + B 30 देगा
- पहले से दूसरे नंबर को घटाता है। एबी देगा -10
* गुणा दो संख्या। A * B 200 देगा
/ दूसरे से विभाजित पहले नंबर। यह फ्लोट में संख्याओं को जमा करता है और फ्लोट का परिणाम देता है A / B 0.5 देगा।
डिव इस फ़ंक्शन का उपयोग विभाजन पर भागफल प्राप्त करने के लिए किया जाता है। div (10,20) 0 देगा
रेम इस फ़ंक्शन का उपयोग विभाजन पर शेष प्राप्त करने के लिए किया जाता है। rem (A, B) 10 देगा

तुलना संचालक

Elixir में तुलना करने वाले ऑपरेटर ज्यादातर अन्य भाषाओं में उपलब्ध कराए जाने वाले आम हैं। निम्न तालिका में अमृत संचालक की तुलना की गई है। चर मान लेंA 10 और चर रखता है B 20 रखती है, तो -

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

ऑपरेटर विवरण उदाहरण
== जाँचता है कि क्या बाईं ओर का मान दाईं ओर के बराबर है (यदि वे टाइप नहीं हैं, तो मानों को टाइप करते हैं)। ए == बी झूठी देगा
! = जाँचता है कि बाईं ओर मान दाईं ओर मान के बराबर नहीं है। A! = B सत्य देगा
=== यह जाँचता है कि क्या बाईं ओर का मान दाईं ओर के प्रकार के बराबर है, यदि हाँ तो मूल्य के लिए समान जाँच करें। A === B असत्य देगा
! == ऊपर के समान लेकिन समानता के बजाय असमानता के लिए जाँच करता है। A! == B सत्य देगा
> जाँच करता है कि क्या बाएं ऑपरेंड का मूल्य सही ऑपरेंड के मूल्य से अधिक है; यदि हाँ, तो स्थिति सच हो जाती है। A> B गलत देगा
< चेक करता है कि क्या बाएं ऑपरेंड का मूल्य सही ऑपरेंड के मूल्य से कम है; यदि हाँ, तो स्थिति सच हो जाती है। A <B सत्य देगा
> = जाँच करता है कि क्या बाएं ऑपरेंड का मूल्य सही ऑपरेंड के मूल्य से अधिक या बराबर है; यदि हाँ, तो स्थिति सच हो जाती है। A> = B गलत देगा
<= यह जाँचता है कि क्या बाएं ऑपरेंड का मूल्य सही ऑपरेंड के मूल्य से कम या उसके बराबर है; यदि हाँ, तो स्थिति सच हो जाती है। A <= B सत्य देगा

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

अमृत ​​6 तार्किक संचालक प्रदान करता है: और, या, नहीं, &&, || तथा !। पहले तीन,and or notसख्त बूलियन ऑपरेटर हैं, जिसका अर्थ है कि वे बूलियन होने के अपने पहले तर्क की उम्मीद करते हैं। गैर बुलियन तर्क एक त्रुटि बढ़ाएगा। जबकि अगले तीन,&&, || and !गैर सख्त हैं, हमें बूलियन के रूप में पहला मूल्य सख्ती से लेने की आवश्यकता नहीं है। वे अपने सख्त समकक्षों की तरह ही काम करते हैं। चर मान लेंA सत्य और परिवर्तनशील है B 20 रखती है, तो -

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

ऑपरेटर विवरण उदाहरण
तथा जाँचता है कि क्या प्रदान किए गए दोनों मान सत्य हैं, यदि हाँ, तो दूसरा चर का मान लौटाता है। (तार्किक और)। A और B 20 देंगे
या जाँचता है कि क्या मूल्य प्रदान किया गया है या नहीं। जो भी मूल्य लौटाता है वह सत्य है। झूठे झूठे होते हैं। (तार्किक या)। A या B सत्य देगा
नहीं यूनरी ऑपरेटर जो दिए गए इनपुट के मूल्य को बदल देता है। नहीं A गलत देगा
&& गैर सख्त and। के रूप में ही काम करता हैand लेकिन पहले तर्क को बूलियन होने की उम्मीद नहीं है। B && A 20 देगा
|| गैर सख्त or। के रूप में ही काम करता हैor लेकिन पहले तर्क को बूलियन होने की उम्मीद नहीं है। B || A सत्य देगा
! गैर सख्त not। के रूप में ही काम करता हैnot लेकिन बूलियन होने के तर्क की उम्मीद नहीं करता है। ! झूठा देगा

NOTE −और , या , && और || || शॉर्ट सर्किट ऑपरेटर हैं। इसका मतलब है कि अगर पहली दलीलandगलत है, तो यह दूसरे के लिए आगे की जांच नहीं करेगा। और अगर का पहला तर्कorसच है, तो यह दूसरे के लिए जाँच नहीं करेगा। उदाहरण के लिए,

false and raise("An error")  
#This won't raise an error as raise function wont get executed because of short
#circuiting nature of and operator

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

बिटवाइज़ ऑपरेटर बिट पर काम करते हैं और बिट ऑपरेशन द्वारा बिट करते हैं। अमृत ​​पैकेज के हिस्से के रूप में बिटवाइज़ मॉड्यूल प्रदान करता हैBitwise, इसलिए इनका उपयोग करने के लिए, आपको बिटवाइज़ मॉड्यूल का उपयोग करने की आवश्यकता है । इसका उपयोग करने के लिए, अपने शेल में निम्नलिखित कमांड दर्ज करें -

use Bitwise

निम्नलिखित उदाहरणों के लिए A को 5 और B को 6 मान लीजिए -

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

ऑपरेटर विवरण उदाहरण
&&& यदि यह दोनों ऑपरेंड में मौजूद है, तो बिटवाइज़ और ऑपरेटर परिणाम की प्रतिलिपि बनाता है। A &&& B 4 देगा
||| बिटवाइज या ऑपरेटर परिणाम के लिए थोड़ा सा कॉपी करता है यदि यह किसी भी ऑपरेंड में मौजूद है। A ||| B 7 देगा
>>> बिटकॉइन राइट शिफ्ट ऑपरेटर दूसरे ऑपरेंड में निर्दिष्ट संख्या द्वारा दाईं ओर पहले ऑपरेंड बिट्स को स्थानांतरित करता है। A >>> B 0 देगा
<<< बिटकॉइन लेफ्ट शिफ्ट ऑपरेटर दूसरे ऑपरेंड में निर्दिष्ट संख्या द्वारा पहले ऑपरेंड बिट्स को बाईं ओर शिफ्ट करता है। A <<< B 320 देगा
^^^ बिटवाइज XOR ऑपरेटर केवल परिणाम के लिए थोड़ा सा कॉपी करता है यदि यह दोनों ऑपरेंड पर अलग हो। A ^ ^ ^ B 3 देगा
~~~ Unary bitwise दिए गए नंबर पर बिट्स को इन्वर्ट नहीं करता है। ~~~ A देगा -6

विविध संचालक

उपरोक्त ऑपरेटरों के अलावा, अमृत भी अन्य ऑपरेटरों की तरह प्रदान करता है Concatenation Operator, Match Operator, Pin Operator, Pipe Operator, String Match Operator, Code Point Operator, Capture Operator, Ternary Operator जो इसे काफी शक्तिशाली भाषा बनाते हैं।

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

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

एक मैच के 2 मुख्य भाग होते हैं, a left और एक rightपक्ष। दाईं ओर किसी भी प्रकार की डेटा संरचना है। बाईं ओर दाईं ओर डेटा संरचना से मेल खाने का प्रयास करता है और बाईं ओर किसी भी चर को दाईं ओर संबंधित उप-संरचना में बांधता है। यदि कोई मेल नहीं मिलता है, तो ऑपरेटर एक त्रुटि उठाता है।

सबसे सरल मैच बाईं ओर एक अकेला चर है और दाईं ओर किसी भी डेटा संरचना है। This variable will match anything। उदाहरण के लिए,

x = 12
x = "Hello"
IO.puts(x)

आप चर को एक संरचना के अंदर रख सकते हैं ताकि आप एक उप-निर्माण पर कब्जा कर सकें। उदाहरण के लिए,

[var_1, _unused_var, var_2] = [{"First variable"}, 25, "Second variable" ]
IO.puts(var_1)
IO.puts(var_2)

यह मानों को संचित करेगा, {"First variable"}में VAR_1 और"Second variable"में VAR_2 । एक खास भी है_ चर (या चर '' _ 'के साथ उपसर्ग करता है) जो अन्य चर की तरह ही काम करता है लेकिन अमृत बताता है, "Make sure something is here, but I don't care exactly what it is."। पिछले उदाहरण में, _unused_var एक ऐसा चर था।

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

[_, [_, {a}]] = ["Random string", [:an_atom, {24}]]
IO.puts(a)

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

24

यह बंध जाएगा a 24. अन्य मानों को अनदेखा किया जाता है क्योंकि हम '_' का उपयोग कर रहे हैं।

यदि हम एक चर का उपयोग करते हैं, तो पैटर्न मिलान में right, इसके मूल्य का उपयोग किया जाता है। यदि आप बाईं ओर एक चर के मूल्य का उपयोग करना चाहते हैं, तो आपको पिन ऑपरेटर का उपयोग करना होगा।

उदाहरण के लिए, यदि आपके पास एक वेरिएबल "a" है जिसका मूल्य 25 है और आप इसे दूसरे वेरिएबल "b" वाले मान 25 के साथ मैच करना चाहते हैं, तो आपको प्रवेश करने की आवश्यकता है -

a = 25
b = 25
^a = b

अंतिम पंक्ति वर्तमान मूल्य से मेल खाती है aके बजाय, इसे निर्दिष्ट करने के लिए, के मूल्य पर b। यदि हमारे पास बाएं और दाएं हाथ का गैर-मिलान सेट है, तो मैच ऑपरेटर एक त्रुटि उठाता है। उदाहरण के लिए, यदि हम किसी सूची या आकार 2 की सूची के साथ आकार 3 की सूची के साथ टपल का मिलान करने का प्रयास करते हैं, तो त्रुटि दिखाई जाएगी।

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

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

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

अमृत ​​निर्णय लेने के निम्नलिखित प्रकार प्रदान करता है। उनके विवरण की जाँच करने के लिए निम्न लिंक पर क्लिक करें।

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

यदि एक कथन बूलियन अभिव्यक्ति के बाद होता है do, एक या अधिक निष्पादन योग्य बयान और अंत में ए endकीवर्ड। यदि बूलियन स्थिति सही का मूल्यांकन करती है, तो केवल कथन निष्पादित करता है।

2 if..else स्टेटमेंट

यदि कोई कथन वैकल्पिक रूप से कथन (do..end ब्लॉक के भीतर) का अनुसरण किया जा सकता है, जो बूलियन अभिव्यक्ति के गलत होने पर निष्पादित होता है।

3 जब तक बयान न हो

जब तक कि स्टेटमेंट में इफ स्टेटमेंट के समान बॉडी न हो। जब तक कथन के भीतर कोड केवल तब ही निष्पादित होता है जब निर्दिष्ट शर्त झूठी है।

4 जब तक..बल्कि बयान नहीं

जब तक..बल्कि कथन का शरीर एक if..else कथन के समान नहीं होता। जब तक कथन के भीतर कोड केवल तब ही निष्पादित होता है जब निर्दिष्ट शर्त झूठी है।

5 cond

एक कंडोम स्टेटमेंट का उपयोग किया जाता है जहां हम कई शर्तों के आधार पर कोड निष्पादित करना चाहते हैं। यह एक तरह से काम करता है जैसे कि ... और अगर ... तो कई अन्य प्रोग्रामिंग भाषाओं में निर्माण।

6 मामला

केस स्टेटमेंट को अनिवार्य भाषाओं में स्विच स्टेटमेंट के प्रतिस्थापन के रूप में माना जा सकता है। मामला एक चर / शाब्दिक लेता है और विभिन्न मामलों के साथ इसके साथ मेल खाता पैटर्न लागू करता है। यदि कोई मामला मेल खाता है, तो अमृत उस केस से जुड़े कोड को निष्पादित करता है और केस स्टेटमेंट से बाहर निकलता है।

एलिक्जिर में स्ट्रिंग्स को दोहरे उद्धरण चिह्नों के बीच डाला जाता है, और उन्हें UTF-8 में एन्कोड किया जाता है। C और C ++ के विपरीत, जहां डिफ़ॉल्ट स्ट्रिंग्स ASCII एन्कोडेड हैं और केवल 256 विभिन्न वर्ण संभव हैं, UTF-8 में 1,112,064 कोड बिंदु हैं। इसका मतलब है कि UTF-8 एन्कोडिंग में कई अलग-अलग संभावित अक्षर होते हैं। चूंकि तार utf-8 का उपयोग करते हैं, हम प्रतीकों का भी उपयोग कर सकते हैं जैसे: ö, ł, आदि।

एक स्ट्रिंग बनाएँ

एक स्ट्रिंग चर बनाने के लिए, बस एक चर को स्ट्रिंग असाइन करें -

str = "Hello world"

इसे अपने कंसोल पर प्रिंट करने के लिए, बस कॉल करें IO.puts फ़ंक्शन और इसे चर str पास करें -

str = str = "Hello world" 
IO.puts(str)

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

Hello World

खाली स्ट्रिंग्स

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

a = ""
if String.length(a) === 0 do
   IO.puts("a is an empty string")
end

उपरोक्त कार्यक्रम निम्नलिखित परिणाम उत्पन्न करता है।

a is an empty string

स्ट्रिंग इंटरपोल

स्ट्रिंग इंटरपोलेशन एक स्ट्रिंग स्ट्रिंग शाब्दिक के अंदर उनके मूल्यों को शामिल करके स्थिरांक, चर, शाब्दिक और भावों के मिश्रण से एक नया स्ट्रिंग मूल्य बनाने का एक तरीका है। अमृत ​​स्ट्रिंग में एक चर का उपयोग करने के लिए स्ट्रिंग प्रक्षेप का समर्थन करता है, इसे लिखते समय, इसे घुंघराले ब्रेस के साथ लपेटें और घुंघराले ब्रेस के साथ प्रीपेन्ड करें'#' संकेत।

उदाहरण के लिए,

x = "Apocalypse" 
y = "X-men #{x}"
IO.puts(y)

यह x का मान लेगा और इसे y में स्थानापन्न करेगा। उपरोक्त कोड निम्नलिखित परिणाम उत्पन्न करेगा -

X-men Apocalypse

स्ट्रिंग कॉनटेनटेशन

हम पिछले अध्यायों में स्ट्रिंग संघनन का उपयोग पहले ही देख चुके हैं। '<>' ऑपरेटर का उपयोग अमृत में तार को समतल करने के लिए किया जाता है। 2 तारों को समतल करने के लिए,

x = "Dark"
y = "Knight"
z = x <> " " <> y
IO.puts(z)

उपरोक्त कोड निम्नलिखित परिणाम उत्पन्न करता है -

Dark Knight

स्ट्रिंग लंबाई

स्ट्रिंग की लंबाई पाने के लिए, हम इसका उपयोग करते हैं String.lengthसमारोह। स्ट्रिंग को एक पैरामीटर के रूप में पास करें और यह आपको अपना आकार दिखाएगा। उदाहरण के लिए,

IO.puts(String.length("Hello"))

कार्यक्रम से ऊपर चलने पर, यह निम्नलिखित परिणाम उत्पन्न करता है -

5

एक स्ट्रिंग उलट

किसी स्ट्रिंग को रिवर्स करने के लिए, इसे String.reverse फ़ंक्शन में पास करें। उदाहरण के लिए,

IO.puts(String.reverse("Elixir"))

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

rixilE

स्ट्रिंग तुलना

2 तारों की तुलना करने के लिए, हम == या === ऑपरेटरों का उपयोग कर सकते हैं। उदाहरण के लिए,

var_1 = "Hello world"
var_2 = "Hello Elixir"
if var_1 === var_2 do
   IO.puts("#{var_1} and #{var_2} are the same")
else
   IO.puts("#{var_1} and #{var_2} are not the same")
end

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

Hello world and Hello elixir are not the same.

स्ट्रिंग मिलान

हमने पहले ही = ~ स्ट्रिंग मैच ऑपरेटर का उपयोग देखा है। यह जांचने के लिए कि क्या एक स्ट्रिंग एक रेग्ज से मेल खाता है, हम स्ट्रिंग मिलान ऑपरेटर या स्ट्रिंगमचैच का भी उपयोग कर सकते हैं? समारोह। उदाहरण के लिए,

IO.puts(String.match?("foo", ~r/foo/))
IO.puts(String.match?("bar", ~r/foo/))

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

true 
false

यह भी = ~ ऑपरेटर का उपयोग करके प्राप्त किया जा सकता है। उदाहरण के लिए,

IO.puts("foo" =~ ~r/foo/)

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

true

स्ट्रिंग फ़ंक्शंस

अमृत ​​तार से संबंधित कार्यों की एक बड़ी संख्या का समर्थन करता है, सबसे अधिक उपयोग किए जाने वाले कुछ निम्न तालिका में सूचीबद्ध हैं।

अनु क्रमांक। समारोह और इसके उद्देश्य
1

at(string, position)

दिए गए utf8 स्ट्रिंग की स्थिति में ग्रेपैम लौटाता है। यदि स्थिति स्ट्रिंग की लंबाई से अधिक है, तो यह शून्य देता है

2

capitalize(string)

दिए गए स्ट्रिंग में पहला वर्ण अपरकेस और शेष लोअरकेस में देता है

3

contains?(string, contents)

जाँच करता है कि स्ट्रिंग में दी गई कोई सामग्री है या नहीं

4

downcase(string)

दिए गए स्ट्रिंग में सभी वर्णों को लोअरकेस में कनवर्ट करता है

5

ends_with?(string, suffixes)

यदि स्ट्रिंग दी गई किसी भी प्रत्यय के साथ समाप्त हो जाती है, तो सही है

6

first(string)

Utf8 स्ट्रिंग से पहला ग्रेपैम लौटाता है, अगर स्ट्रिंग खाली है तो nil करें

7

last(string)

Utf8 स्ट्रिंग से अंतिम ग्रेपैम लौटाता है, यदि स्ट्रिंग खाली है तो शून्य करें

8

replace(subject, pattern, replacement, options \\ [])

प्रतिस्थापन के साथ विषय में पैटर्न की घटनाओं को प्रतिस्थापित करके बनाया गया एक नया स्ट्रिंग देता है

9

slice(string, start, len)

ऑफसेट प्रारंभ में और लंबाई लेन पर एक विकल्प शुरू करता है

10

split(string)

प्रमुख और अनुगामी व्हाट्सएप की अनदेखी के साथ प्रत्येक यूनिकोड व्हाट्सएप घटना पर सबस्ट्रिंग में एक स्ट्रिंग को विभाजित करता है। व्हाट्सएप के समूहों को एक ही घटना के रूप में माना जाता है। विभाजन न टूटने वाले व्हाट्सएप पर नहीं होता है

1 1

upcase(string)

दिए गए स्ट्रिंग के सभी वर्णों को अपरकेस में परिवर्तित करता है

बाइनरी

एक बाइनरी सिर्फ बाइट्स का एक क्रम है। बायनेरिज़ का उपयोग करके परिभाषित किया गया है<< >>। उदाहरण के लिए:

<< 0, 1, 2, 3 >>

बेशक, उन बाइट्स को किसी भी तरह से व्यवस्थित किया जा सकता है, यहां तक ​​कि एक क्रम में भी जो उन्हें वैध स्ट्रिंग नहीं बनाता है। उदाहरण के लिए,

<< 239, 191, 191 >>

स्ट्रिंग्स भी बायनेरिज़ हैं। और स्ट्रिंग संघनन ऑपरेटर<> वास्तव में एक द्विआधारी संघनन ऑपरेटर है:

IO.puts(<< 0, 1 >> <> << 2, 3 >>)

उपरोक्त कोड निम्नलिखित परिणाम उत्पन्न करता है -

<< 0, 1, 2, 3 >>

Ł वर्ण नोट करें। चूंकि यह utf-8 एनकोडेड है, इसलिए यह कैरेक्टर रिप्रेजेंटेशन 2 बाइट्स लेता है।

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

IO.puts(<< 256 >>) # truncated, it'll print << 0 >>
IO.puts(<< 256 :: size(16) >>) #Takes 16 bits/2 bytes, will print << 1, 0 >>

उपरोक्त कार्यक्रम निम्नलिखित परिणाम उत्पन्न करेगा -

<< 0 >>
<< 1, 0 >>

हम utf8 संशोधक का भी उपयोग कर सकते हैं, यदि कोई वर्ण कोड बिंदु है, तो इसे आउटपुट में उत्पादित किया जाएगा; और बाइट्स -

IO.puts(<< 256 :: utf8 >>)

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

Ā

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

Bitstrings

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

bs = << 1 :: size(1) >>
IO.puts(bs)
IO.puts(is_binary(bs))
IO.puts(is_bitstring(bs))

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

<< 1::size(1) >>
false
true

इसका मतलब है कि चर bsएक बाइनरी नहीं है, बल्कि एक बिटस्ट्रिंग है। हम यह भी कह सकते हैं कि एक द्विआधारी एक बिटस्ट्रिंग है जहां बिट्स की संख्या 8 से विभाज्य है। पैटर्न मिलान मिलान बायनेरिज़ के साथ-साथ बिटस्ट्रिंग्स पर भी उसी तरह काम करता है।

एक वर्ण सूची वर्णों की सूची से अधिक कुछ नहीं है। उसी को समझने के लिए निम्नलिखित कार्यक्रम पर विचार करें।

IO.puts('Hello')
IO.puts(is_list('Hello'))

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

Hello
true

बाइट्स रखने के बजाय, एक चार्ट सूची में एकल-उद्धरणों के बीच वर्णों के कोड बिंदु शामिल हैं। So while the double-quotes represent a string (i.e. a binary), singlequotes represent a char list (i.e. a list)। ध्यान दें कि IEx ASCII सीमा के बाहर होने पर IEx केवल कोड पॉइंट आउटपुट के रूप में उत्पन्न करेगा।

चार सूचियों का उपयोग ज्यादातर एरलंग के साथ हस्तक्षेप करते समय किया जाता है, विशेष रूप से पुराने पुस्तकालयों में जो बायनेरिज़ को तर्क के रूप में स्वीकार नहीं करते हैं। आप to_string (char_list) और to_char_list (string) फ़ंक्शंस का उपयोग करके एक चर सूची को स्ट्रिंग और बैक में बदल सकते हैं -

IO.puts(is_list(to_char_list("hełło")))
IO.puts(is_binary(to_string ('hełło')))

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

true
true

NOTE - कार्य to_string तथा to_char_list पॉलीमोर्फिक हैं, यानी, वे कई प्रकार के इनपुट जैसे परमाणु, पूर्णांक ले सकते हैं और उन्हें क्रमशः स्ट्रिंग्स और चार सूचियों में बदल सकते हैं।

(लिंक्ड) सूची

एक लिंक की गई सूची उन तत्वों की एक विषम सूची है जो स्मृति में विभिन्न स्थानों पर संग्रहीत हैं और संदर्भों का उपयोग करके ट्रैक की जाती हैं। लिंक्ड सूची विशेष रूप से कार्यात्मक प्रोग्रामिंग में उपयोग की जाने वाली डेटा संरचनाएं हैं।

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

[1, 2, true, 3]

जब अमृत प्रिंट करने योग्य ASCII नंबरों की एक सूची देखता है, तो अमृत एक चार सूची (अक्षर की सूची) के रूप में मुद्रित करेगा। जब भी आपको IEx में कोई मान दिखाई देता है और आप निश्चित नहीं हैं कि यह क्या है, तो आप इसका उपयोग कर सकते हैंi इसके बारे में जानकारी प्राप्त करने के लिए कार्य करते हैं।

IO.puts([104, 101, 108, 108, 111])

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

hello

आप एकल उद्धरण का उपयोग करते हुए सूचियों को दूसरे तरीके से भी परिभाषित कर सकते हैं -

IO.puts(is_list('Hello'))

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

true

ध्यान रखें कि सिंगल-कोटेड और डबल-कोटेड अभ्यावेदन अमृत में समान नहीं हैं क्योंकि वे विभिन्न प्रकारों द्वारा दर्शाए जाते हैं।

एक सूची की लंबाई

किसी सूची की लंबाई जानने के लिए, हम निम्न कार्यक्रम की तरह लंबाई फ़ंक्शन का उपयोग करते हैं -

IO.puts(length([1, 2, :true, "str"]))

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

4

समास और घटाव

दो सूचियों का उपयोग करके संक्षिप्त और घटाया जा सकता है ++ तथा --ऑपरेटरों। कार्यों को समझने के लिए निम्नलिखित उदाहरण पर विचार करें।

IO.puts([1, 2, 3] ++ [4, 5, 6])
IO.puts([1, true, 2, false, 3, true] -- [true, false])

यह आपको पहले मामले में एक संक्षिप्त स्ट्रिंग और दूसरे में एक घटा हुआ स्ट्रिंग देगा। उपरोक्त कार्यक्रम निम्नलिखित परिणाम उत्पन्न करता है -

[1, 2, 3, 4, 5, 6]
[1, 2, 3, true]

एक सूची के प्रमुख और पूंछ

सिर एक सूची का पहला तत्व है और पूंछ एक सूची का शेष है। उन्हें कार्यों के साथ पुनः प्राप्त किया जा सकता हैhd तथा tl। आइए हम एक चर को एक सूची असाइन करें और उसके सिर और पूंछ को पुनः प्राप्त करें।

list = [1, 2, 3]
IO.puts(hd(list))
IO.puts(tl(list))

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

1
[2, 3]

Note - खाली सूची का सिर या पूंछ प्राप्त करना एक त्रुटि है।

अन्य सूची कार्य

अमृत ​​मानक पुस्तकालय सूचियों से निपटने के लिए कई प्रकार के कार्य प्रदान करता है। हम उनमें से कुछ लोगों पर एक नज़र डालेंगे। आप बाकी की सूची यहां देख सकते हैं ।

क्र.सं. समारोह का नाम और विवरण
1

delete(list, item)

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

2

delete_at(list, index)

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

3

first(list)

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

4

flatten(list)

नेस्टेड सूचियों की दी गई सूची को समतल करता है।

5

insert_at(list, index, value)

निर्दिष्ट सूचकांक पर डाले गए मूल्य के साथ एक सूची लौटाता है। ध्यान दें कि सूची की लंबाई पर सूचकांक छाया हुआ है। नकारात्मक सूचकांक सूची के अंत से एक ऑफसेट का संकेत देते हैं।

6

last(list)

यदि सूची खाली है, तो सूची में अंतिम तत्व या शून्य लौटाता है।

tuples

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

अमृत ​​टुपल्स को परिभाषित करने के लिए घुंघराले कोष्ठक का उपयोग करता है। सूचियों की तरह, टुपल्स किसी भी मूल्य को पकड़ सकते हैं -

{:ok, "hello"}

एक टपल की लंबाई

टपल की लंबाई पाने के लिए, का उपयोग करें tuple_size निम्नलिखित कार्यक्रम के रूप में कार्य -

IO.puts(tuple_size({:ok, "hello"}))

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

2

एक मूल्य लागू करना

टपल के मान को जोड़ने के लिए, Tuple.append फ़ंक्शन का उपयोग करें -

tuple = {:ok, "Hello"}
Tuple.append(tuple, :world)

यह एक नया tuple बनाएगा और लौटाएगा: {: ok, "Hello",: world}

मान डालना

किसी दिए गए स्थान पर एक मूल्य सम्मिलित करने के लिए, हम या तो उपयोग कर सकते हैं Tuple.insert_at समारोह या put_elemसमारोह। समान समझने के लिए निम्नलिखित उदाहरण पर विचार करें -

tuple = {:bar, :baz}
new_tuple_1 = Tuple.insert_at(tuple, 0, :foo)
new_tuple_2 = put_elem(tuple, 1, :foobar)

नोटिस जो put_elem तथा insert_atनए tuples लौटे। ट्यूपल चर में संग्रहित मूल ट्यूपल को संशोधित नहीं किया गया था क्योंकि अमृत डेटा प्रकार अपरिवर्तनीय हैं। अपरिवर्तनीय होने के कारण, एलिक्सिर कोड के बारे में तर्क करना आसान है क्योंकि आपको कभी भी चिंता करने की आवश्यकता नहीं है यदि एक विशेष कोड आपके डेटा संरचना को जगह में बदल रहा है।

Tuples बनाम सूचियाँ

सूचियों और टुपल्स के बीच अंतर क्या है?

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

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

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

अमृत ​​में, हमारे पास दो मुख्य साहचर्य डेटा संरचनाएँ हैं: कीवर्ड सूचियाँ और मानचित्र। इस अध्याय में, हम कीवर्ड सूचियों पर ध्यान केंद्रित करेंगे।

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

list = [{:a, 1}, {:b, 2}]

अमृत ​​ऐसी सूचियों को परिभाषित करने के लिए एक विशेष वाक्यविन्यास का समर्थन करता है। हम प्रत्येक परमाणु के अंत में कोलन को रख सकते हैं और पूरी तरह से ट्यूपल्स से छुटकारा पा सकते हैं। उदाहरण के लिए,

list_1 = [{:a, 1}, {:b, 2}]
list_2 = [a: 1, b: 2]
IO.puts(list_1 == list_2)

उपरोक्त कार्यक्रम निम्नलिखित परिणाम उत्पन्न करेगा -

true

ये दोनों एक खोजशब्द सूची का प्रतिनिधित्व करते हैं। चूँकि कीवर्ड सूचियाँ भी सूचियाँ हैं, हम उन सभी सूचियों का उपयोग कर सकते हैं जिनका उपयोग हमने उन सूचियों पर किया है।

कीवर्ड सूची में एक परमाणु से जुड़े मूल्य को पुनः प्राप्त करने के लिए, सूची के नाम के बाद परमाणु को [] के रूप में पास करें -

list = [a: 1, b: 2]
IO.puts(list[:a])

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

1

कीवर्ड सूचियों की तीन विशेष विशेषताएं हैं -

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

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

एक कुंजी का उपयोग

किसी दिए गए कुंजी से जुड़े मूल्यों तक पहुंचने के लिए, हम इसका उपयोग करते हैं Keyword.getसमारोह। यह दी गई कुंजी के साथ जुड़ा पहला मान लौटाता है। सभी मान प्राप्त करने के लिए, हम Keyword.get_values ​​फ़ंक्शन का उपयोग करते हैं। उदाहरण के लिए -

kl = [a: 1, a: 2, b: 3] 
IO.puts(Keyword.get(kl, :a)) 
IO.puts(Keyword.get_values(kl))

उपरोक्त कार्यक्रम निम्नलिखित परिणाम उत्पन्न करेगा -

1
[1, 2]

एक कुंजी सम्मिलित करना

एक नया मान जोड़ने के लिए, का उपयोग करें Keyword.put_new। यदि कुंजी पहले से मौजूद है, तो इसका मान अपरिवर्तित रहता है -

kl = [a: 1, a: 2, b: 3]
kl_new = Keyword.put_new(kl, :c, 5)
IO.puts(Keyword.get(kl_new, :c))

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह अतिरिक्त कुंजी के साथ एक नई कीवर्ड सूची बनाता है, सी और निम्नलिखित परिणाम उत्पन्न करता है -

5

एक कुंजी को हटाना

यदि आप किसी कुंजी के लिए सभी प्रविष्टियों को हटाना चाहते हैं, तो उपयोग करें Keyword.delete; कुंजी के लिए केवल पहली प्रविष्टि को हटाने के लिए, उपयोग करें Keyword.delete_first

kl = [a: 1, a: 2, b: 3, c: 0]
kl = Keyword.delete_first(kl, :b)
kl = Keyword.delete(kl, :a)

IO.puts(Keyword.get(kl, :a))
IO.puts(Keyword.get(kl, :b))
IO.puts(Keyword.get(kl, :c))

यह पहले हटा देगा b सूची में और सभी aसूची मैं। जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करेगा -

0

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

यह वह जगह है जहाँ नक्शे आपके बचाव में आते हैं। जब भी आपको एक कुंजी-मूल्य की दुकान की आवश्यकता होती है, नक्शे एलिक्सिर में "गो टू" डेटा संरचना होते हैं।

नक्शा बनाना

% {} सिंटैक्स का उपयोग करके एक मानचित्र बनाया जाता है -

map = %{:a => 1, 2 => :b}

कीवर्ड सूचियों की तुलना में, हम पहले से ही दो अंतर देख सकते हैं -

  • नक्शे किसी भी मूल्य को एक कुंजी के रूप में अनुमति देते हैं।
  • मानचित्र की कुंजियाँ किसी भी क्रम का पालन नहीं करती हैं।

एक कुंजी का उपयोग

किसी कुंजी के साथ जुड़े मूल्य को अर्जित करने के लिए, मैप्स कीवर्ड सूचियों के समान सिंटैक्स का उपयोग करते हैं -

map = %{:a => 1, 2 => :b}
IO.puts(map[:a])
IO.puts(map[2])

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

1
b

एक कुंजी सम्मिलित करना

मानचित्र में एक कुंजी सम्मिलित करने के लिए, हम इसका उपयोग करते हैं Dict.put_new फ़ंक्शन जो नक्शे, नई कुंजी और नए मान को तर्क के रूप में लेता है -

map = %{:a => 1, 2 => :b}
new_map = Dict.put_new(map, :new_val, "value") 
IO.puts(new_map[:new_val])

यह कुंजी-मूल्य जोड़ी को सम्मिलित करेगा :new_val - "value"एक नए नक्शे में जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

"value"

एक मूल्य अद्यतन करना

मानचित्र में पहले से मौजूद मान को अपडेट करने के लिए, आप निम्नलिखित सिंटैक्स का उपयोग कर सकते हैं -

map = %{:a => 1, 2 => :b}
new_map = %{ map | a: 25}
IO.puts(new_map[:a])

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

25

पैटर्न मिलान

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

%{:a => a} = %{:a => 1, 2 => :b}
IO.puts(a)

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

1

यह मैच होगा a साथ में 1। और इसलिए, यह आउटपुट उत्पन्न करेगा1

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

मानचित्र कुंजियों को एक्सेस करने, मिलान करने और जोड़ने पर चर का उपयोग किया जा सकता है -

n = 1
map = %{n => :one}
%{^n => :one} = %{1 => :one, 2 => :two, 3 => :three}

मैप मॉड्यूल नक्शे में हेरफेर करने के लिए सुविधा कार्यों के साथ कीवर्ड मॉड्यूल के लिए बहुत समान एपीआई प्रदान करता है। आप जैसे कार्यों का उपयोग कर सकते हैंMap.get, Map.delete, नक्शे में हेरफेर करने के लिए।

एटम कुंजी के साथ मैप्स

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

map = %{:a => 1, 2 => :b} 
IO.puts(map.a)

मानचित्रों की एक और दिलचस्प संपत्ति यह है कि वे परमाणु कुंजियों को अद्यतन और एक्सेस करने के लिए अपना स्वयं का सिंटैक्स प्रदान करते हैं -

map = %{:a => 1, 2 => :b}
IO.puts(map.a)

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

1

ध्यान दें कि इस तरह से एटम कीज को एक्सेस करने के लिए, यह मौजूद होना चाहिए या प्रोग्राम काम करने में विफल हो जाएगा।

अमृत ​​में, हम कई कार्यों को मॉड्यूल में समूहित करते हैं। हमने पिछले अध्यायों में पहले से ही अलग-अलग मॉड्यूलों का उपयोग किया है जैसे स्ट्रिंग मॉड्यूल, बिटवाइज़ मॉड्यूल, ट्यूपल मॉड्यूल, आदि।

अमृत ​​में अपने स्वयं के मॉड्यूल बनाने के लिए, हम उपयोग करते हैं defmoduleमैक्रो। हम उपयोग करते हैंdef उस मॉड्यूल में कार्यों को परिभाषित करने के लिए मैक्रो -

defmodule Math do
   def sum(a, b) do
      a + b
   end
end

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

संकलन

फ़ाइलों में मॉड्यूल लिखना हमेशा सुविधाजनक होता है ताकि उन्हें संकलित और पुन: उपयोग किया जा सके। चलिए मान लेते हैं कि हमारे पास निम्नलिखित सामग्री के साथ math.ex नामक एक फाइल है -

defmodule Math do
   def sum(a, b) do
      a + b
   end
end

हम कमांड का उपयोग करके फाइल संकलित कर सकते हैं -elixirc :

$ elixirc math.ex

इससे एक फाइल बनेगी जिसका नाम है Elixir.Math.beamपरिभाषित मॉड्यूल के लिए बाइटकोड युक्त। अगर हम शुरू करेंiexफिर से, हमारी मॉड्यूल परिभाषा उपलब्ध होगी (बशर्ते कि iex को उसी निर्देशिका में शुरू किया जाए जो बाइटकोड फ़ाइल है)। उदाहरण के लिए,

IO.puts(Math.sum(1, 2))

उपरोक्त कार्यक्रम निम्नलिखित परिणाम उत्पन्न करेगा -

3

स्क्रिप्टेड मोड

अमृत ​​फाइल एक्सटेंशन के अलावा .ex, अमृत भी समर्थन करता है .exsस्क्रिप्टिंग के लिए फ़ाइलें। अमृत ​​दोनों फाइलों को बिल्कुल उसी तरह व्यवहार करता है, केवल उद्देश्य में अंतर होता है।.ex फ़ाइलों को संकलित किया जाना है जबकि .exs फ़ाइलों का उपयोग किया जाता है scripting। जब निष्पादित किया जाता है, तो दोनों एक्सटेंशन अपने मॉड्यूल को मेमोरी में संकलित और लोड करते हैं, हालांकि केवल.ex फ़ाइलें .beam फ़ाइलों के प्रारूप में डिस्क पर अपना बायोटेक लिखती हैं।

उदाहरण के लिए, यदि हम दौड़ना चाहते हैं Math.sum उसी फ़ाइल में, हम .exs का उपयोग निम्न तरीके से कर सकते हैं -

Math.exs

defmodule Math do
   def sum(a, b) do
      a + b
   end
end
IO.puts(Math.sum(1, 2))

हम इसे अमृत कमांड का उपयोग करके चला सकते हैं -

$ elixir math.exs

उपरोक्त कार्यक्रम निम्नलिखित परिणाम उत्पन्न करेगा -

3

फ़ाइल को मेमोरी में संकलित किया जाएगा और परिणाम के रूप में "3" प्रिंट किया जाएगा। कोई बाइटकोड फ़ाइल नहीं बनाई जाएगी।

मॉड्यूल घोंसले का शिकार

मॉड्यूल अमृत में घोंसले के शिकार हो सकते हैं। भाषा की यह विशेषता हमें अपने कोड को बेहतर तरीके से व्यवस्थित करने में मदद करती है। नेस्टेड मॉड्यूल बनाने के लिए, हम निम्नलिखित सिंटैक्स का उपयोग करते हैं -

defmodule Foo do
   #Foo module code here
   defmodule Bar do
      #Bar module code here
   end
end

ऊपर दिया गया उदाहरण दो मॉड्यूल को परिभाषित करेगा: Foo तथा Foo.Bar। दूसरे के रूप में पहुँचा जा सकता हैBar के भीतर Fooजब तक वे एक ही शाब्दिक दायरे में हैं। यदि, बाद में,Bar मॉड्यूल को फू मॉड्यूल परिभाषा के बाहर ले जाया जाता है, इसे इसके पूर्ण नाम (Foo.Bar) द्वारा संदर्भित किया जाना चाहिए या उपनाम अन्य अध्याय में चर्चा किए गए उपनाम निर्देश का उपयोग करके सेट किया जाना चाहिए।

Note- अमृत में, फू मॉड्यूल को परिभाषित करने के लिए फू मॉड्यूल को परिभाषित करने की आवश्यकता नहीं है, क्योंकि भाषा सभी मॉड्यूल नामों को परमाणुओं में अनुवाद करती है। आप श्रृंखला में किसी भी मॉड्यूल को परिभाषित किए बिना मनमाने ढंग से मॉड्यूल को परिभाषित कर सकते हैं। उदाहरण के लिए, आप परिभाषित कर सकते हैंFoo.Bar.Baz परिभाषित किए बिना Foo या Foo.Bar

सॉफ्टवेयर के पुन: उपयोग की सुविधा के लिए, अमृत तीन निर्देश प्रदान करता है - alias, require तथा import। यह एक मैक्रो नामक उपयोग भी प्रदान करता है जिसे नीचे संक्षेप में प्रस्तुत किया गया है -

# Alias the module so it can be called as Bar instead of Foo.Bar
alias Foo.Bar, as: Bar

# Ensure the module is compiled and available (usually for macros)
require Foo

# Import functions from Foo so they can be called without the `Foo.` prefix
import Foo

# Invokes the custom code defined in Foo as an extension point
use Foo

आइए अब प्रत्येक निर्देश के बारे में विस्तार से समझते हैं।

उपनाम

उपनाम निर्देश आपको किसी भी दिए गए मॉड्यूल नाम के लिए उपनाम सेट करने की अनुमति देता है। उदाहरण के लिए, यदि आप एक उपनाम देना चाहते हैं'Str' स्ट्रिंग मॉड्यूल के लिए, आप बस लिख सकते हैं -

alias String, as: Str
IO.puts(Str.length("Hello"))

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

5

को एक उपनाम दिया जाता है String मॉड्यूल के रूप में Str। अब जब हम Str शाब्दिक का उपयोग करके किसी भी फ़ंक्शन को कॉल करते हैं, तो यह वास्तव में इसका संदर्भ देता हैStringमापांक। यह तब बहुत मददगार होता है जब हम बहुत लंबे मॉड्यूल नामों का उपयोग करते हैं और उन लोगों को मौजूदा दायरे में स्थानापन्न करना चाहते हैं।

NOTE - उपनाम MUST बड़े अक्षर से शुरू करें।

उपनाम केवल भीतर मान्य हैं lexical scope उन्हें अंदर बुलाया जाता है। उदाहरण के लिए, यदि आपके पास फ़ाइल में 2 मॉड्यूल हैं और किसी एक मॉड्यूल के भीतर एक उपनाम बनाते हैं, तो दूसरा मॉड्यूल में उपनाम उपलब्ध नहीं होगा।

यदि आप अंतर्निहित मॉड्यूल का नाम देते हैं, जैसे स्ट्रिंग या ट्यूपल, कुछ अन्य मॉड्यूल के लिए एक अन्य के रूप में, इनबिल्ट मॉड्यूल तक पहुंचने के लिए, आपको इसके साथ प्रस्तुत करने की आवश्यकता होगी "Elixir."। उदाहरण के लिए,

alias List, as: String
#Now when we use String we are actually using List.
#To use the string module: 
IO.puts(Elixir.String.length("Hello"))

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

5

की आवश्यकता होती है

Elixir मेटा-प्रोग्रामिंग के लिए एक तंत्र के रूप में मैक्रोज़ प्रदान करता है (कोड जो कोड उत्पन्न करता है)।

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

Integer.is_odd(3)

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करेगा -

** (CompileError) iex:1: you must require Integer before invoking the macro Integer.is_odd/1

अमृत ​​में, Integer.is_odd के रूप में परिभाषित किया गया है macro। इस मैक्रो का उपयोग गार्ड के रूप में किया जा सकता है। इसका मतलब है कि, आह्वान करने के लिएInteger.is_odd, हमें इंटेगर मॉड्यूल की आवश्यकता होगी।

उपयोग require Integer कार्य करें और नीचे दिखाए अनुसार कार्यक्रम चलाएं।

require Integer
Integer.is_odd(3)

इस बार कार्यक्रम चलेगा और उत्पादन का उत्पादन निम्नानुसार करेगा: true

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

आयात

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

import List, only: [duplicate: 2]

इस मामले में, हम सूची से केवल फ़ंक्शन डुप्लिकेट (तर्क सूची लंबाई 2 के साथ) आयात कर रहे हैं। हालांकि:only वैकल्पिक है, इसके उपयोग की सिफारिश की जाती है ताकि नामपत्र के अंदर दिए गए मॉड्यूल के सभी कार्यों को आयात करने से बचा जा सके। :except कार्यों की सूची को छोड़कर एक मॉड्यूल में सब कुछ आयात करने के लिए एक विकल्प के रूप में भी दिया जा सकता है।

import निर्देशन भी समर्थन करता है :macros तथा :functions को दिया जाए :only। उदाहरण के लिए, सभी मैक्रोज़ को आयात करने के लिए, एक उपयोगकर्ता लिख ​​सकता है -

import Integer, only: :macros

ध्यान दें कि आयात भी है Lexically scopedबस आवश्यकता और उपनाम निर्देश की तरह। उस पर भी ध्यान दें'import'ing a module also 'require's it

उपयोग

हालांकि एक निर्देश नहीं, use एक मैक्रो कसकर संबंधित है requireजो आपको वर्तमान संदर्भ में एक मॉड्यूल का उपयोग करने की अनुमति देता है। उपयोग मैक्रो अक्सर डेवलपर्स द्वारा बाहरी कार्यक्षमता को वर्तमान लेक्सिकल दायरे में लाने के लिए उपयोग किया जाता है, अक्सर मॉड्यूल। आइए एक उदाहरण के माध्यम से उपयोग निर्देश को समझते हैं -

defmodule Example do 
   use Feature, option: :value 
end

उपयोग एक मैक्रो है जो उपरोक्त को में बदल देता है -

defmodule Example do
   require Feature
   Feature.__using__(option: :value)
end

use Module पहले मॉड्यूल की आवश्यकता होती है और फिर कॉल की जाती है __using__मॉड्यूल पर मैक्रो। अमृत ​​में महान रूपक क्षमताएँ होती हैं और संकलन समय पर कोड उत्पन्न करने के लिए इसमें मैक्रोज़ होते हैं। _ _Use__ मैक्रो को उपरोक्त उदाहरण में कहा जाता है, और कोड हमारे स्थानीय संदर्भ में इंजेक्ट किया जाता है। स्थानीय संदर्भ वह है जहां संकलन के समय उपयोग मैक्रो को बुलाया गया था।

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

अमृत ​​में 2 प्रकार के कार्य हैं -

अनाम फ़ंक्शन

का उपयोग कर परिभाषित कार्य fn..end constructअनाम कार्य हैं। इन कार्यों को कभी-कभी लंबोदर भी कहा जाता है। उन्हें चर नामों से निर्दिष्ट करके उपयोग किया जाता है।

नाम दिया गया कार्य

का उपयोग कर परिभाषित कार्य def keywordनामित कार्य हैं। ये मूल कार्य हैं जो अमृत में प्रदान किए गए हैं।

अनाम कार्य

जैसा कि नाम का अर्थ है, एक अनाम फ़ंक्शन का कोई नाम नहीं है। इन्हें अक्सर अन्य कार्यों के लिए पारित किया जाता है। अमृत ​​में एक अनाम फ़ंक्शन को परिभाषित करने के लिए, हमें इसकी आवश्यकता हैfn तथा endकीवर्ड। इनके भीतर, हम किसी भी पैरामीटर और फ़ंक्शन बॉडी को अलग से परिभाषित कर सकते हैं->। उदाहरण के लिए,

sum = fn (a, b) -> a + b end
IO.puts(sum.(1, 5))

जब प्रोग्राम से ऊपर चला जाता है, चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

6

ध्यान दें कि इन कार्यों को नामित कार्यों की तरह नहीं कहा जाता है। हमारे पास एक '.'फ़ंक्शन नाम और उसके तर्कों के बीच।

कैप्चर ऑपरेटर का उपयोग करना

हम कैप्चर ऑपरेटर का उपयोग करके इन कार्यों को भी परिभाषित कर सकते हैं। यह फ़ंक्शन बनाने का एक आसान तरीका है। अब हम कैप्चर ऑपरेटर का उपयोग करके उपरोक्त योग फ़ंक्शन को परिभाषित करेंगे,

sum = &(&1 + &2) 
IO.puts(sum.(1, 2))

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

3

शॉर्टहैंड संस्करण में, हमारे पैरामीटर का नाम नहीं है, लेकिन हमारे लिए & 1, & 2, & 3, और इसी तरह उपलब्ध हैं।

पैटर्न मिलान कार्य

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

handle_result = fn
   {var1} -> IO.puts("#{var1} found in a tuple!")
   {var_2, var_3} -> IO.puts("#{var_2} and #{var_3} found!")
end
handle_result.({"Hey people"})
handle_result.({"Hello", "World"})

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Hey people found in a tuple!
Hello and World found!

नामित कार्य

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

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

def function_name(argument_1, argument_2) do
   #code to be executed when function is called
end

अब हम गणित मॉड्यूल के भीतर हमारे नामित फ़ंक्शन योग को परिभाषित करते हैं।

defmodule Math do
   def sum(a, b) do
      a + b
   end
end

IO.puts(Math.sum(5, 6))

कार्यक्रम से ऊपर चलने पर, यह निम्नलिखित परिणाम उत्पन्न करता है -

11

1-लाइनर कार्यों के लिए, इन कार्यों को परिभाषित करने के लिए शॉर्टहैंड नोटेशन का उपयोग किया जाता है do:। उदाहरण के लिए -

defmodule Math do
   def sum(a, b), do: a + b
end
IO.puts(Math.sum(5, 6))

कार्यक्रम से ऊपर चलने पर, यह निम्नलिखित परिणाम उत्पन्न करता है -

11

निजी कार्य

अमृत ​​हमें उन निजी कार्यों को परिभाषित करने की क्षमता प्रदान करता है जिन्हें मॉड्यूल के भीतर से एक्सेस किया जा सकता है जिसमें वे परिभाषित हैं। एक निजी फ़ंक्शन को परिभाषित करने के लिए, उपयोग करेंdefp के बजाय def। उदाहरण के लिए,

defmodule Greeter do
   def hello(name), do: phrase <> name
   defp phrase, do: "Hello "
end

Greeter.hello("world")

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Hello world

लेकिन अगर हम केवल वाक्यांश फ़ंक्शन को स्पष्ट रूप से कॉल करने का प्रयास करते हैं, तो Greeter.phrase() फ़ंक्शन, यह एक त्रुटि बढ़ाएगा।

डिफ़ॉल्ट तर्क

यदि हम किसी तर्क के लिए डिफ़ॉल्ट मान चाहते हैं, तो हम इसका उपयोग करते हैं argument \\ value वाक्य रचना -

defmodule Greeter do
   def hello(name, country \\ "en") do
      phrase(country) <> name
   end

   defp phrase("en"), do: "Hello, "
   defp phrase("es"), do: "Hola, "
end

Greeter.hello("Ayush", "en")
Greeter.hello("Ayush")
Greeter.hello("Ayush", "es")

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Hello, Ayush
Hello, Ayush
Hola, Ayush

रिकर्सियन एक ऐसी विधि है जहां किसी समस्या का समाधान उसी समस्या के छोटे उदाहरणों के समाधान पर निर्भर करता है। अधिकांश कंप्यूटर प्रोग्रामिंग भाषाएं प्रोग्राम टेक्स्ट के भीतर एक फ़ंक्शन को कॉल करने की अनुमति देकर पुनरावृत्ति का समर्थन करती हैं।

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

defmodule Math do
   def fact(res, num) do
   if num === 1 do
      res
   else
      new_res = res * num
      fact(new_res, num-1)
      end
   end
end

IO.puts(Math.fact(1,5))

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

120

इसलिए उपरोक्त कार्य में, Math.fact, हम एक संख्या के भाज्य की गणना कर रहे हैं। ध्यान दें कि हम फ़ंक्शन को अपने भीतर बुला रहे हैं। आइए अब समझते हैं कि यह कैसे काम करता है।

हमने इसे 1 और उस संख्या के साथ प्रदान किया है जिसकी गुत्थी हम गणना करना चाहते हैं। फ़ंक्शन यह जांचता है कि संख्या 1 है या नहीं और यदि यह 1 है तो रिटर्न फिर से शुरू होता है(Ending condition)। यदि नहीं, तो यह एक वैरिएबल new_res बनाता है और इसे पिछले res * current num का मान प्रदान करता है। यह हमारे फ़ंक्शन कॉल तथ्य (new_res, num-1) द्वारा लौटाया गया मान लौटाता है । यह दोहराता है जब तक कि हम संख्या 1 प्राप्त नहीं करते हैं। एक बार ऐसा होने पर, हमें परिणाम मिलता है।

आइए एक और उदाहरण पर विचार करें, सूची के प्रत्येक तत्व को एक-एक करके प्रिंट करें। ऐसा करने के लिए, हम इसका उपयोग करेंगेhd तथा tl कार्यों में सूचियों और पैटर्न मिलान के कार्य -

a = ["Hey", 100, 452, :true, "People"]
defmodule ListPrint do
   def print([]) do
   end
   def print([head | tail]) do 
      IO.puts(head)
      print(tail)
   end
end

ListPrint.print(a)

जब हमारे पास एक खाली सूची होती है, तो पहले प्रिंट फ़ंक्शन को कहा जाता है(ending condition)। यदि नहीं, तो दूसरे प्रिंट फ़ंक्शन को बुलाया जाएगा जो सूची को 2 में विभाजित करेगा और सूची के पहले तत्व को सिर पर और शेष सूची को पूंछ में असाइन करेगा। फिर सिर प्रिंट हो जाता है और हम बाकी की सूची के साथ फिर से प्रिंट फ़ंक्शन को कॉल करते हैं, अर्थात, पूंछ। जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Hey
100
452
true
People

अपरिवर्तनीयता के कारण, एलिक्सिर (किसी भी कार्यात्मक प्रोग्रामिंग भाषा में) में छोरों को अलग-अलग भाषाओं में लिखा जाता है। उदाहरण के लिए, C जैसी अनिवार्य भाषा में, आप लिखेंगे -

for(i = 0; i < 10; i++) {
   printf("%d", array[i]);
}

ऊपर दिए गए उदाहरण में, हम सरणी और चर दोनों को बदल रहे हैं i। अमृत ​​में संभव नहीं है। इसके बजाय, कार्यात्मक भाषाएं पुनरावृत्ति पर निर्भर करती हैं: एक फ़ंक्शन को पुनरावर्ती कहा जाता है जब तक कि एक स्थिति तक नहीं पहुंच जाती है जो पुनरावर्ती कार्रवाई को जारी रखने से रोकती है। इस प्रक्रिया में कोई डेटा म्यूट नहीं किया गया है।

आइए अब हम एक सरल लूप लिखते हैं जिसमें रिकर्सन का उपयोग किया जाता है जो हैलो प्रिंट करता है n बार।

defmodule Loop do
   def print_multiple_times(msg, n) when n <= 1 do
      IO.puts msg
   end

   def print_multiple_times(msg, n) do
      IO.puts msg
      print_multiple_times(msg, n - 1)
   end
end

Loop.print_multiple_times("Hello", 10)

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Hello
Hello
Hello
Hello
Hello
Hello
Hello
Hello
Hello
Hello

हमने लूप को सफलतापूर्वक लागू करने के लिए फ़ंक्शन के पैटर्न मिलान तकनीक और पुनरावृत्ति का उपयोग किया है। पुनरावर्ती परिभाषाओं को समझना मुश्किल है लेकिन लूप को पुनरावृत्ति में परिवर्तित करना आसान है।

अमृत ​​हमें प्रदान करता है Enum module। यह मॉड्यूल सबसे पुनरावृत्त लूपिंग कॉल के लिए उपयोग किया जाता है क्योंकि उसी के लिए पुनरावर्ती परिभाषाओं को जानने की कोशिश करने वालों की तुलना में इसका उपयोग करना बहुत आसान है। हम अगले अध्याय में उन पर चर्चा करेंगे। आपकी स्वयं की पुनरावर्ती परिभाषा का उपयोग केवल तब किया जाना चाहिए जब आप उस मॉड्यूल का उपयोग करके समाधान नहीं ढूंढते। उन कार्यों पूंछ कॉल अनुकूलित और काफी तेजी से कर रहे हैं।

एक enumerable एक ऑब्जेक्ट है जिसे enumerated किया जा सकता है। "एनुमरेटेड" का अर्थ है एक सेट / संग्रह / श्रेणी के सदस्यों को एक-एक करके गिनना (आमतौर पर क्रम में, आमतौर पर नाम से)।

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

ये सभी कार्य पहले तत्व के रूप में एक और दूसरे के रूप में एक फ़ंक्शन के रूप में एक enumerable लेते हैं और उन पर काम करते हैं। कार्यों का वर्णन नीचे दिया गया है।

सब?

जब हम उपयोग करते हैं all? फ़ंक्शन, पूरे संग्रह को सही का मूल्यांकन करना चाहिए अन्यथा झूठे को वापस कर दिया जाएगा। उदाहरण के लिए, यह जांचने के लिए कि सूची के सभी तत्व विषम संख्या में हैं, तो।

res = Enum.all?([1, 2, 3, 4], fn(s) -> rem(s,2) == 1 end) 
IO.puts(res)

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

false

ऐसा इसलिए है क्योंकि इस सूची के सभी तत्व विषम नहीं हैं।

कोई भी?

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

res = Enum.any?([1, 2, 3, 4], fn(s) -> rem(s,2) == 1 end)
IO.puts(res)

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

true

टुकड़ा

यह फ़ंक्शन हमारे संग्रह को दूसरे तर्क के रूप में प्रदान किए गए आकार के छोटे हिस्से में विभाजित करता है। उदाहरण के लिए -

res = Enum.chunk([1, 2, 3, 4, 5, 6], 2)
IO.puts(res)

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

[[1, 2], [3, 4], [5, 6]]

से प्रत्येक

एक नए मूल्य का उत्पादन किए बिना एक संग्रह पर पुनरावृति करना आवश्यक हो सकता है, इस मामले के लिए हम उपयोग करते हैं each कार्य -

Enum.each(["Hello", "Every", "one"], fn(s) -> IO.puts(s) end)

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Hello
Every
one

नक्शा

प्रत्येक आइटम के लिए हमारे फ़ंक्शन को लागू करने और एक नया संग्रह बनाने के लिए हम मानचित्र फ़ंक्शन का उपयोग करते हैं। यह कार्यात्मक प्रोग्रामिंग में सबसे उपयोगी निर्माणों में से एक है क्योंकि यह काफी अभिव्यंजक और छोटा है। इसे समझने के लिए एक उदाहरण पर विचार करते हैं। हम एक सूची में संग्रहीत मानों को दोगुना करेंगे और इसे एक नई सूची में संग्रहीत करेंगेres -

res = Enum.map([2, 5, 3, 6], fn(a) -> a*2 end)
IO.puts(res)

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

[4, 10, 6, 12]

कम करना

reduceफ़ंक्शन हमें एक मूल्य के लिए हमारे गणना करने में मदद करता है। ऐसा करने के लिए, हम अपने फ़ंक्शन में पारित होने के लिए एक वैकल्पिक संचायक (इस उदाहरण में 5) की आपूर्ति करते हैं; यदि कोई संचायक प्रदान नहीं किया जाता है, तो पहले मूल्य का उपयोग किया जाता है -

res = Enum.reduce([1, 2, 3, 4], 5, fn(x, accum) -> x + accum end)
IO.puts(res)

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

15

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

res = Enum.reduce([1, 2, 3, 4], fn(x, accum) -> x + accum end)
IO.puts(res)

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

10

uniq

यूनीक फ़ंक्शन हमारे संग्रह से डुप्लिकेट को निकालता है और संग्रह में तत्वों का केवल सेट लौटाता है। उदाहरण के लिए -

res = Enum.uniq([1, 2, 2, 3, 3, 3, 4, 4, 4, 4])
IO.puts(res)

कार्यक्रम से ऊपर चलने पर, यह निम्नलिखित परिणाम उत्पन्न करता है -

[1, 2, 3, 4]

ईगर मूल्यांकन

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

odd? = &(odd? = &(rem(&1, 2) != 0) 
res = 1..100_000 |> Enum.map(&(&1 * 3)) |> Enum.filter(odd?) |> Enum.sum 
IO.puts(res)

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

7500000000

ऊपर के उदाहरण में संचालन की एक पाइपलाइन है। हम एक सीमा से शुरू करते हैं और फिर 3. में प्रत्येक तत्व को 3 से गुणा करते हैं। यह पहला ऑपरेशन अब 100_000 आइटम के साथ एक सूची बनाएगा और वापस करेगा। फिर हम सूची से सभी विषम तत्वों को रखते हैं, एक नई सूची बनाते हैं, अब 50_000 वस्तुओं के साथ, और फिर हम सभी प्रविष्टियों को जोड़ते हैं।

|> ऊपर दिए गए स्निपेट में प्रयुक्त प्रतीक है pipe operator: यह बस अपनी बाईं ओर के अभिव्यक्ति से आउटपुट लेता है और इसे अपने दाहिने तरफ फ़ंक्शन कॉल के पहले तर्क के रूप में पास करता है। यह यूनिक्स के समान है | ऑपरेटर। इसका उद्देश्य कार्यों की एक श्रृंखला द्वारा परिवर्तित किए जा रहे डेटा के प्रवाह को उजागर करना है।

के बिना pipe ऑपरेटर, कोड जटिल लग रहा है -

Enum.sum(Enum.filter(Enum.map(1..100_000, &(&1 * 3)), odd?))

हमारे पास कई अन्य कार्य हैं, हालांकि, केवल कुछ महत्वपूर्ण ही यहां वर्णित किए गए हैं।

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

धाराएं आलसियों द्वारा किए गए उत्सुक अभियानों के विपरीत आलसी कार्यों का समर्थन करती हैं। संक्षेप में,streams are lazy, composable enumerables। इसका मतलब यह है कि जब तक यह बिल्कुल आवश्यक नहीं है तब तक स्ट्रीम एक ऑपरेशन नहीं करती है। आइए इसे समझने के लिए एक उदाहरण पर विचार करें -

odd? = &(rem(&1, 2) != 0)
res = 1..100_000 |> Stream.map(&(&1 * 3)) |> Stream.filter(odd?) |> Enum.sum
IO.puts(res)

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

7500000000

ऊपर दिए गए उदाहरण में, 1..100_000 |> Stream.map(&(&1 * 3))एक डेटा प्रकार, एक वास्तविक स्ट्रीम देता है, जो 1..100_000 की सीमा पर मानचित्र गणना का प्रतिनिधित्व करता है। इसने अभी तक इस प्रतिनिधित्व का मूल्यांकन नहीं किया है। इंटरमीडिएट लिस्ट जनरेट करने के बजाय, धाराएँ अभिकलन की एक श्रंखला का निर्माण करती हैं, जो केवल तब संलग्न होती हैं जब हम अंतर्निहित स्ट्रीम को Enum मॉड्यूल में पास करते हैं। धाराएँ बड़े, संभवतः अनंत, संग्रह के साथ काम करते समय उपयोगी होती हैं।

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

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

chunk(enum, n, step, leftover \\ nil)

प्रत्येक आइटम में एन्यूमरेबल को प्रवाहित करता है, जिसमें प्रत्येक आइटम एन होता है, जहां प्रत्येक नया हिस्सा एन्यूमरेबल में स्टेप एलिमेंट्स शुरू करता है।

2

concat(enumerables)

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

3

each(enum, fun)

प्रत्येक आइटम के लिए दिए गए फ़ंक्शन को निष्पादित करता है।

4

filter(enum, fun)

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

5

map(enum, fun)

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

6

drop(enum, n)

लाजिली एनुमरबल से अगली एन वस्तुओं को गिराती है।

संरचनाएं नक्शे के शीर्ष पर निर्मित एक्सटेंशन हैं जो संकलन-समय की जांच और डिफ़ॉल्ट मान प्रदान करते हैं।

संरचनाओं को परिभाषित करना

एक संरचना को परिभाषित करने के लिए, बाधा निर्माण का उपयोग किया जाता है -

defmodule User do
   defstruct name: "John", age: 27
end

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

new_john = %User{})
ayush = %User{name: "Ayush", age: 20}
megan = %User{name: "Megan"})

उपरोक्त कोड मानों के साथ तीन अलग-अलग संरचनाएं उत्पन्न करेगा -

%User{age: 27, name: "John"}
%User{age: 20, name: "Ayush"}
%User{age: 27, name: "Megan"}

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

एक्सेसिंग और अपडेटिंग स्ट्रक्चर

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

defmodule User do
   defstruct name: "John", age: 27
end
john = %User{}
#john right now is: %User{age: 27, name: "John"}

#To access name and age of John, 
IO.puts(john.name)
IO.puts(john.age)

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

John
27

किसी संरचना में एक मूल्य को अद्यतन करने के लिए, हम फिर से उसी प्रक्रिया का उपयोग करेंगे जिसका उपयोग हमने मानचित्र अध्याय में किया था,

meg = %{john | name: "Meg"}

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

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

आइए हम प्रोटोकॉल का उपयोग करने के एक उदाहरण पर विचार करें। हमने एक फ़ंक्शन का उपयोग किया हैto_stringअन्य प्रकारों से स्ट्रिंग प्रकार में बदलने के लिए पिछले अध्यायों में। यह वास्तव में एक प्रोटोकॉल है। यह उस इनपुट के अनुसार कार्य करता है जो त्रुटि उत्पन्न किए बिना दिया जाता है। ऐसा लग सकता है कि हम पैटर्न मिलान कार्यों पर चर्चा कर रहे हैं, लेकिन जैसे-जैसे हम आगे बढ़ते हैं, यह अलग हो जाता है।

प्रोटोकॉल तंत्र को और समझने के लिए निम्नलिखित उदाहरण पर विचार करें।

आइए हम एक प्रोटोकॉल बनाएं जो यह प्रदर्शित करेगा कि दिया गया इनपुट खाली है या नहीं। हम इस प्रोटोकॉल को कहेंगेblank?

एक प्रोटोकॉल को परिभाषित करना

हम निम्नलिखित तरीके से अमृत में एक प्रोटोकॉल को परिभाषित कर सकते हैं -

defprotocol Blank do
   def blank?(data)
end

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

तो यह प्रोटोकॉल कह रहा है कि जो कुछ भी लागू होता है वह एक होना चाहिए empty?फ़ंक्शन, हालांकि यह कार्यान्वयनकर्ता पर निर्भर है कि फ़ंक्शन कैसे प्रतिक्रिया करता है। परिभाषित प्रोटोकॉल के साथ, आइए हम समझते हैं कि कार्यान्वयन के एक जोड़े को कैसे जोड़ा जाए।

एक प्रोटोकॉल को लागू करना

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

#Defining the protocol
defprotocol Blank do
   def blank?(data)
end

#Implementing the protocol for lists
defimpl Blank, for: List do
   def blank?([]), do: true
   def blank?(_), do: false
end

#Implementing the protocol for strings
defimpl Blank, for: BitString do
   def blank?(""), do: true
   def blank?(_), do: false
end

#Implementing the protocol for maps
defimpl Blank, for: Map do
   def blank?(map), do: map_size(map) == 0
end

IO.puts(Blank.blank? [])
IO.puts(Blank.blank? [:true, "Hello"])
IO.puts(Blank.blank? "")
IO.puts(Blank.blank? "Hi")

आप अपने प्रोटोकॉल को जितने चाहें उतने या कुछ प्रकार के लिए लागू कर सकते हैं, जो कुछ भी आपके प्रोटोकॉल के उपयोग के लिए समझ में आता है। यह प्रोटोकॉल का एक बहुत ही बुनियादी उपयोग मामला था। जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

true
false
true
false

Note - यदि आप इसके लिए किसी अन्य प्रकार के लिए इसका उपयोग करते हैं, जिसके लिए आपने प्रोटोकॉल को परिभाषित किया है, तो यह एक त्रुटि उत्पन्न करेगा।

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

पथ मॉड्यूल

pathमॉड्यूल एक बहुत छोटा मॉड्यूल है जिसे फाइलसिस्टम संचालन के लिए सहायक मॉड्यूल माना जा सकता है। फ़ाइल मॉड्यूल में अधिकांश कार्य तर्कों के रूप में पथों की अपेक्षा करते हैं। आमतौर पर, वे रास्ते नियमित बायनेरिज़ होंगे। पथ मॉड्यूल ऐसे रास्तों के साथ काम करने के लिए सुविधाएं प्रदान करता है। पथ मॉड्यूल से फ़ंक्शन का उपयोग करना केवल बायनेरिज़ के साथ छेड़छाड़ करने के विरोध के रूप में पसंद किया जाता है क्योंकि पथ मॉड्यूल पारदर्शी रूप से विभिन्न ऑपरेटिंग सिस्टम का ख्याल रखता है। यह देखा जाना चाहिए कि एलिक्सिर फ़ाइल संचालन करते समय विंडोज पर स्लैश (/) को बैकस्लैश (\) में स्वचालित रूप से बदल देगा।

पथ मॉड्यूल को और समझने के लिए निम्नलिखित उदाहरण पर विचार करें -

IO.puts(Path.join("foo", "bar"))

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

foo/bar

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

फ़ाइल मॉड्यूल

फ़ाइल मॉड्यूल में ऐसे कार्य हैं जो हमें IO उपकरणों के रूप में फाइल खोलने की अनुमति देते हैं। डिफ़ॉल्ट रूप से, फाइलें बाइनरी मोड में खोली जाती हैं, जिसके लिए डेवलपर्स को विशिष्ट का उपयोग करने की आवश्यकता होती हैIO.binread तथा IO.binwriteIO मॉड्यूल से कार्य करता है। आइये एक फाइल बनाते हैं जिसका नाम हैnewfile और इसके लिए कुछ डेटा लिखें।

{:ok, file} = File.read("newfile", [:write]) 
# Pattern matching to store returned stream
IO.binwrite(file, "This will be written to the file")

यदि आप हमारे द्वारा लिखी गई फ़ाइल को खोलने के लिए जाते हैं, तो सामग्री निम्नलिखित तरीके से प्रदर्शित की जाएगी -

This will be written to the file

आइए अब समझते हैं कि फ़ाइल मॉड्यूल का उपयोग कैसे करें।

एक फ़ाइल खोलना

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

{:ok, file} = File.open("newfile")
file = File.open!("newfile")

आइए अब हम इस अंतर को समझते हैं File.open समारोह और File.open!() समारोह।

  • File.openफ़ंक्शन हमेशा एक टपल देता है। यदि फ़ाइल सफलतापूर्वक खोली गई है, तो यह टुपल में पहला मान देता है:okऔर दूसरा मान io_device का शाब्दिक है। यदि कोई त्रुटि होती है, तो यह पहले मान के साथ एक टपल लौटाएगा:error और कारण के रूप में दूसरा मूल्य।

  • File.open!() दूसरी ओर फ़ंक्शन वापस आ जाएगा io_deviceयदि फ़ाइल सफलतापूर्वक खोली जाती है तो यह एक त्रुटि उत्पन्न करेगी। नोट: यह वह पैटर्न है जिसके बारे में हम चर्चा करने जा रहे फाइल मॉड्यूल फंक्शन में करते हैं।

हम उन मोड्स को भी निर्दिष्ट कर सकते हैं जिनमें हम इस फाइल को खोलना चाहते हैं। फ़ाइल को केवल पढ़ने के लिए और utf-8 एन्कोडिंग मोड में खोलने के लिए, हम निम्नलिखित कोड का उपयोग करते हैं -

file = File.open!("newfile", [:read, :utf8])

एक फ़ाइल के लिए लेखन

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

File.write("newfile", "Hello")

लेकिन इसका उपयोग नहीं किया जाना चाहिए यदि आप एक ही फाइल में कई राइट्स बना रहे हैं। हर बार इस फ़ंक्शन को लागू करने के बाद, एक फ़ाइल डिस्क्रिप्टर खोला जाता है और फ़ाइल पर लिखने के लिए एक नई प्रक्रिया शुरू की जाती है। यदि आप एक लूप में एकाधिक लेखन कर रहे हैं, तो फ़ाइल को खोलेंFile.openऔर IO मॉड्यूल में विधियों का उपयोग करके इसे लिखें। आइए हम इसे समझने के लिए एक उदाहरण पर विचार करें -

#Open the file in read, write and utf8 modes. 
file = File.open!("newfile_2", [:read, :utf8, :write])

#Write to this "io_device" using standard IO functions
IO.puts(file, "Random text")

आप अन्य IO मॉड्यूल विधियों का उपयोग कर सकते हैं जैसे IO.write तथा IO.binwrite io_device के रूप में खोली गई फ़ाइलों को लिखने के लिए।

एक फ़ाइल से पढ़ना

हमारे पास फ़ाइलों से पढ़ने के दो तरीके हैं। फाइल मॉड्यूल से रीड फंक्शन का उपयोग करते हुए पहले देखें।

IO.puts(File.read("newfile"))

इस कोड को चलाते समय, आपको पहले तत्व के साथ एक ट्यूपल प्राप्त करना चाहिए :ok और दूसरा नयाफाइल की सामग्री के रूप में

हम भी उपयोग कर सकते हैं File.read! हमारे पास लौटाई गई फ़ाइलों की सामग्री प्राप्त करने के लिए कार्य करें।

ओपन फाइल को बंद करना

जब भी आप File.open फ़ंक्शन का उपयोग करके किसी फ़ाइल को खोलते हैं, तो इसका उपयोग करने के बाद, आपको इसका उपयोग करके इसे बंद कर देना चाहिए File.close कार्य -

File.close(file)

अमृत ​​में, सभी कोड प्रक्रियाओं के अंदर चलते हैं। प्रक्रियाएं एक-दूसरे से अलग-थलग हैं, एक दूसरे के समांतर चलती हैं और संदेश गुजरने के माध्यम से संवाद करती हैं। अमृत ​​की प्रक्रियाओं को ऑपरेटिंग सिस्टम प्रक्रियाओं के साथ भ्रमित नहीं होना चाहिए। एलिक्सिर में प्रक्रियाएं मेमोरी और सीपीयू (कई अन्य प्रोग्रामिंग भाषाओं में धागे के विपरीत) के मामले में बेहद हल्की हैं। इस वजह से, दसियों या यहां तक ​​कि सैकड़ों हजारों प्रक्रियाओं का एक साथ चलना असामान्य नहीं है।

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

स्पॉन समारोह

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

pid = spawn(fn -> 2 * 2 end)
Process.alive?(pid)

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

false

स्पॉन फ़ंक्शन का रिटर्न मान एक पीआईडी ​​है। यह प्रक्रिया के लिए एक विशिष्ट पहचानकर्ता है और इसलिए यदि आप अपने पीआईडी ​​के ऊपर कोड चलाते हैं, तो यह अलग होगा। जैसा कि आप इस उदाहरण में देख सकते हैं, प्रक्रिया मृत है जब हम यह देखने के लिए जाँच करते हैं कि क्या यह जीवित है। ऐसा इसलिए है क्योंकि प्रक्रिया दी गई फ़ंक्शन को पूरा करते ही बाहर निकल जाएगी।

जैसा कि पहले ही उल्लेख किया गया है, सभी अमृत कोड प्रक्रियाओं के अंदर चलते हैं। यदि आप स्वयं फ़ंक्शन चलाते हैं तो आप अपने वर्तमान सत्र के लिए पीआईडी ​​देखेंगे -

pid = self
 
Process.alive?(pid)

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

true

संदेश देना

हम एक प्रक्रिया के साथ संदेश भेज सकते हैं send और उन्हें प्राप्त करें receive। हमें वर्तमान प्रक्रिया के लिए एक संदेश पास करते हैं और इसे उसी पर प्राप्त करते हैं।

send(self(), {:hello, "Hi people"})

receive do
   {:hello, msg} -> IO.puts(msg)
   {:another_case, msg} -> IO.puts("This one won't match!")
end

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Hi people

हमने भेजें फ़ंक्शन का उपयोग करके वर्तमान प्रक्रिया को एक संदेश भेजा और इसे स्वयं के पीआईडी ​​को पास कर दिया। फिर हमने इनकमिंग मैसेज को हैंडल कियाreceive समारोह।

जब एक संदेश एक प्रक्रिया को भेजा जाता है, तो संदेश में संग्रहीत किया जाता है process mailbox। प्राप्त ब्लॉक वर्तमान प्रक्रिया मेलबॉक्स के माध्यम से एक संदेश के लिए खोज करता है जो किसी भी दिए गए पैटर्न से मेल खाता है। प्राप्त ब्लॉक गार्ड और कई खंडों का समर्थन करता है, जैसे कि मामला।

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

receive do
   {:hello, msg}  -> msg
after
   1_000 -> "nothing after 1s"
end

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

nothing after 1s

NOTE - जब आप पहले से ही मेलबॉक्स में संदेश होने की उम्मीद करते हैं, तो 0 का टाइमआउट दिया जा सकता है।

लिंक

एलिक्सिर में स्पॉनिंग का सबसे आम रूप वास्तव में है spawn_linkसमारोह। स्पॉन_लिंक के साथ एक उदाहरण पर एक नज़र डालने से पहले, आइए समझते हैं कि जब कोई प्रक्रिया विफल होती है तो क्या होता है।

spawn fn -> raise "oops" end

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित त्रुटि पैदा करता है -

[error] Process #PID<0.58.00> raised an exception
** (RuntimeError) oops
   :erlang.apply/2

इसने एक त्रुटि लॉग की लेकिन स्पानिंग प्रक्रिया अभी भी चल रही है। ऐसा इसलिए है क्योंकि प्रक्रियाएं अलग-थलग हैं। यदि हम चाहते हैं कि एक प्रक्रिया में विफलता दूसरे को प्रचारित करने के लिए है, तो हमें उन्हें लिंक करने की आवश्यकता है। इसके साथ किया जा सकता हैspawn_linkसमारोह। आइए हम इसे समझने के लिए एक उदाहरण पर विचार करें -

spawn_link fn -> raise "oops" end

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित त्रुटि पैदा करता है -

** (EXIT from #PID<0.41.0>) an exception was raised:
   ** (RuntimeError) oops
      :erlang.apply/2

अगर आप इसमें दौड़ रहे हैं iexशेल तब शेल इस त्रुटि को संभालता है और बाहर नहीं निकलता है। लेकिन अगर आप पहले स्क्रिप्ट फाइल बनाकर चलते हैं और फिर उपयोग करते हैंelixir <file-name>.exsइस विफलता के कारण मूल प्रक्रिया को भी नीचे लाया जाएगा।

दोष-सहिष्णु प्रणालियों के निर्माण के दौरान प्रक्रियाएं और लिंक एक महत्वपूर्ण भूमिका निभाते हैं। अमृत ​​अनुप्रयोगों में, हम अक्सर अपनी प्रक्रियाओं को पर्यवेक्षकों से जोड़ते हैं जो यह पता लगाते हैं कि कोई प्रक्रिया कब मर जाती है और इसके स्थान पर एक नई प्रक्रिया शुरू होती है। यह केवल इसलिए संभव है क्योंकि प्रक्रियाएँ अलग-थलग हैं और डिफ़ॉल्ट रूप से कुछ भी साझा नहीं करती हैं। और चूंकि प्रक्रियाएं अलग-थलग हैं, इसलिए कोई रास्ता नहीं है कि एक प्रक्रिया में विफलता दूसरे के राज्य को दुर्घटनाग्रस्त या भ्रष्ट कर देगी। जबकि अन्य भाषाओं को हमें अपवादों को पकड़ने / संभालने की आवश्यकता होगी; अमृत ​​में, हम वास्तव में प्रक्रियाओं को विफल होने के साथ ठीक हैं क्योंकि हम पर्यवेक्षकों से अपेक्षा करते हैं कि वे हमारे सिस्टम को ठीक से पुनः आरंभ करें।

राज्य

यदि आप एक ऐसे एप्लिकेशन का निर्माण कर रहे हैं, जिसके लिए राज्य की आवश्यकता होती है, उदाहरण के लिए, आपके एप्लिकेशन कॉन्फ़िगरेशन को बनाए रखने के लिए, या आपको किसी फ़ाइल को पार्स करने और इसे मेमोरी में रखने की आवश्यकता है, तो आप इसे कहां संग्रहीत करेंगे? इस तरह की चीजें करते समय अमृत की कार्यशीलता काम में आ सकती है।

हम उन प्रक्रियाओं को लिख सकते हैं जो लूप को असीम रूप से बनाए रखते हैं, राज्य बनाए रखते हैं, और संदेश भेजते हैं और प्राप्त करते हैं। एक उदाहरण के रूप में, हम एक मॉड्यूल लिखते हैं जो नई प्रक्रियाओं को शुरू करता है जो नाम की फ़ाइल में कुंजी-मूल्य स्टोर के रूप में काम करता हैkv.exs

defmodule KV do
   def start_link do
      Task.start_link(fn -> loop(%{}) end)
   end

   defp loop(map) do
      receive do
         {:get, key, caller} ->
         send caller, Map.get(map, key)
         loop(map)
         {:put, key, value} ->
         loop(Map.put(map, key, value))
      end
   end
end

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

आइये अब हम निम्नलिखित चलाते हैं -

iex kv.exs

अब आप में होना चाहिए iexखोल। हमारे मॉड्यूल का परीक्षण करने के लिए, निम्नलिखित प्रयास करें -

{:ok, pid} = KV.start_link

# pid now has the pid of our new process that is being 
# used to get and store key value pairs 

# Send a KV pair :hello, "Hello" to the process
send pid, {:put, :hello, "Hello"}

# Ask for the key :hello
send pid, {:get, :hello, self()}

# Print all the received messages on the current process.
flush()

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

"Hello"

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

regex

Elixir में Regexes sigils हैं। हमने स्ट्रिंग अध्याय में उनके उपयोग को देखा है। आइए हम फिर से एक उदाहरण लें कि हम एलिक्सिर में रेगेक्स का उपयोग कैसे कर सकते हैं।

# A regular expression that matches strings which contain "foo" or
# "bar":
regex = ~r/foo|bar/
IO.puts("foo" =~ regex)
IO.puts("baz" =~ regex)

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

true
false

सिगिल्स 8 अलग-अलग सीमांकक का समर्थन करते हैं -

~r/hello/
~r|hello|
~r"hello"
~r'hello'
~r(hello)
~r[hello]
~r{hello}
~r<hello>

अलग-अलग सीमांकक का समर्थन करने के पीछे कारण यह है कि अलग-अलग सिगिल अलग-अलग सिगिल के लिए अधिक अनुकूल हो सकते हैं। उदाहरण के लिए, नियमित अभिव्यक्ति के लिए कोष्ठक का उपयोग करना एक भ्रामक विकल्प हो सकता है क्योंकि वे रेगेक्स के अंदर कोष्ठक के साथ मिश्रित हो सकते हैं। हालाँकि, कोष्ठक अन्य सिगिल के लिए उपयोगी हो सकता है, जैसा कि हम अगले भाग में देखेंगे।

अमृत ​​पर्ल संगत रेगीज़ का समर्थन करता है और संशोधक का भी समर्थन करता है। आप यहां रेगीज़ के उपयोग के बारे में अधिक पढ़ सकते हैं ।

स्ट्रिंग्स, चार सूचियाँ और वर्ड सूचियाँ

रेगीक्स के अलावा, एलिक्जिर में 3 और इनबिल्ट सिगिल हैं। आइए हम sigils पर एक नजर डालें।

स्ट्रिंग्स

~ S sIG का उपयोग स्ट्रिंग्स उत्पन्न करने के लिए किया जाता है, जैसे डबल कोट्स हैं। ~ S sigil उपयोगी है, उदाहरण के लिए, जब एक स्ट्रिंग में दोहरे और एकल दोनों उद्धरण होते हैं -

new_string = ~s(this is a string with "double" quotes, not 'single' ones)
IO.puts(new_string)

यह सर्जन तार पैदा करता है। जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

"this is a string with \"double\" quotes, not 'single' ones"

चार सूचियाँ

~ C sigil का उपयोग चार सूचियों को उत्पन्न करने के लिए किया जाता है -

new_char_list = ~c(this is a char list containing 'single quotes')
IO.puts(new_char_list)

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

this is a char list containing 'single quotes'

शब्द सूची

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

new_word_list = ~w(foo bar bat)
IO.puts(new_word_list)

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

foobarbat

~ W sigil भी स्वीकार करता है c, s तथा a संशोधक (चार सूचियों, स्ट्रिंग्स और परमाणुओं के लिए, क्रमशः), जो परिणामी सूची के तत्वों के डेटा प्रकार को निर्दिष्ट करते हैं -

new_atom_list = ~w(foo bar bat)a
IO.puts(new_atom_list)

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

[:foo, :bar, :bat]

सिगिल में प्रक्षेप और पलायन

लोअरकेस सिगिल्स के अलावा, एलिक्सीर बचने के पात्रों और प्रक्षेप से निपटने के लिए अपरकेस सिगल्स का समर्थन करता है। हालांकि, दोनों ~ s और ~ S तार वापस आ जाएंगे, जबकि पूर्व भागने के कोड और प्रक्षेप की अनुमति देता है जबकि बाद वाला नहीं करता है। आइए इसे समझने के लिए एक उदाहरण पर विचार करें -

~s(String with escape codes \x26 #{"inter" <> "polation"})
# "String with escape codes & interpolation"
~S(String without escape codes \x26 without #{interpolation})
# "String without escape codes \\x26 without \#{interpolation}"

कस्टम सिगिल

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

defmodule CustomSigil do
   def sigil_u(string, []), do: String.upcase(string)
end

import CustomSigil

IO.puts(~u/tutorials point/)

जब हम उपरोक्त कोड चलाते हैं, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

TUTORIALS POINT

पहले हम CustomSigil नामक एक मॉड्यूल को परिभाषित करते हैं और उस मॉड्यूल के भीतर, हमने एक फ़ंक्शन बनाया जिसे sigil_u कहा जाता है। जैसा कि मौजूदा sigil space में कोई मौजूदा ~ u sigil नहीं है, हम इसका उपयोग करेंगे। _U इंगित करता है कि हम चाहते हैं कि आप tilde के बाद यू का उपयोग करें। फ़ंक्शन की परिभाषा में दो तर्क, एक इनपुट और एक सूची होनी चाहिए।

सूची की समझ अमृत में enumerables के माध्यम से पाशन के लिए सिंथेटिक चीनी है। इस अध्याय में हम पुनरावृत्ति और पीढ़ी के लिए समझ का उपयोग करेंगे।

मूल बातें

जब हमने Enumerables अध्याय में Enum मॉड्यूल को देखा, तो हम मानचित्र फ़ंक्शन में आए।

Enum.map(1..3, &(&1 * 2))

इस उदाहरण में, हम दूसरे तर्क के रूप में एक फ़ंक्शन पास करेंगे। रेंज में प्रत्येक आइटम को फंक्शन में पास किया जाएगा, और फिर एक नई सूची दी जाएगी जिसमें नए मान होंगे।

मैपिंग, फ़िल्टरिंग और ट्रांसफ़ॉर्मिंग अमृत में बहुत ही सामान्य क्रियाएं हैं और इसलिए पिछले उदाहरण के समान परिणाम प्राप्त करने का थोड़ा अलग तरीका है -

for n <- 1..3, do: n * 2

जब हम उपरोक्त कोड चलाते हैं, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

[2, 4, 6]

दूसरा उदाहरण एक बोधगम्यता है, और जैसा कि आप शायद देख सकते हैं, यह बस आप क्या उपयोग कर सकते हैं के लिए आप क्या हासिल कर सकते हैं के लिए कृत्रिम चीनी है Enum.mapसमारोह। हालांकि, प्रदर्शन के मामले में Enum मॉड्यूल से एक फ़ंक्शन पर समझ का उपयोग करने के लिए कोई वास्तविक लाभ नहीं हैं।

समझ केवल सूचियों तक सीमित नहीं है, बल्कि सभी गणनाओं के साथ उपयोग की जा सकती है।

फ़िल्टर

आप समझ के लिए एक प्रकार के गार्ड के रूप में फिल्टर के बारे में सोच सकते हैं। जब एक फ़िल्टर किया गया मान वापस आ जाता हैfalse या nilइसे अंतिम सूची से बाहर रखा गया है। आइए हम एक सीमा पर लूप करें और केवल संख्याओं की चिंता करें। हम उपयोग करेंगेis_even यह मानने के लिए कि क्या कोई मान है या नहीं, इंटेगर मॉड्यूल से कार्य करता है।

import Integer
IO.puts(for x <- 1..10, is_even(x), do: x)

जब उपरोक्त कोड चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

[2, 4, 6, 8, 10]

हम एक ही समझ में कई फिल्टर का उपयोग कर सकते हैं। एक और फ़िल्टर जोड़ें जिसे आप चाहते हैंis_even फ़िल्टर अल्पविराम द्वारा अलग किया गया।

: विकल्प में

ऊपर दिए गए उदाहरणों में, सभी समझदारों ने अपने परिणाम के रूप में सूचियां वापस कीं। हालाँकि, एक परिणाम के परिणाम को पास करके विभिन्न डेटा संरचनाओं में डाला जा सकता है:into समझ का विकल्प।

उदाहरण के लिए, ए bitstring एक स्ट्रिंग में सभी स्थानों को आसानी से हटाने के लिए जनरेटर का उपयोग इस विकल्प के साथ किया जा सकता है -

IO.puts(for <<c <- " hello world ">>, c != ?\s, into: "", do: <<c>>)

जब उपरोक्त कोड चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

helloworld

उपरोक्त कोड का उपयोग करके स्ट्रिंग से सभी रिक्त स्थान निकालता है c != ?\s फ़िल्टर करें और फिर उपयोग में: विकल्प में, यह सभी लौटे वर्णों को एक स्ट्रिंग में रखता है।

अमृत ​​एक गतिशील रूप से टाइप की जाने वाली भाषा है, इसलिए अमृत में सभी प्रकार के रनटाइम से अनुमान लगाया जाता है। फिर भी, एलिक्सिर टाइपसेक के साथ आता है, जो एक संकेतन के लिए उपयोग किया जाता हैdeclaring custom data types and declaring typed function signatures (specifications)

समारोह विनिर्देशों (चश्मा)

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

round(number) :: integer

उपर्युक्त विवरण से तात्पर्य है कि बाईं ओर का कार्य इस तर्क के रूप में लेता है कि कोष्ठक में निर्दिष्ट क्या है और रिटर्न क्या है ::, अर्थात, पूर्णांक के दाईं ओर। समारोह चश्मा के साथ लिखे गए हैं@specनिर्देश, फ़ंक्शन परिभाषा से ठीक पहले रखा गया। दौर समारोह के रूप में लिखा जा सकता है -

@spec round(number) :: integer
def round(number), do: # Function implementation
...

टाइपसेक्स जटिल प्रकारों का भी समर्थन करते हैं, उदाहरण के लिए, यदि आप पूर्णांकों की सूची वापस करना चाहते हैं, तो आप उपयोग कर सकते हैं [Integer]

कस्टम प्रकार

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

defmodule FunnyCalculator do
   @type number_with_joke :: {number, String.t}

   @spec add(number, number) :: number_with_joke
   def add(x, y), do: {x + y, "You need a calculator to do that?"}

   @spec multiply(number, number) :: number_with_joke
   def multiply(x, y), do: {x * y, "It is like addition on steroids."}
end

{result, comment} = FunnyCalculator.add(10, 20)
IO.puts(result)
IO.puts(comment)

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

30
You need a calculator to do that?

NOTE - @ प्रकार के माध्यम से परिभाषित कस्टम प्रकारों को उनके द्वारा परिभाषित मॉड्यूल के बाहर निर्यात और उपलब्ध किया जाता है। यदि आप कस्टम प्रकार को निजी रखना चाहते हैं, तो आप मॉड्यूल का उपयोग कर सकते हैं @typep के बजाय निर्देश @type

Elixir (और Erlang) में व्यवहार एक घटक के जेनेरिक भाग (जो व्यवहार मॉड्यूल बन जाता है) को विशिष्ट भाग से अलग करने और अमूर्त करने का एक तरीका है (जो कॉलबैक मॉड्यूल बन जाता है)। व्यवहार के लिए एक रास्ता प्रदान करते हैं -

  • मॉड्यूल द्वारा कार्यान्वित किए जाने वाले कार्यों का एक सेट निर्धारित करें।
  • सुनिश्चित करें कि एक मॉड्यूल उस सेट में सभी कार्यों को लागू करता है।

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

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

आइए हम अपने व्यवहार को बनाने के लिए एक उदाहरण पर विचार करें और फिर एक मॉड्यूल बनाने के लिए इस सामान्य व्यवहार का उपयोग करें। हम ऐसे व्यवहार को परिभाषित करेंगे जो विभिन्न भाषाओं में लोगों को नमस्ते और अलविदा कहे।

defmodule GreetBehaviour do
   @callback say_hello(name :: string) :: nil
   @callback say_bye(name :: string) :: nil
end

@callbackनिर्देश का उपयोग उन कार्यों को सूचीबद्ध करने के लिए किया जाता है जिन्हें अपनाने के लिए मॉड्यूल को परिभाषित करना होगा। यह भी नहीं निर्दिष्ट करता है। तर्क, उनके प्रकार और उनके रिटर्न मान

व्यवहार को अपनाना

हमने एक व्यवहार को सफलतापूर्वक परिभाषित किया है। अब हम इसे कई मॉड्यूल में अपनाएंगे और लागू करेंगे। आइए अंग्रेजी और स्पेनिश में इस व्यवहार को लागू करने वाले दो मॉड्यूल बनाएं।

defmodule GreetBehaviour do
   @callback say_hello(name :: string) :: nil
   @callback say_bye(name :: string) :: nil
end

defmodule EnglishGreet do
   @behaviour GreetBehaviour
   def say_hello(name), do: IO.puts("Hello " <> name)
   def say_bye(name), do: IO.puts("Goodbye, " <> name)
end

defmodule SpanishGreet do
   @behaviour GreetBehaviour
   def say_hello(name), do: IO.puts("Hola " <> name)
   def say_bye(name), do: IO.puts("Adios " <> name)
end

EnglishGreet.say_hello("Ayush")
EnglishGreet.say_bye("Ayush")
SpanishGreet.say_hello("Ayush")
SpanishGreet.say_bye("Ayush")

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Hello Ayush
Goodbye, Ayush
Hola Ayush
Adios Ayush

जैसा कि आप पहले ही देख चुके हैं, हम व्यवहार का उपयोग करते हुए एक व्यवहार अपनाते हैं @behaviourमॉड्यूल में निर्देश। हमें सभी चाइल्ड मॉड्यूल के लिए व्यवहार में लागू सभी कार्यों को परिभाषित करना होगा । इसे मोटे तौर पर OOP भाषाओं में इंटरफेस के समकक्ष माना जा सकता है।

अमृत ​​में तीन त्रुटि तंत्र हैं: त्रुटियां, फेंकता और बाहर निकलता है। आइए हम प्रत्येक तंत्र को विस्तार से देखें।

त्रुटि

जब कोड में असाधारण चीजें होती हैं तो त्रुटियों (या अपवादों) का उपयोग किया जाता है। एक स्ट्रिंग में एक संख्या जोड़ने की कोशिश करके एक नमूना त्रुटि प्राप्त की जा सकती है -

IO.puts(1 + "Hello")

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित त्रुटि पैदा करता है -

** (ArithmeticError) bad argument in arithmetic expression
   :erlang.+(1, "Hello")

यह एक इनबिल्ट त्रुटि का नमूना था।

उठी हुई त्रुटियाँ

हम कर सकते हैं raiseफ़ंक्शंस का उपयोग करते हुए त्रुटियाँ। आइए हम इसे समझने के लिए एक उदाहरण पर विचार करें -

#Runtime Error with just a message
raise "oops"  # ** (RuntimeError) oops

अन्य त्रुटियों को त्रुटि नाम और कीवर्ड तर्कों की एक सूची को बढ़ाने / 2 के साथ उठाया जा सकता है

#Other error type with a message
raise ArgumentError, message: "invalid argument foo"

आप अपनी स्वयं की त्रुटियों को भी परिभाषित कर सकते हैं और उन्हें बढ़ा सकते हैं। निम्नलिखित उदाहरण पर विचार करें -

defmodule MyError do
   defexception message: "default message"
end

raise MyError  # Raises error with default message
raise MyError, message: "custom message"  # Raises error with custom message

बचाव की त्रुटियाँ

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

err = try do
   raise "oops"
rescue
   e in RuntimeError -> e
end

IO.puts(err.message)

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

oops

हमने पैटर्न मिलान का उपयोग करके बचाव वक्तव्य में त्रुटियों को संभाला है। यदि हमारे पास त्रुटि का कोई उपयोग नहीं है, और बस इसे पहचान के प्रयोजनों के लिए उपयोग करना चाहते हैं, तो हम फॉर्म का उपयोग भी कर सकते हैं -

err = try do
   1 + "Hello"
rescue
   RuntimeError -> "You've got a runtime error!"
   ArithmeticError -> "You've got a Argument error!"
end

IO.puts(err)

कार्यक्रम से ऊपर चलने पर, यह निम्नलिखित परिणाम उत्पन्न करता है -

You've got a Argument error!

NOTE- अमृत मानक पुस्तकालय में अधिकांश कार्य दो बार कार्यान्वित किए जाते हैं, एक बार ट्यूपल्स को लौटाने और दूसरी बार त्रुटियों को बढ़ाते हुए। उदाहरण के लिए,File.read और यह File.read!कार्य करता है। यदि फ़ाइल सफलतापूर्वक पढ़ी गई थी और यदि कोई त्रुटि सामने आई थी, तो पहले ने एक टपल लौटाया था, इस टपल का उपयोग त्रुटि का कारण बताने के लिए किया गया था। यदि कोई त्रुटि सामने आई, तो दूसरे ने एक त्रुटि उठाई।

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

फेंकता

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

पुस्तकालयों के साथ हस्तक्षेप करते हुए उदाहरण अभ्यास में काफी असामान्य हैं। उदाहरण के लिए, अब मान लेते हैं कि Enum मॉड्यूल ने कोई मान प्राप्त करने के लिए कोई API प्रदान नहीं किया है और हमें संख्याओं की सूची में 13 के पहले एकाधिक को खोजने की आवश्यकता है -

val = try do
   Enum.each 20..100, fn(x) ->
      if rem(x, 13) == 0, do: throw(x)
   end
   "Got nothing"
catch
   x -> "Got #{x}"
end

IO.puts(val)

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Got 26

बाहर जाएं

जब एक प्रक्रिया "प्राकृतिक कारणों" की मृत्यु हो जाती है (उदाहरण के लिए, बिना किसी अपवाद के), तो यह एक निकास संकेत भेजता है। एक प्रक्रिया भी स्पष्ट रूप से एक निकास संकेत भेजकर मर सकती है। आइए हम निम्नलिखित उदाहरण पर विचार करें -

spawn_link fn -> exit(1) end

ऊपर के उदाहरण में, लिंक की गई प्रक्रिया 1. के मूल्य के साथ एक एक्जिट सिग्नल भेजकर मर गई। ध्यान दें कि बाहर निकलने को भी कोशिश / पकड़ का उपयोग करके "पकड़ा" जा सकता है। उदाहरण के लिए -

val = try do
   exit "I am exiting"
catch
   :exit, _ -> "not really"
end

IO.puts(val)

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

not really

उपरांत

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

{:ok, file} = File.open "sample", [:utf8, :write]
try do
   IO.write file, "olá"
   raise "oops, something went wrong"
after
   File.close(file)
end

जब हम इस कार्यक्रम को चलाते हैं, तो यह हमें एक त्रुटि देगा। लेकिन वोafter यह सुनिश्चित करेगा कि फ़ाइल डिस्क्रिप्टर ऐसी किसी भी घटना पर बंद हो।

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

उद्धरण

इससे पहले कि हम मैक्रों के बारे में बात करना शुरू करें, आइए हम पहले एलिक्जिर इंटर्नल्स को देखें। एक अमृत कार्यक्रम को अपने स्वयं के डेटा संरचनाओं द्वारा दर्शाया जा सकता है। एलिक्जिर प्रोग्राम का बिल्डिंग ब्लॉक तीन तत्वों वाला एक ट्यूल है। उदाहरण के लिए, फ़ंक्शन कॉल योग (1, 2, 3) आंतरिक रूप से दर्शाया गया है -

{:sum, [], [1, 2, 3]}

पहला तत्व फ़ंक्शन नाम है, दूसरा एक कीवर्ड सूची है जिसमें मेटाडेटा है और तीसरा तर्क सूची है। यदि आप निम्नलिखित लिखते हैं तो आप इसे iex शेल में आउटपुट के रूप में प्राप्त कर सकते हैं -

quote do: sum(1, 2, 3)

ऑपरेटर्स को ऐसे टुपल्स के रूप में भी दर्शाया जाता है। चर को भी ऐसे ट्रिपल का उपयोग करके दर्शाया जाता है, सिवाय इसके कि अंतिम तत्व एक सूची के बजाय एक परमाणु है। अधिक जटिल अभिव्यक्तियों को उद्धृत करते हुए, हम देख सकते हैं कि कोड को ऐसे टुपल्स में दर्शाया गया है, जो अक्सर एक पेड़ से मिलते-जुलते संरचना में एक दूसरे के अंदर रहते हैं। कई भाषाएं ऐसे अभ्यावेदन कहती हैंAbstract Syntax Tree (AST)। अमृत ​​इन उद्धृत भावों को कहता है।

गंदें शब्द बोलना

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

num = 25

quote do: sum(15, num)

quote do: sum(15, unquote(num))

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

{:sum, [], [15, {:num, [], Elixir}]}
{:sum, [], [15, 25]}

उद्धरण अभिव्यक्ति के लिए उदाहरण में, यह स्वचालित रूप से 25 के साथ संख्या को प्रतिस्थापित नहीं करता है। हमें एएसटी को संशोधित करने के लिए इस चर को प्राप्त करने की आवश्यकता है।

मैक्रो

इसलिए अब जब हम उद्धरण और अयोग्यता से परिचित हैं, तो हम मैक्रोज़ का उपयोग करके एलिक्सिर में मेटाप्रोग्रामिंग का पता लगा सकते हैं।

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

आइए हम एक मैक्रो के रूप में लागू करते हैं। हम मैक्रो को परिभाषित करके शुरू करेंगेdefmacroमैक्रो। याद रखें कि हमारे मैक्रो को उद्धृत अभिव्यक्ति वापस करने की आवश्यकता है।

defmodule OurMacro do
   defmacro unless(expr, do: block) do
      quote do
         if !unquote(expr), do: unquote(block)
      end
   end
end

require OurMacro

OurMacro.unless true, do: IO.puts "True Expression"

OurMacro.unless false, do: IO.puts "False expression"

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

False expression

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

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

Elixir Erlang पुस्तकालयों के साथ उत्कृष्ट अंतर प्रदान करता है। आइए संक्षिप्त में कुछ पुस्तकालयों पर चर्चा करते हैं।

बाइनरी मॉड्यूल

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

# UTF-8
IO.puts(String.to_char_list("Ø"))

# binary
IO.puts(:binary.bin_to_list "Ø")

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

[216]
[195, 152]

उपरोक्त उदाहरण अंतर दिखाता है; स्ट्रिंग मॉड्यूल UTF-8 कोडपॉइंट्स लौटाता है, जबकि: कच्चे डेटा बाइट्स के साथ द्विआधारी सौदे।

क्रिप्टो मॉड्यूल

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

IO.puts(Base.encode16(:crypto.hash(:sha256, "Elixir")))

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

3315715A7A3AD57428298676C5AE465DADA38D951BDFAC9348A8A31E9C7401CB

डिग्राफ मॉड्यूल

डिग्राफ मॉड्यूल में वर्टिकल और किनारों से निर्मित निर्देशित ग्राफ से निपटने के लिए कार्य शामिल हैं। ग्राफ़ का निर्माण करने के बाद, वहाँ के एल्गोरिदम को खोजने में मदद मिलेगी, उदाहरण के लिए, दो कोने के बीच सबसे छोटा रास्ता, या ग्राफ़ में छोरों। ध्यान दें कि फ़ंक्शनin :digraph अतिरिक्त संरचना को साइड इफेक्ट के रूप में बदल देते हैं, जबकि जोड़े गए कोने या किनारों को वापस करते हैं।

digraph = :digraph.new()
coords = [{0.0, 0.0}, {1.0, 0.0}, {1.0, 1.0}]
[v0, v1, v2] = (for c <- coords, do: :digraph.add_vertex(digraph, c))
:digraph.add_edge(digraph, v0, v1)
:digraph.add_edge(digraph, v1, v2)
for point <- :digraph.get_short_path(digraph, v0, v2) do 
   {x, y} = point
   IO.puts("#{x}, #{y}")
end

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

0.0, 0.0
1.0, 0.0
1.0, 1.0

मठ का मॉड्यूल

गणित मॉड्यूल में त्रिकोणमिति, घातीय और लघुगणक कार्यों को कवर करने वाले सामान्य गणितीय कार्य शामिल हैं। गणित मॉड्यूल कैसे काम करता है, इसे समझने के लिए हम निम्नलिखित उदाहरण पर विचार करें -

# Value of pi
IO.puts(:math.pi())

# Logarithm
IO.puts(:math.log(7.694785265142018e23))

# Exponentiation
IO.puts(:math.exp(55.0))

#...

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

3.141592653589793
55.0
7.694785265142018e23

कतार मॉड्यूल

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

q = :queue.new
q = :queue.in("A", q)
q = :queue.in("B", q)
{{:value, val}, q} = :queue.out(q)
IO.puts(val)
{{:value, val}, q} = :queue.out(q)
IO.puts(val)

जब उपरोक्त कार्यक्रम चलाया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

A
B