लुआ - कोराटाइन
परिचय
Coroutines प्रकृति में सहयोगी हैं, जो दो या दो से अधिक तरीकों को नियंत्रित तरीके से निष्पादित करने की अनुमति देता है। किसी भी समय, केवल एक coroutine चलता है और यह coroutine केवल तब ही इसके निष्पादन को निलंबित कर देता है, जब यह स्पष्ट रूप से निलंबित करने का अनुरोध करता है।
उपरोक्त परिभाषा अस्पष्ट लग सकती है। चलिए मान लेते हैं कि हमारे पास दो विधियाँ हैं, एक मुख्य कार्यक्रम विधि और एक कोरटाइन। जब हम फिर से शुरू होने वाले फ़ंक्शन का उपयोग करके एक कोरटाइन कहते हैं, तो यह निष्पादित करना शुरू कर देता है और जब हम उपज फ़ंक्शन कहते हैं, तो यह निष्पादन को निलंबित कर देता है। फिर से वही कोरआउट एक और रेज़मै फ़ंक्शन कॉल के साथ निष्पादित करना जारी रख सकता है जहां से इसे निलंबित कर दिया गया था। यह प्रक्रिया कोरटाइन के निष्पादन के अंत तक जारी रह सकती है।
Coroutines में उपलब्ध कार्य
निम्न तालिका लुआ में कॉरटाइन के लिए उपलब्ध सभी कार्यों और उनके संबंधित उपयोग को सूचीबद्ध करती है।
अनु क्रमांक। | विधि और उद्देश्य |
---|---|
1 | coroutine.create (f) एक फ़ंक्शन एफ के साथ एक नया कोरआउट बनाता है और "थ्रेड" प्रकार की एक वस्तु लौटाता है। |
2 | coroutine.resume (co [, val1, ...]) कोरटाइन सह को फिर से शुरू करता है और यदि कोई हो तो मापदंडों को पारित करता है। यह ऑपरेशन की स्थिति और वैकल्पिक अन्य रिटर्न मान लौटाता है। |
3 | coroutine.running () यदि मुख्य थ्रेड में कहा जाता है तो चल रहे कोरआउट या नील को लौटाता है। |
4 | coroutine.status (co) कोरटाइन की स्थिति के आधार पर चलने, सामान्य, निलंबित या मृत होने से एक मान लौटाता है। |
5 | coroutine.wrap (f) Coroutine.create की तरह, coroutine.wrap फ़ंक्शन भी एक coroutine बनाता है, लेकिन स्वयं coroutine को वापस करने के बजाय, यह एक फ़ंक्शन देता है, जिसे जब बुलाया जाता है, तो coroutine फिर से शुरू होता है। |
6 | coroutine.yield (...) रनिंग कॉरआउट को निलंबित करता है। इस पद्धति को दिया गया पैरामीटर फिर से शुरू होने वाले फ़ंक्शन के लिए अतिरिक्त रिटर्न मान के रूप में कार्य करता है। |
उदाहरण
आइए कोरटाइन की अवधारणा को समझने के लिए एक उदाहरण देखें।
co = coroutine.create(function (value1,value2)
local tempvar3 = 10
print("coroutine section 1", value1, value2, tempvar3)
local tempvar1 = coroutine.yield(value1+1,value2+1)
tempvar3 = tempvar3 + value1
print("coroutine section 2",tempvar1 ,tempvar2, tempvar3)
local tempvar1, tempvar2= coroutine.yield(value1+value2, value1-value2)
tempvar3 = tempvar3 + value1
print("coroutine section 3",tempvar1,tempvar2, tempvar3)
return value2, "end"
end)
print("main", coroutine.resume(co, 3, 2))
print("main", coroutine.resume(co, 12,14))
print("main", coroutine.resume(co, 5, 6))
print("main", coroutine.resume(co, 10, 20))
जब हम उपरोक्त कार्यक्रम चलाते हैं, तो हमें निम्न आउटपुट मिलेगा।
coroutine section 1 3 2 10
main true 4 3
coroutine section 2 12 nil 13
main true 5 1
coroutine section 3 5 6 16
main true 2 end
main false cannot resume dead coroutine
उपरोक्त उदाहरण क्या करता है?
जैसा कि पहले उल्लेख किया गया है, हम ऑपरेशन को रोकने के लिए फिर से शुरू समारोह का उपयोग करते हैं और ऑपरेशन को रोकने के लिए फ़ंक्शन का उत्पादन करते हैं। इसके अलावा, आप देख सकते हैं कि कोरटाइन के फिर से शुरू फ़ंक्शन द्वारा कई रिटर्न मान प्राप्त किए गए हैं।
सबसे पहले, हम एक coroutine बनाते हैं और इसे एक चर नाम सह को असाइन करते हैं और coroutine इसके मापदंडों के रूप में दो चर में लेता है।
जब हम पहले रेज़्यूमे फ़ंक्शन को कहते हैं, तो मान 3 और 2 को अस्थायी चर मान 1 और मान 2 में कॉरटीन के अंत तक बनाए रखा जाता है।
आपको यह समझने के लिए, हमने एक tempvar3 का उपयोग किया है, जो कि शुरू में 10 है और इसे coroutines की बाद की कॉल द्वारा 13 और 16 तक अद्यतन किया जाता है क्योंकि value1 को coroutine के निष्पादन के दौरान 3 के रूप में बनाए रखा जाता है।
पहला coroutine.yield दो मानों 4 और 3 को फिर से शुरू करने वाले फ़ंक्शन पर लौटाता है, जो हमें उपज कथन में इनपुट 3 और 2 को अपडेट करके मिलता है। यह कोरटाइन निष्पादन की सही / गलत स्थिति को भी प्राप्त करता है।
कोरआउट के बारे में एक और बात यह है कि उपरोक्त उदाहरण में फिर से शुरू कॉल के अगले पैरामेट्स का ध्यान कैसे रखा गया है; आप देख सकते हैं कि चर coroutine.yield अगला कॉल परम प्राप्त करता है जो मौजूदा परम मूल्यों की अवधारण के साथ नया ऑपरेशन करने का एक शक्तिशाली तरीका प्रदान करता है।
अंत में, एक बार जब कोरटाइन में सभी कथनों को निष्पादित कर दिया जाता है, तो बाद की कॉलें झूठी हो जाएंगी और प्रतिक्रिया के रूप में "मृत कोआउटिन को फिर से शुरू नहीं कर सकती हैं"।
एक और Coroutine उदाहरण
आइए हम एक साधारण कोरआउट को देखें जो उपज फ़ंक्शन और फिर से शुरू होने वाले फ़ंक्शन की सहायता से 1 से 5 तक की संख्या देता है। यदि यह उपलब्ध नहीं है तो कोरटाइन बनाता है या फिर मौजूदा कोरटाइन को फिर से शुरू करता है।
function getNumber()
local function getNumberHelper()
co = coroutine.create(function ()
coroutine.yield(1)
coroutine.yield(2)
coroutine.yield(3)
coroutine.yield(4)
coroutine.yield(5)
end)
return co
end
if(numberHelper) then
status, number = coroutine.resume(numberHelper);
if coroutine.status(numberHelper) == "dead" then
numberHelper = getNumberHelper()
status, number = coroutine.resume(numberHelper);
end
return number
else
numberHelper = getNumberHelper()
status, number = coroutine.resume(numberHelper);
return number
end
end
for index = 1, 10 do
print(index, getNumber())
end
जब हम उपरोक्त कार्यक्रम चलाते हैं, तो हमें निम्न आउटपुट मिलेगा।
1 1
2 2
3 3
4 4
5 5
6 1
7 2
8 3
9 4
10 5
बहुभाषाविद भाषाओं के धागों के साथ अक्सर तुलना की जाती है, लेकिन हमें यह समझने की आवश्यकता है कि कोरटाइन्स में धागे की समान विशेषताएं होती हैं, लेकिन वे एक समय में केवल एक ही निष्पादित करते हैं और कभी भी समवर्ती रूप से निष्पादित नहीं करते हैं।
हम अस्थायी रूप से कुछ जानकारी को बनाए रखने के प्रावधान के साथ जरूरतों को पूरा करने के लिए कार्यक्रम निष्पादन अनुक्रम को नियंत्रित करते हैं। कोराउटाइन के साथ वैश्विक चर का उपयोग करना, कोरटाइन को और भी अधिक लचीलापन प्रदान करता है।