रूबी - मल्टीथ्रेडिंग

पारंपरिक कार्यक्रमों के निष्पादन के एक ही सूत्र में कथन या निर्देश शामिल होते हैं जो प्रोग्राम को समाप्त होने तक क्रमिक रूप से निष्पादित करते हैं।

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

रूबी थ्रेड वर्ग के साथ बहु-थ्रेडेड प्रोग्राम लिखना आसान बनाता है । रूबी धागे अपने कोड में संगामिति प्राप्त करने के लिए एक हल्का और कुशल तरीका है।

रूबी थ्रेड बनाना

एक नया धागा शुरू करने के लिए, बस एक ब्लॉक को 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 निर्धारण के लिए पात्र के रूप में, यह अभी भी पर मैं / हे अवरोधित ही सकता है, तथापि।