रूबी - मल्टीथ्रेडिंग
पारंपरिक कार्यक्रमों के निष्पादन के एक ही सूत्र में कथन या निर्देश शामिल होते हैं जो प्रोग्राम को समाप्त होने तक क्रमिक रूप से निष्पादित करते हैं।
एक बहुस्तरीय कार्यक्रम में निष्पादन के एक से अधिक धागे होते हैं। प्रत्येक थ्रेड के भीतर, स्टेटमेंट्स को क्रमिक रूप से निष्पादित किया जाता है, लेकिन थ्रेड्स को मल्टीकोर सीपीयू पर समानांतर में निष्पादित किया जा सकता है, उदाहरण के लिए। अक्सर एक ही सीपीयू मशीन पर, कई थ्रेड्स वास्तव में समानांतर में निष्पादित नहीं किए जाते हैं, लेकिन थ्रेड्स के निष्पादन को इंटरलायड करके समांतरवाद का अनुकरण किया जाता है।
रूबी थ्रेड वर्ग के साथ बहु-थ्रेडेड प्रोग्राम लिखना आसान बनाता है । रूबी धागे अपने कोड में संगामिति प्राप्त करने के लिए एक हल्का और कुशल तरीका है।
रूबी थ्रेड बनाना
एक नया धागा शुरू करने के लिए, बस एक ब्लॉक को Thread.new पर कॉल करें । ब्लॉक में कोड निष्पादित करने के लिए एक नया थ्रेड बनाया जाएगा, और थ्रेड थ्रेड से तुरंत वापस आएगा और अगले स्टेटमेंट के साथ निष्पादन फिर से शुरू करेगा -
# Thread #1 is running here
Thread.new {
# Thread #2 runs this code
}
# Thread #1 runs this code
उदाहरण
यहां एक उदाहरण है, जो दिखाता है कि हम बहु-थ्रेडेड रूबी प्रोग्राम का उपयोग कैसे कर सकते हैं।
#!/usr/bin/ruby
def func1
i = 0
while i<=2
puts "func1 at: #{Time.now}"
sleep(2)
i = i+1
end
end
def func2
j = 0
while j<=2
puts "func2 at: #{Time.now}"
sleep(1)
j = j+1
end
end
puts "Started At #{Time.now}"
t1 = Thread.new{func1()}
t2 = Thread.new{func2()}
t1.join
t2.join
puts "End at #{Time.now}"
यह निम्नलिखित परिणाम देगा -
Started At Wed May 14 08:21:54 -0700 2008
func1 at: Wed May 14 08:21:54 -0700 2008
func2 at: Wed May 14 08:21:54 -0700 2008
func2 at: Wed May 14 08:21:55 -0700 2008
func1 at: Wed May 14 08:21:56 -0700 2008
func2 at: Wed May 14 08:21:56 -0700 2008
func1 at: Wed May 14 08:21:58 -0700 2008
End at Wed May 14 08:22:00 -0700 2008
धागा जीवनचक्र
थ्रेड.न्यू के साथ एक नया थ्रेड बनाया जाता है । आप समानार्थी शब्द Thread.start और Thread.fork का भी उपयोग कर सकते हैं ।
इसे बनाने के बाद एक धागा शुरू करने की आवश्यकता नहीं है, यह सीपीयू संसाधन उपलब्ध होने पर स्वचालित रूप से चलना शुरू कर देता है।
थ्रेड क्लास चलने के दौरान थ्रेड को क्वेरी और हेरफेर करने के लिए कई तरीकों को परिभाषित करता है। थ्रेड में एक थ्रेड कोड को कॉल से संबंधित चलाता है। फिर वह चलना बंद कर देता है।
उस ब्लॉक में अंतिम अभिव्यक्ति का मूल्य थ्रेड का मूल्य है, और थ्रेड ऑब्जेक्ट के मूल्य विधि को कॉल करके प्राप्त किया जा सकता है । यदि थ्रेड पूर्ण होने के लिए चला गया है, तो मान थ्रेड का मान तुरंत लौटाता है। अन्यथा, मान विधि ब्लॉक हो जाती है और तब तक वापस नहीं आती जब तक कि धागा पूरा नहीं हो जाता।
क्लास विधि Thread.current वर्तमान थ्रेड का प्रतिनिधित्व करने वाली थ्रेड ऑब्जेक्ट देता है। इससे धागे खुद को हेरफेर करने की अनुमति देते हैं। थ्रेडमेन को क्लास विधि मुख्य ऑब्जेक्ट का प्रतिनिधित्व करने वाले थ्रेड ऑब्जेक्ट देता है। यह निष्पादन का प्रारंभिक धागा है जो रूबी कार्यक्रम शुरू होने पर शुरू हुआ था।
आप उस थ्रेड के Thread.join विधि को कॉल करके किसी विशेष थ्रेड के समाप्त होने की प्रतीक्षा कर सकते हैं । कॉलिंग थ्रेड तब तक ब्लॉक रहेगा जब तक कि दिया गया धागा समाप्त नहीं हो जाता।
सूत्र और अपवाद
यदि कोई अपवाद मुख्य थ्रेड में उठाया गया है, और कहीं भी संभाला नहीं गया है, रूबी दुभाषिया एक संदेश प्रिंट करता है और बाहर निकलता है। थ्रेड्स में, मुख्य थ्रेड के अलावा, अनहेल्डेड अपवाद के कारण थ्रेड चलना बंद हो जाता है।
अगर एक धागा t एक अखंड अपवाद के कारण बाहर निकलता है, और एक और धागा sकॉल t.join या t.value, तो अपवाद है कि में हुईt धागे में उठाया जाता है s।
यदि Thread.abort_on_exception है झूठे , डिफ़ॉल्ट हालत, एक बिना क्रिया का अपवाद केवल वर्तमान धागा मारता है और सभी बाकी चलाने के लिए जारी।
यदि आप किसी थ्रेड में किसी भी अनहेल्दी अपवाद को चाहेंगे तो दुभाषिया से बाहर निकलने का कारण होगा, क्लास विधि Thread.abort_on_exception को सही पर सेट करें ।
t = Thread.new { ... }
t.abort_on_exception = true
धागा चर
एक धागा आम तौर पर किसी भी चर का उपयोग कर सकता है जो कि गुंजाइश तब होती है जब धागा बनाया जाता है। किसी थ्रेड के ब्लॉक के लिए चर स्थानीय थ्रेड के लिए स्थानीय होते हैं, और साझा नहीं किए जाते हैं।
थ्रेड क्लास में एक विशेष सुविधा है जो थ्रेड-लोकल वैरिएबल को नाम से बनाने और एक्सेस करने की सुविधा देती है। आप बस थ्रेड ऑब्जेक्ट का इलाज करते हैं जैसे कि यह एक हैश था, [] = का उपयोग करने वाले तत्वों को लिख रहा है और उन्हें [] का उपयोग करके वापस पढ़ रहा है।
इस उदाहरण में, प्रत्येक थ्रेड कुंजी गणना के साथ थ्रेडलोक वैरिएबल में चर संख्या के वर्तमान मान को रिकॉर्ड करता है ।
#!/usr/bin/ruby
count = 0
arr = []
10.times do |i|
arr[i] = Thread.new {
sleep(rand(0)/10.0)
Thread.current["mycount"] = count
count += 1
}
end
arr.each {|t| t.join; print t["mycount"], ", " }
puts "count = #{count}"
यह निम्न परिणाम उत्पन्न करता है -
8, 0, 3, 7, 2, 1, 6, 5, 4, 9, count = 10
मुख्य धागा खत्म होने का इंतजार करता है और फिर प्रत्येक द्वारा कब्जा की गई गणना के मूल्य को प्रिंट करता है ।
सूत्र प्राथमिकताएँ
थ्रेड शेड्यूलिंग को प्रभावित करने वाला पहला कारक थ्रेड प्राथमिकता है: उच्च-प्राथमिकता वाले थ्रेड्स को कम-प्राथमिकता वाले थ्रेड्स से पहले शेड्यूल किया जाता है। अधिक सटीक रूप से, एक थ्रेड को केवल CPU समय मिलेगा यदि कोई उच्च प्राथमिकता वाले थ्रेड्स चलाने के लिए इंतजार नहीं कर रहे हैं।
आप रूबी थ्रेड ऑब्जेक्ट की प्राथमिकता को प्राथमिकता = और प्राथमिकता के साथ सेट और क्वेरी कर सकते हैं । एक नया बनाया गया धागा उसी प्राथमिकता पर शुरू होता है जिस धागे ने उसे बनाया है। मुख्य धागा प्राथमिकता 0 पर शुरू होता है।
चलने से पहले किसी थ्रेड की प्राथमिकता निर्धारित करने का कोई तरीका नहीं है। हालाँकि, थ्रेड अपनी पहली प्राथमिकता के रूप में अपनी प्राथमिकता को बढ़ा या घटा सकता है।
धागा बहिष्कार
यदि दो थ्रेड्स एक ही डेटा तक पहुंच साझा करते हैं, और कम से कम एक थ्रेड उस डेटा को संशोधित करता है, तो आपको यह सुनिश्चित करने के लिए विशेष ध्यान रखना होगा कि कोई भी धागा डेटा को कभी भी असंगत स्थिति में न देख सके। इसे थ्रेड एक्सक्लूजन कहा जाता है ।
Mutexएक ऐसा वर्ग है जो कुछ साझा संसाधनों के लिए पारस्परिक रूप से अनन्य पहुंच के लिए एक साधारण सेमाफोर लॉक को लागू करता है। यही है, केवल एक धागा किसी निश्चित समय पर लॉक को पकड़ सकता है। अन्य थ्रेड्स लॉक उपलब्ध होने के लिए लाइन में प्रतीक्षा करने का विकल्प चुन सकते हैं, या बस एक तत्काल त्रुटि प्राप्त करने के लिए चुन सकते हैं जो दर्शाता है कि लॉक उपलब्ध नहीं है।
म्यूटेक्स के नियंत्रण में साझा डेटा तक सभी पहुंच रखकर , हम स्थिरता और परमाणु संचालन सुनिश्चित करते हैं। आइए उदाहरणों के लिए प्रयास करें, पहला बिना म्लेक्ष के साथ और दूसरा मटैक्स के साथ -
मटेक्स के बिना उदाहरण
#!/usr/bin/ruby
require 'thread'
count1 = count2 = 0
difference = 0
counter = Thread.new do
loop do
count1 += 1
count2 += 1
end
end
spy = Thread.new do
loop do
difference += (count1 - count2).abs
end
end
sleep 1
puts "count1 : #{count1}"
puts "count2 : #{count2}"
puts "difference : #{difference}"
यह निम्नलिखित परिणाम का उत्पादन करेगा -
count1 : 1583766
count2 : 1583766
difference : 0
#!/usr/bin/ruby
require 'thread'
mutex = Mutex.new
count1 = count2 = 0
difference = 0
counter = Thread.new do
loop do
mutex.synchronize do
count1 += 1
count2 += 1
end
end
end
spy = Thread.new do
loop do
mutex.synchronize do
difference += (count1 - count2).abs
end
end
end
sleep 1
mutex.lock
puts "count1 : #{count1}"
puts "count2 : #{count2}"
puts "difference : #{difference}"
यह निम्नलिखित परिणाम का उत्पादन करेगा -
count1 : 696591
count2 : 696591
difference : 0
गतिरोध से निपटने
जब हम थ्रेड बहिष्करण के लिए म्यूटेक्स वस्तुओं का उपयोग करना शुरू करते हैं तो हमें गतिरोध से बचने के लिए सावधान रहना चाहिए । डेडलॉक वह स्थिति है जो तब होती है जब सभी थ्रेड दूसरे थ्रेड द्वारा रखे गए संसाधन को प्राप्त करने की प्रतीक्षा कर रहे होते हैं। क्योंकि सभी धागे अवरुद्ध हैं, वे अपने द्वारा पकड़े गए ताले को जारी नहीं कर सकते। और क्योंकि वे ताले को मुक्त नहीं कर सकते हैं, कोई अन्य धागा उन तालों को प्राप्त नहीं कर सकता है।
यह वह जगह है जहाँ हालत चर चित्र में आते हैं। एक कंडीशन वैरिएबल केवल एक सेमाफोर है जो एक संसाधन के साथ जुड़ा हुआ है और एक विशेष म्यूटेक्स के संरक्षण में उपयोग किया जाता है । जब आपको एक संसाधन की आवश्यकता होती है जो अनुपलब्ध है, तो आप एक स्थिति चर पर प्रतीक्षा करते हैं। यह क्रिया संबंधित म्यूटेक्स पर लॉक जारी करती है । जब कुछ अन्य थ्रेड संकेत देते हैं कि संसाधन उपलब्ध है, मूल थ्रेड प्रतीक्षा से बाहर आता है और साथ ही महत्वपूर्ण क्षेत्र पर लॉक को फिर से प्राप्त करता है।
उदाहरण
#!/usr/bin/ruby
require 'thread'
mutex = Mutex.new
cv = ConditionVariable.new
a = Thread.new {
mutex.synchronize {
puts "A: I have critical section, but will wait for cv"
cv.wait(mutex)
puts "A: I have critical section again! I rule!"
}
}
puts "(Later, back at the ranch...)"
b = Thread.new {
mutex.synchronize {
puts "B: Now I am critical, but am done with cv"
cv.signal
puts "B: I am still critical, finishing up"
}
}
a.join
b.join
यह निम्नलिखित परिणाम का उत्पादन करेगा -
A: I have critical section, but will wait for cv
(Later, back at the ranch...)
B: Now I am critical, but am done with cv
B: I am still critical, finishing up
A: I have critical section again! I rule!
थ्रेड स्टेट्स
निम्नलिखित तालिका में दिखाए गए अनुसार पांच संभव राज्यों के अनुरूप पांच संभावित रिटर्न मान हैं। स्थिति विधि थ्रेड की स्थिति देता है।
धागा राज्य | प्रतिलाभ की मात्रा |
---|---|
runnable | Daud |
सोया हुआ | सोया हुआ |
निरस्त | निरस्त |
सामान्य रूप से समाप्त | असत्य |
अपवाद के साथ समाप्त हुआ | शून्य |
थ्रेड क्लास के तरीके
थ्रेड क्लास द्वारा निम्नलिखित तरीके प्रदान किए जाते हैं और वे कार्यक्रम में उपलब्ध सभी थ्रेड्स पर लागू होते हैं। इन विधियों को थ्रेड श्रेणी के नाम का उपयोग करते हुए कहा जाएगा -
Thread.abort_on_exception = true
अनु क्रमांक। | तरीके और विवरण |
---|---|
1 | Thread.abort_on_exception अपवाद स्थिति पर वैश्विक गर्भपात की स्थिति लौटाता है । डिफ़ॉल्ट गलत है । जब सही पर सेट किया जाता है , तो सभी थ्रेड्स निरस्त हो जाएंगे (यदि प्रक्रिया किसी भी थ्रेड में अपवाद उठाया गया है तो (0) निकल जाएगी) |
2 | Thread.abort_on_exception= जब सही पर सेट किया जाता है , तो अपवाद होने पर सभी धागे निरस्त हो जाएंगे। नया राज्य लौटाता है। |
3 | Thread.critical वैश्विक धागे की स्थिति को गंभीर स्थिति में लौटाता है । |
4 | Thread.critical= वैश्विक थ्रेड की स्थिति को गंभीर स्थिति सेट करता है और उसे लौटाता है। जब सही पर सेट किया जाता है , तो किसी मौजूदा थ्रेड के शेड्यूलिंग को प्रतिबंधित करता है। नए थ्रेड्स को बनने और चलाने से ब्लॉक नहीं करता है। कुछ थ्रेड ऑपरेशन (जैसे किसी थ्रेड को रोकना या मारना, वर्तमान थ्रेड में सोना और अपवाद उठाना) किसी महत्वपूर्ण सेक्शन में भी थ्रेड को शेड्यूल किया जा सकता है। |
5 | Thread.current वर्तमान में निष्पादित थ्रेड लौटाता है। |
6 | Thread.exit वर्तमान में चल रहे थ्रेड को टर्मिनेट करता है और दूसरे थ्रेड को शेड्यूल करता है। यदि यह थ्रेड पहले से ही मारा गया है, तो थ्रेड से बाहर निकलें । यदि यह मुख्य धागा है, या अंतिम धागा है, तो प्रक्रिया से बाहर निकलें। |
7 | Thread.fork { block } थ्रेड.न्यू का पर्यायवाची। |
8 | Thread.kill( aThread ) दिए गए धागे को बाहर निकलने का कारण बनता है |
9 | Thread.list सभी थ्रेड्स के लिए थ्रेड ऑब्जेक्ट्स की एक सरणी देता है जो या तो रन करने योग्य हैं या बंद हो गए हैं। थ्रेड। |
10 | Thread.main प्रक्रिया के लिए मुख्य सूत्र देता है। |
1 1 | Thread.new( [ arg ]* ) {| args | block } ब्लॉक में दिए गए निर्देशों को निष्पादित करने के लिए एक नया धागा बनाता है, और इसे चलाना शुरू करता है। Thread.new में दिए गए किसी भी तर्क को ब्लॉक में पारित किया गया है। |
12 | Thread.pass थ्रेड शेड्यूलर को किसी अन्य थ्रेड को निष्पादन पास करने के लिए आमंत्रित करता है। |
13 | Thread.start( [ args ]* ) {| args | block } मूल रूप से थ्रेड.न्यू के समान है । हालांकि, यदि क्लास थ्रेड को उप-वर्गित किया जाता है, तो उस उपवर्ग में कॉलिंग शुरू करने से उप-वर्ग की इनिशियलाइज़ विधि नहीं होगी । |
14 | Thread.stop वर्तमान थ्रेड के निष्पादन को रोकता है, इसे एक नींद की स्थिति में डाल देता है , और दूसरे थ्रेड के निष्पादन को शेड्यूल करता है। झूठा करने के लिए गंभीर स्थिति को हल करता है । |
थ्रेड इंस्टेंस के तरीके
ये विधियाँ एक सूत्र के उदाहरण पर लागू होती हैं। इन विधियों को एक थ्रेड के उदाहरण के रूप में निम्नानुसार बुलाया जाएगा -
#!/usr/bin/ruby
thr = Thread.new do # Calling a class method new
puts "In second thread"
raise "Raise exception"
end
thr.join # Calling an instance method join
अनु क्रमांक। | तरीके और विवरण |
---|---|
1 | thr[ aSymbol ] गुण का संदर्भ - एक थ्रेड-लोकल वैरिएबल का मान लौटाता है, या तो एक सिंबल या anSymbol नाम का उपयोग करते हुए । यदि निर्दिष्ट चर मौजूद नहीं है, तो शून्य लौटाता है । |
2 | thr[ aSymbol ] = एट्रिब्यूट असाइनमेंट - एक प्रतीक या एक स्ट्रिंग का उपयोग करके थ्रेड-लोकल वैरिएबल का मान सेट या बनाता है। |
3 | thr.abort_on_exception रिटर्न की स्थिति अपवाद पर बीच में बंद करें के लिए शर्त thr । डिफ़ॉल्ट गलत है । |
4 | thr.abort_on_exception= करने के लिए सेट करते हैं सच है, अगर एक अपवाद में उठाया है निरस्त करने के लिए (मुख्य कार्यक्रम सहित) सभी धागे का कारण बनता है thr । प्रक्रिया प्रभावी रूप से बाहर निकल जाएगी (0) । |
5 | thr.alive? रिटर्न सच अगर thr चल रहे हैं या सो रही है। |
6 | thr.exit समाप्त thr और कार्यक्रम एक और धागा चलाने के लिए। यदि यह थ्रेड पहले से ही मारा गया है, तो थ्रेड से बाहर निकलें । यदि यह मुख्य धागा है, या अंतिम धागा, प्रक्रिया से बाहर निकलता है। |
7 | thr.join कॉलिंग थ्रेड निष्पादन को निलंबित करेगा और थ्रोड चलाएगा । थ्रोट एक्सिट तक वापस नहीं आता है। मुख्य कार्यक्रम से बाहर निकलने पर शामिल नहीं होने वाले किसी भी धागे को मार दिया जाएगा। |
8 | thr.key? रिटर्न सच अगर दिए गए स्ट्रिंग (या चिह्न) एक धागे की स्थानीय चर के रूप में मौजूद है। |
9 | thr.kill के लिए पर्याय Thread.exit । |
10 | thr.priority रिटर्न की प्राथमिकता thr । डिफ़ॉल्ट शून्य है; उच्च प्राथमिकता वाले धागे निचले प्राथमिकता वाले धागे से पहले चलेंगे। |
1 1 | thr.priority= एक पूर्णांक के लिए सिंहासन की प्राथमिकता निर्धारित करता है। उच्च प्राथमिकता वाले धागे निचले प्राथमिकता वाले धागे से पहले चलेंगे। |
12 | thr.raise( anException ) से एक अपवाद को जन्म देती है thr । कॉल करने वाले को थ्रोट नहीं होना चाहिए । |
13 | thr.run जाग thr , यह समय निर्धारण के लिए योग्य बना रही है। यदि एक महत्वपूर्ण खंड में नहीं है, तो अनुसूचक को आमंत्रित करता है। |
14 | thr.safe_level रिटर्न के लिए प्रभाव में सुरक्षित स्तर thr । |
15 | thr.status रिटर्न की स्थिति thr : नींद अगर thr सो रहा है या पर आई / ओ इंतजार कर, चलाने यदि thr निष्पादित हो रहा है, झूठी अगर thr सामान्य रूप से समाप्त हो गया, और नहीं के बराबर है, तो thr एक अपवाद के साथ समाप्त हो गया। |
16 | thr.stop? रिटर्न सच अगर thr मृत या नींद की है। |
17 | thr.value Thr.join के माध्यम से पूरा करने के लिए इंतजार कर रहा है और इसके मूल्य देता है। |
18 | thr.wakeup मार्क्स thr निर्धारण के लिए पात्र के रूप में, यह अभी भी पर मैं / हे अवरोधित ही सकता है, तथापि। |