अमृत - प्रक्रियाएँ
अमृत में, सभी कोड प्रक्रियाओं के अंदर चलते हैं। प्रक्रियाओं को एक दूसरे से अलग किया जाता है, एक दूसरे के समवर्ती चलाते हैं और संदेश गुजरने के माध्यम से संवाद करते हैं। अमृत की प्रक्रियाओं को ऑपरेटिंग सिस्टम प्रक्रियाओं के साथ भ्रमित नहीं होना चाहिए। एलिक्सिर में प्रक्रियाएं मेमोरी और सीपीयू (कई अन्य प्रोग्रामिंग भाषाओं में धागे के विपरीत) के मामले में बेहद हल्की हैं। इस वजह से, दसियों या यहां तक कि सैकड़ों हजारों प्रक्रियाओं का एक साथ चलना असामान्य नहीं है।
इस अध्याय में, हम नई प्रक्रियाओं को बनाने के लिए बुनियादी निर्माणों के बारे में जानेंगे, साथ ही विभिन्न प्रक्रियाओं के बीच संदेश भेजना और प्राप्त करना।
स्पॉन समारोह
एक नई प्रक्रिया बनाने का सबसे आसान तरीका है, का उपयोग करना 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"