अमृत ​​- प्रक्रियाएँ

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

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

स्पॉन समारोह

एक नई प्रक्रिया बनाने का सबसे आसान तरीका है, का उपयोग करना 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"