पाइथन में कंसीडर - क्विक गाइड

इस अध्याय में, हम पायथन में कंसिस्टेंसी की अवधारणा को समझेंगे और विभिन्न थ्रेड्स और प्रक्रियाओं के बारे में जानेंगे।

कंसीडर क्या है?

सरल शब्दों में, समवर्ती एक ही समय में दो या अधिक घटनाओं की घटना है। कंसीडर एक प्राकृतिक घटना है क्योंकि कई घटनाएं किसी भी समय एक साथ होती हैं।

प्रोग्रामिंग के संदर्भ में, संक्षिप्तता तब है जब दो कार्य निष्पादन में ओवरलैप होते हैं। समवर्ती प्रोग्रामिंग के साथ, हमारे अनुप्रयोगों और सॉफ्टवेयर सिस्टम के प्रदर्शन में सुधार किया जा सकता है क्योंकि हम पिछले एक के पूरा होने की प्रतीक्षा करने के बजाय अनुरोधों का सामना कर सकते हैं।

कंसीडर की ऐतिहासिक समीक्षा

निम्नलिखित बिंदु हमें संक्षिप्तता की संक्षिप्त ऐतिहासिक समीक्षा देंगे -

रेलमार्ग की अवधारणा से

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

शिक्षाविद में समवर्ती कंप्यूटिंग

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

उच्च-स्तरीय संगामिति आदिम

हाल के दिनों में, उच्च-स्तरीय संगामिति आदिम की शुरूआत के कारण प्रोग्रामर्स को समवर्ती समाधानों में सुधार हो रहा है।

प्रोग्रामिंग भाषाओं के साथ बेहतर संगामिति

Google की गोलंग, रस्ट और पायथन जैसी प्रोग्रामिंग भाषाओं ने उन क्षेत्रों में अविश्वसनीय विकास किया है जो हमें बेहतर समवर्ती समाधान प्राप्त करने में मदद करते हैं।

धागा और मल्टीथ्रेडिंग क्या है?

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

एक धागे में निम्नलिखित घटक होते हैं -

  • प्रोग्राम काउंटर जिसमें अगले निष्पादन योग्य अनुदेश का पता होता है

  • Stack

  • रजिस्टरों का सेट

  • एक अनोखी आईडी

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

उदाहरण

मान लीजिए हम एक विशेष प्रक्रिया चला रहे हैं जिसमें हम एमएस वर्ड को उसमें टाइप करने के लिए खोलते हैं। एक धागा एमएस वर्ड खोलने के लिए सौंपा जाएगा और इसमें सामग्री टाइप करने के लिए दूसरे धागे की आवश्यकता होगी। और अब, यदि हम मौजूदा को संपादित करना चाहते हैं, तो संपादन कार्य करने के लिए एक और धागा की आवश्यकता होगी।

प्रक्रिया और बहुक्रिया क्या है?

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

निम्नलिखित आरेख प्रक्रिया के विभिन्न चरणों को दर्शाता है -

एक प्रक्रिया में केवल एक धागा हो सकता है, जिसे प्राथमिक धागा कहा जाता है, या एकाधिक धागे में रजिस्टरों का अपना सेट, प्रोग्राम काउंटर और स्टैक होता है। निम्नलिखित आरेख हमें अंतर दिखाएगा -

Multiprocessing,दूसरी ओर, एक एकल कंप्यूटर प्रणाली के भीतर दो या अधिक सीपीयू इकाइयों का उपयोग होता है। हमारा प्राथमिक लक्ष्य हमारे हार्डवेयर से पूरी क्षमता प्राप्त करना है। इसे प्राप्त करने के लिए, हमें अपने कंप्यूटर सिस्टम में उपलब्ध सीपीयू कोर की पूरी संख्या का उपयोग करने की आवश्यकता है। ऐसा करने के लिए मल्टीप्रोसेसिंग सबसे अच्छा तरीका है।

पायथन सबसे लोकप्रिय प्रोग्रामिंग भाषाओं में से एक है। अनुवर्ती कुछ कारण हैं जो इसे समवर्ती अनुप्रयोगों के लिए उपयुक्त बनाते हैं -

सिंथेटिक चीनी

सिंथेटिक चीनी एक प्रोग्रामिंग भाषा के भीतर वाक्यविन्यास है जो चीजों को पढ़ने या व्यक्त करने के लिए आसान बनाने के लिए डिज़ाइन किया गया है। यह मानव उपयोग के लिए भाषा को "मीठा" बनाता है: चीजों को प्राथमिकता के आधार पर अधिक स्पष्ट रूप से, अधिक संक्षिप्त रूप से या वैकल्पिक शैली में व्यक्त किया जा सकता है। पायथन मैजिक विधियों के साथ आता है, जिसे वस्तुओं पर कार्य करने के लिए परिभाषित किया जा सकता है। इन मैजिक मेथडों को सिन्सेटिक शुगर के रूप में उपयोग किया जाता है और अधिक आसानी से समझने वाले कीवर्ड के लिए बाध्य किया जाता है।

बड़ा समुदाय

पायथन भाषा ने एआई, मशीन लर्निंग, डीप लर्निंग और क्वांटिटेटिव एनालिसिस के क्षेत्र में काम करने वाले डेटा वैज्ञानिकों और गणितज्ञों के बीच व्यापक गोद लेने की दर देखी है।

समवर्ती प्रोग्रामिंग के लिए उपयोगी एपीआई

पायथन 2 और 3 में समानांतर / समवर्ती प्रोग्रामिंग के लिए बड़ी संख्या में एपीआई समर्पित हैं। उनमें से सबसे लोकप्रिय हैंthreading, concurrent.features, multiprocessing, asyncio, gevent and greenlets, आदि।

समवर्ती अनुप्रयोगों को लागू करने में पायथन की सीमाएं

पायथन समवर्ती अनुप्रयोगों के लिए एक सीमा के साथ आता है। इस सीमा कहा जाता हैGIL (Global Interpreter Lock)अजगर के भीतर मौजूद है। जीआईएल हमें सीपीयू के कई कोर का उपयोग करने की अनुमति नहीं देता है और इसलिए हम कह सकते हैं कि पायथन में कोई सच्चे धागे नहीं हैं। GIL की अवधारणा को हम इस प्रकार समझ सकते हैं -

GIL (ग्लोबल इंटरप्रेटर लॉक)

यह पायथन दुनिया में सबसे विवादास्पद विषयों में से एक है। सीपीथॉन में, जीआईएल म्यूटेक्स है - पारस्परिक बहिष्करण ताला, जो चीजों को सुरक्षित धागा बनाता है। दूसरे शब्दों में, हम कह सकते हैं कि जीआईएल कई धागों को समानांतर में पायथन कोड को निष्पादित करने से रोकता है। ताला को एक समय में केवल एक धागे से रखा जा सकता है और अगर हम किसी धागे को निष्पादित करना चाहते हैं तो उसे पहले ताला प्राप्त करना होगा। नीचे दिए गए आरेख आपको GIL के काम को समझने में मदद करेंगे।

हालांकि, पायथन में कुछ पुस्तकालय और कार्यान्वयन हैं जैसे कि Numpy, Jpython तथा IronPytbhon. ये पुस्तकालय जीआईएल के साथ बिना किसी बातचीत के काम करते हैं।

बहुसांस्कृतिक कार्यक्रमों के संबंध में संगामिति और समानता दोनों का उपयोग किया जाता है, लेकिन उनके बीच समानता और अंतर के बारे में बहुत भ्रम है। इस संबंध में बड़ा सवाल है: संगामिति समानता या नहीं? यद्यपि दोनों शब्द काफी समान हैं, लेकिन उपरोक्त प्रश्न का उत्तर NO है, संगामिति और समानता समान नहीं हैं। अब, यदि वे समान नहीं हैं, तो उनके बीच बुनियादी अंतर क्या है?

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

विस्तार से चिंता

सम्‍मिलन तब होता है जब दो कार्य निष्पादन में ओवरलैप होते हैं। यह एक ऐसी स्थिति हो सकती है, जहां एक ही समय में एक से अधिक कार्य करने पर एक एप्लिकेशन प्रगति कर रहा है। हम इसे आरेखीय रूप से समझ सकते हैं; कई कार्य एक ही समय में प्रगति कर रहे हैं, निम्नानुसार हैं -

सम्‍मिलन का स्‍तर

इस खंड में, हम प्रोग्रामिंग के संदर्भ में संगामिति के तीन महत्वपूर्ण स्तरों पर चर्चा करेंगे -

लो-लेवल का कंसीडर

समसामयिकता के इस स्तर में, परमाणु संचालन का स्पष्ट उपयोग है। हम एप्लिकेशन बिल्डिंग के लिए इस तरह के कॉन्सेप्ट का उपयोग नहीं कर सकते हैं, क्योंकि यह बहुत त्रुटि-प्रवण है और डीबग करना कठिन है। यहां तक ​​कि पाइथन इस तरह के कंसिस्टेंसी का समर्थन नहीं करता है।

मिड-लेवल कॉनसेरी

इस संगामिति में, स्पष्ट परमाणु संचालन का उपयोग नहीं किया गया है। यह स्पष्ट तालों का उपयोग करता है। पायथन और अन्य प्रोग्रामिंग लैंग्वेज इस तरह के कॉनसेप्ट का समर्थन करती हैं। अधिकतर एप्लिकेशन प्रोग्रामर इस कॉन्सिक्वेंसी का उपयोग करते हैं।

हाई-लेवल कॉन्सिक्वेंसी

इस संगोष्ठी में, न तो स्पष्ट परमाणु संचालन और न ही स्पष्ट ताले का उपयोग किया जाता है। अजगर हैconcurrent.futures मॉड्यूल इस तरह के समादेश का समर्थन करने के लिए।

समवर्ती प्रणालियों के गुण

किसी प्रोग्राम या समवर्ती प्रणाली के सही होने के लिए, कुछ गुणों को इससे संतुष्ट होना चाहिए। प्रणाली की समाप्ति से संबंधित गुण इस प्रकार हैं -

सही संपत्ति

शुद्धता संपत्ति का मतलब है कि प्रोग्राम या सिस्टम को वांछित सही उत्तर प्रदान करना चाहिए। इसे सरल रखने के लिए, हम कह सकते हैं कि सिस्टम को प्रोग्राम प्रोग्राम स्टेट को अंतिम रूप से सही ढंग से मैप करना होगा।

सुरक्षा संपत्ति

सुरक्षा संपत्ति का मतलब है कि प्रोग्राम या सिस्टम में रहना चाहिए “good” या “safe” राज्य और कभी कुछ नहीं करता “bad”

लीवर की संपत्ति

इस संपत्ति का मतलब है कि एक प्रोग्राम या सिस्टम होना चाहिए “make progress” और यह कुछ वांछनीय स्थिति पर पहुंच जाएगा।

समवर्ती प्रणालियों के अभिनेता

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

समवर्ती प्रणालियों के संसाधन

अभिनेताओं को अपने कार्यों को करने के लिए संसाधनों का उपयोग करना चाहिए जैसे कि मेमोरी, डिस्क, प्रिंटर आदि।

नियमों का निश्चित सेट

प्रत्येक समवर्ती प्रणाली में अभिनेताओं द्वारा किए जाने वाले कार्यों के प्रकार और प्रत्येक के लिए समय निर्धारित करने के लिए नियमों का एक समूह होना चाहिए। कार्य तालों को प्राप्त करना, मेमोरी साझा करना, राज्य को संशोधित करना आदि हो सकते हैं।

समवर्ती प्रणालियों के अवरोध

डेटा साझा करना

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

डेटा शेयरिंग प्रतिबंध

सबसे सरल उपाय किसी भी परस्पर डेटा को साझा करना नहीं है। इस मामले में, हमें स्पष्ट लॉकिंग का उपयोग करने की आवश्यकता नहीं है और आपसी डेटा के कारण संगामिति का अवरोध हल हो जाएगा।

डेटा संरचना सहायता

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

अपरिवर्तनीय डेटा अंतरण

कभी-कभी, हम जिस डेटा संरचना का उपयोग कर रहे हैं, वह कहते हैं कि संगामिति कतार, उपयुक्त नहीं है तो हम बिना लॉक किए अपरिवर्तनीय डेटा को पास कर सकते हैं।

म्यूटेबल डेटा ट्रांसफर

उपरोक्त समाधान की निरंतरता में, मान लें कि यदि अपरिवर्तनीय डेटा के बजाय केवल उत्परिवर्तित डेटा को पास करना आवश्यक है, तो हम केवल पढ़ने योग्य म्यूटेबल डेटा पास कर सकते हैं।

I / O संसाधन साझा करना

समवर्ती प्रणालियों को लागू करने में एक और महत्वपूर्ण मुद्दा थ्रेड्स या प्रक्रियाओं द्वारा I / O संसाधनों का उपयोग है। समस्या तब उत्पन्न होती है जब एक धागा या प्रक्रिया इतने लंबे समय तक I / O का उपयोग कर रही है और अन्य बेकार बैठी है। I / O भारी एप्लिकेशन के साथ काम करते समय हम इस तरह की बाधा देख सकते हैं। इसे एक उदाहरण की मदद से समझा जा सकता है, वेब ब्राउज़र से पृष्ठों का अनुरोध। यह एक भारी एप्लिकेशन है। यहां, यदि जिस दर पर डेटा का अनुरोध किया जाता है, वह उस दर से धीमा होता है जिस दर पर उसका उपभोग किया जाता है, तो हमारे पास हमारे समवर्ती प्रणाली में I / O अवरोध है।

निम्नलिखित पायथन स्क्रिप्ट एक वेब पेज का अनुरोध करने और हमारे नेटवर्क को अनुरोधित पेज प्राप्त करने में लगने वाले समय के लिए है -

import urllib.request
import time
ts = time.time()
req = urllib.request.urlopen('http://www.tutorialspoint.com')
pageHtml = req.read()
te = time.time()
print("Page Fetching Time : {} Seconds".format (te-ts))

उपरोक्त स्क्रिप्ट को निष्पादित करने के बाद, हम पृष्ठ लाने का समय प्राप्त कर सकते हैं जैसा कि नीचे दिखाया गया है।

उत्पादन

Page Fetching Time: 1.0991398811340332 Seconds

हम देख सकते हैं कि पेज लाने का समय एक सेकंड से अधिक है। अब क्या होगा अगर हम हजारों विभिन्न वेब पेज लाना चाहते हैं, तो आप समझ सकते हैं कि हमारे नेटवर्क को कितना समय लगेगा।

समानांतरवाद क्या है?

समानांतरवाद को कार्यों को उप-मुखौटे में विभाजित करने की कला के रूप में परिभाषित किया जा सकता है जिसे एक साथ संसाधित किया जा सकता है। यह चर्चा के विपरीत है, जैसा कि ऊपर चर्चा की गई है, जिसमें एक ही समय में दो या अधिक घटनाएं हो रही हैं। हम इसे आरेखीय रूप से समझ सकते हैं; एक कार्य को कई उप-प्रकारों में विभाजित किया जाता है जिन्हें समानांतर में संसाधित किया जा सकता है, इस प्रकार है -

समवर्ती और समानता के बीच अंतर के बारे में अधिक विचार प्राप्त करने के लिए, निम्नलिखित बिंदुओं पर विचार करें -

समवर्ती लेकिन समानांतर नहीं

एक अनुप्रयोग समवर्ती हो सकता है, लेकिन समानांतर का मतलब यह नहीं है कि यह एक ही समय में एक से अधिक कार्य करता है, लेकिन कार्य उप-योगों में टूट नहीं रहे हैं।

समानांतर लेकिन समवर्ती नहीं

एक अनुप्रयोग समानांतर हो सकता है लेकिन समवर्ती नहीं है इसका मतलब है कि यह एक समय में केवल एक कार्य पर काम करता है और उप-कार्य में टूटे हुए कार्यों को समानांतर में संसाधित किया जा सकता है।

न तो समानांतर और न ही समवर्ती

एक आवेदन न तो समानांतर और न ही समवर्ती हो सकता है। इसका मतलब यह है कि यह एक समय में केवल एक ही कार्य पर काम करता है और कार्य कभी भी उप-कार्य में नहीं टूटा है।

समानांतर और समवर्ती दोनों

एक आवेदन समानांतर और समवर्ती दोनों हो सकता है इसका मतलब यह है कि यह दोनों एक समय में कई कार्यों पर काम करता है और कार्य समानांतर में निष्पादित करने के लिए उप-प्रकारों में टूट जाता है।

समानता की आवश्यकता

हम एकल सीपीयू के विभिन्न कोर के बीच या एक नेटवर्क के भीतर जुड़े कई कंप्यूटरों के बीच उपशीर्षक वितरित करके समानता प्राप्त कर सकते हैं।

समानता को प्राप्त करने के लिए क्यों आवश्यक है यह समझने के लिए निम्नलिखित महत्वपूर्ण बिंदुओं पर विचार करें -

कुशल कोड निष्पादन

समानांतरवाद की मदद से, हम अपने कोड को कुशलता से चला सकते हैं। यह हमारे समय की बचत करेगा क्योंकि भागों में समान कोड समानांतर में चल रहा है।

अनुक्रमिक कंप्यूटिंग की तुलना में तेज़

अनुक्रमिक कंप्यूटिंग शारीरिक और व्यावहारिक कारकों से विवश है जिसके कारण तेजी से कंप्यूटिंग परिणाम प्राप्त करना संभव नहीं है। दूसरी ओर, यह समस्या समानांतर कंप्यूटिंग द्वारा हल की गई है और हमें क्रमिक कंप्यूटिंग की तुलना में तेजी से कंप्यूटिंग परिणाम देती है।

कम निष्पादन समय

समानांतर प्रसंस्करण कार्यक्रम कोड के निष्पादन समय को कम करता है।

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

कार्यान्वयन के लिए प्रोसेसर की समझ

हम समवर्ती, समानता और उनके बीच अंतर के बारे में जानते हैं लेकिन उस प्रणाली के बारे में क्या है जिस पर इसे लागू किया जाना है। सिस्टम की समझ होना बहुत जरूरी है, जिस पर हम अमल करने जा रहे हैं, क्योंकि यह हमें सॉफ्टवेयर डिजाइन करते समय सूचित निर्णय लेने का लाभ देता है। हमारे पास निम्नलिखित दो प्रकार के प्रोसेसर हैं -

सिंगल-कोर प्रोसेसर

सिंगल-कोर प्रोसेसर किसी भी समय एक थ्रेड को निष्पादित करने में सक्षम हैं। ये प्रोसेसर इस्तेमाल करते हैंcontext switchingकिसी विशिष्ट समय पर थ्रेड के लिए सभी आवश्यक जानकारी संग्रहीत करने और फिर बाद में जानकारी को पुनर्स्थापित करने के लिए। संदर्भ स्विचिंग तंत्र हमें दिए गए सेकंड के भीतर कई थ्रेड पर प्रगति करने में मदद करता है और ऐसा लगता है जैसे सिस्टम कई चीजों पर काम कर रहा है।

सिंगल-कोर प्रोसेसर कई फायदे के साथ आते हैं। इन प्रोसेसर को कम बिजली की आवश्यकता होती है और कई कोर के बीच कोई जटिल संचार प्रोटोकॉल नहीं होता है। दूसरी ओर, सिंगल-कोर प्रोसेसर की गति सीमित है और यह बड़े अनुप्रयोगों के लिए उपयुक्त नहीं है।

मल्टी-कोर प्रोसेसर

मल्टी-कोर प्रोसेसर में कई स्वतंत्र प्रसंस्करण इकाइयाँ भी होती हैं cores

ऐसे प्रोसेसर को संदर्भ स्विचिंग तंत्र की आवश्यकता नहीं होती है क्योंकि प्रत्येक कोर में संग्रहीत निर्देशों के अनुक्रम को निष्पादित करने की आवश्यकता होती है।

फ़ेच-डिकोड-एक्सेल्यूट साइकिल

मल्टी-कोर प्रोसेसर के कोर निष्पादन के लिए एक चक्र का पालन करते हैं। इस चक्र को कहा जाता हैFetch-Decode-Executeचक्र। इसमें निम्नलिखित चरण शामिल हैं -

लाना

यह चक्र का पहला चरण है, जिसमें प्रोग्राम मेमोरी से निर्देशों को लाना शामिल है।

व्याख्या करना

हाल ही में लाए गए निर्देशों को संकेतों की एक श्रृंखला में बदल दिया जाएगा जो सीपीयू के अन्य भागों को ट्रिगर करेगा।

निष्पादित

यह अंतिम चरण है जिसमें भ्रूण और डिकोड किए गए निर्देशों को निष्पादित किया जाएगा। निष्पादन का परिणाम एक सीपीयू रजिस्टर में संग्रहीत किया जाएगा।

यहाँ पर एक फायदा यह है कि मल्टी-कोर प्रोसेसर में निष्पादन सिंगल-कोर प्रोसेसर की तुलना में तेज़ होता है। यह बड़े अनुप्रयोगों के लिए उपयुक्त है। दूसरी ओर, कई कोर के बीच जटिल संचार प्रोटोकॉल एक मुद्दा है। सिंगल-कोर प्रोसेसर की तुलना में मल्टीपल कोर को अधिक शक्ति की आवश्यकता होती है।

अलग-अलग सिस्टम और मेमोरी आर्किटेक्चर स्टाइल हैं जिन्हें प्रोग्राम या समवर्ती सिस्टम को डिजाइन करते समय विचार करने की आवश्यकता होती है। यह बहुत आवश्यक है क्योंकि एक प्रणाली और स्मृति शैली एक कार्य के लिए उपयुक्त हो सकती है लेकिन अन्य कार्य के लिए त्रुटि प्रवण हो सकती है।

संगामिति का समर्थन करने वाला कंप्यूटर सिस्टम आर्किटेक्चर

माइकल फ्लिन ने 1972 में कंप्यूटर सिस्टम आर्किटेक्चर की विभिन्न शैलियों को श्रेणीबद्ध करने के लिए कर दिया। यह वर्गीकरण चार विभिन्न शैलियों को परिभाषित करता है -

  • एकल निर्देश धारा, एकल डेटा स्ट्रीम (SISD)
  • एकल निर्देश धारा, एकाधिक डेटा स्ट्रीम (SIMD)
  • एकाधिक निर्देश धारा, एकल डेटा स्ट्रीम (MISD)
  • मल्टीपल इंस्ट्रक्शन स्ट्रीम, मल्टीपल डेटा स्ट्रीम (MIMD)।

एकल निर्देश धारा, एकल डेटा स्ट्रीम (SISD)

जैसा कि नाम से पता चलता है, इस तरह के सिस्टम में डेटा स्ट्रीम निष्पादित करने के लिए एक अनुक्रमिक आवक डेटा स्ट्रीम और एक एकल प्रोसेसिंग यूनिट होगी। वे समरूप कंप्यूटिंग आर्किटेक्चर वाले यूनिप्रोसेसर सिस्टम की तरह हैं। निम्नलिखित SISD की वास्तुकला है -

SISD के लाभ

SISD वास्तुकला के फायदे इस प्रकार हैं -

  • इसके लिए कम शक्ति की आवश्यकता होती है।
  • कई कोर के बीच जटिल संचार प्रोटोकॉल का कोई मुद्दा नहीं है।

SISD के नुकसान

SISD वास्तुकला के नुकसान इस प्रकार हैं -

  • SISD आर्किटेक्चर की गति सिंगल-कोर प्रोसेसर की तरह ही सीमित है।
  • यह बड़े अनुप्रयोगों के लिए उपयुक्त नहीं है।

एकल निर्देश धारा, एकाधिक डेटा स्ट्रीम (SIMD)

जैसा कि नाम से पता चलता है, इस तरह के सिस्टम में कई इनकमिंग डेटा स्ट्रीम और प्रोसेसिंग यूनिट की संख्या होती है जो किसी भी समय एक ही निर्देश पर कार्य कर सकते हैं। वे समानांतर कंप्यूटिंग आर्किटेक्चर वाले मल्टीप्रोसेसर सिस्टम की तरह हैं। निम्नलिखित SIMD की वास्तुकला है -

SIMD के लिए सबसे अच्छा उदाहरण ग्राफिक्स कार्ड है। इन कार्डों में सैकड़ों व्यक्तिगत प्रसंस्करण इकाइयाँ होती हैं। अगर हम SISD और SIMD के बीच कम्प्यूटेशनल अंतर के बारे में बात करते हैं तो एडिंग एरे के लिए[5, 15, 20] तथा [15, 25, 10],SISD आर्किटेक्चर को तीन अलग-अलग ऐड ऑपरेशन करने होंगे। दूसरी ओर, SIMD आर्किटेक्चर के साथ, हम तब सिंगल ऐड ऑपरेशन में जोड़ सकते हैं।

SIMD के फायदे

SIMD वास्तुकला के फायदे इस प्रकार हैं -

  • कई तत्वों पर एक ही ऑपरेशन केवल एक निर्देश का उपयोग करके किया जा सकता है।

  • प्रोसेसर के कोर की संख्या बढ़ाकर सिस्टम के थ्रूपुट को बढ़ाया जा सकता है।

  • प्रसंस्करण गति SISD वास्तुकला से अधिक है।

SIMD के नुकसान

SIMD वास्तुकला के नुकसान इस प्रकार हैं -

  • प्रोसेसर के कोर की संख्या के बीच जटिल संचार है।
  • लागत SISD वास्तुकला की तुलना में अधिक है।

एकाधिक निर्देश एकल डेटा (MISD) स्ट्रीम

MISD स्ट्रीम वाले सिस्टम में एक ही डेटा सेट पर अलग-अलग निर्देशों को निष्पादित करके विभिन्न संचालन करने वाली प्रसंस्करण इकाइयाँ होती हैं। निम्नलिखित MISD की वास्तुकला है -

MISD वास्तुकला के प्रतिनिधि अभी तक व्यावसायिक रूप से मौजूद नहीं हैं।

एकाधिक निर्देश एकाधिक डेटा (MIMD) स्ट्रीम

MIMD वास्तुकला का उपयोग करते हुए सिस्टम में, मल्टीप्रोसेसर सिस्टम में प्रत्येक प्रोसेसर समानांतर में सेट किए गए डेटा के अलग-अलग सेट पर स्वतंत्र रूप से निर्देशों के विभिन्न सेटों को निष्पादित कर सकता है। यह SIMD वास्तुकला के विपरीत है जिसमें एकल ऑपरेशन कई डेटा सेट पर निष्पादित किया जाता है। निम्नलिखित MIMD की वास्तुकला है -

एक सामान्य मल्टीप्रोसेसर MIMD आर्किटेक्चर का उपयोग करता है। ये आर्किटेक्चर मूल रूप से कई अनुप्रयोग क्षेत्रों जैसे कंप्यूटर-एडेड डिज़ाइन / कंप्यूटर-एडेड विनिर्माण, सिमुलेशन, मॉडलिंग, संचार स्विच, आदि में उपयोग किए जाते हैं।

मेमोरी आर्किटेक्चर समसामयिक समर्थन करते हैं

समवर्ती और समानता जैसी अवधारणाओं के साथ काम करते समय, कार्यक्रमों को गति देने की हमेशा आवश्यकता होती है। कंप्यूटर डिजाइनरों द्वारा पाया गया एक समाधान साझा-मेमोरी मल्टी-कंप्यूटर बनाना है, अर्थात, एकल भौतिक पता स्थान वाले कंप्यूटर, जो सभी कोर तक पहुंचते हैं जो एक प्रोसेसर है। इस परिदृश्य में, वास्तुकला की विभिन्न शैलियों की संख्या हो सकती है लेकिन निम्नलिखित तीन महत्वपूर्ण वास्तुकला शैलियाँ हैं -

UMA (यूनिफ़ॉर्म मेमोरी एक्सेस)

इस मॉडल में, सभी प्रोसेसर समान रूप से भौतिक मेमोरी साझा करते हैं। सभी प्रोसेसरों के पास सभी मेमोरी शब्दों के लिए समान पहुंच समय है। प्रत्येक प्रोसेसर में एक निजी कैश मेमोरी हो सकती है। परिधीय उपकरण नियमों के एक सेट का पालन करते हैं।

जब सभी प्रोसेसर सभी परिधीय उपकरणों के लिए समान पहुंच रखते हैं, तो सिस्टम को ए कहा जाता है symmetric multiprocessor। जब केवल एक या कुछ प्रोसेसर परिधीय उपकरणों तक पहुंच सकते हैं, तो सिस्टम को ए कहा जाता हैasymmetric multiprocessor

गैर-समान मेमोरी एक्सेस (NUMA)

NUMA मल्टीप्रोसेसर मॉडल में, मेमोरी शब्द के स्थान के साथ एक्सेस टाइम बदलता रहता है। यहां, साझा मेमोरी सभी प्रोसेसर के बीच भौतिक रूप से वितरित की जाती है, जिसे स्थानीय मेमोरी कहा जाता है। सभी स्थानीय यादों का संग्रह एक वैश्विक पता स्थान बनाता है जिसे सभी प्रोसेसर द्वारा पहुँचा जा सकता है।

कैश केवल मेमोरी आर्किटेक्चर (COMA)

COMA मॉडल NUMA मॉडल का एक विशेष संस्करण है। यहां, सभी वितरित मुख्य यादें कैश यादों में बदल जाती हैं।

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

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

थ्रेड के राज्य

थ्रेड्स की कार्यक्षमता को गहराई से समझने के लिए, हमें थ्रेड्स या विभिन्न थ्रेड स्टेट्स के जीवनचक्र के बारे में जानने की आवश्यकता है। आमतौर पर, एक धागा पांच अलग-अलग राज्यों में मौजूद हो सकता है। विभिन्न राज्यों को नीचे दिखाया गया है -

नया सूत्र

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

runnable

जैसा कि नए जन्मे धागे को शुरू किया जाता है, धागा चलने योग्य हो जाता है यानी दौड़ने की प्रतीक्षा करता है। इस राज्य में, इसके पास सभी संसाधन हैं लेकिन फिर भी कार्य अनुसूचक ने इसे चलाने के लिए निर्धारित नहीं किया है।

दौड़ना

इस स्थिति में, थ्रेड प्रगति करता है और कार्य को निष्पादित करता है, जिसे चलाने के लिए कार्य शेड्यूलर द्वारा चुना गया है। अब, थ्रेड या तो मृत अवस्था या गैर-रननीय / प्रतीक्षा स्थिति में जा सकता है।

गैर चल / प्रतीक्षा

इस स्थिति में, थ्रेड को रोक दिया जाता है क्योंकि यह या तो कुछ I / O अनुरोध की प्रतिक्रिया की प्रतीक्षा कर रहा है या अन्य थ्रेड के निष्पादन के पूरा होने की प्रतीक्षा कर रहा है।

मृत

जब वह अपने कार्य को पूरा करता है या अन्यथा समाप्त हो जाता है, तो एक रननीय थ्रेड समाप्त अवस्था में प्रवेश करता है।

निम्नलिखित चित्र एक धागे के पूर्ण जीवन चक्र को दर्शाता है -

धागे के प्रकार

इस खंड में, हम विभिन्न प्रकार के धागे देखेंगे। प्रकार नीचे वर्णित हैं -

उपयोगकर्ता स्तर के धागे

ये उपयोगकर्ता द्वारा प्रबंधित धागे हैं।

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

उपयोगकर्ता स्तर के धागे के उदाहरण हैं -

  • जावा धागे
  • POSIX धागे

उपयोगकर्ता स्तर के लाभ

उपयोगकर्ता स्तर के धागे के विभिन्न फायदे निम्नलिखित हैं -

  • थ्रेड स्विचिंग के लिए कर्नेल मोड विशेषाधिकारों की आवश्यकता नहीं होती है।
  • उपयोगकर्ता स्तर का धागा किसी भी ऑपरेटिंग सिस्टम पर चल सकता है।
  • निर्धारण उपयोगकर्ता स्तर के थ्रेड में विशिष्ट हो सकता है।
  • उपयोगकर्ता स्तर थ्रेड्स बनाने और प्रबंधित करने के लिए तेज़ हैं।

उपयोगकर्ता स्तर थ्रेड्स का नुकसान

उपयोगकर्ता स्तर के धागे के विभिन्न नुकसान निम्नलिखित हैं -

  • एक सामान्य ऑपरेटिंग सिस्टम में, अधिकांश सिस्टम कॉल अवरुद्ध हैं।
  • Multithreaded एप्लिकेशन मल्टीप्रोसेसिंग का लाभ नहीं ले सकता है।

कर्नेल स्तर सूत्र

ऑपरेटिंग सिस्टम प्रबंधित थ्रेड कर्नेल पर कार्य करता है, जो एक ऑपरेटिंग सिस्टम कोर है।

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

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

कर्नेल स्तर के धागे के लाभ

कर्नेल स्तर के धागे के विभिन्न फायदे निम्नलिखित हैं -

  • कर्नेल एक साथ कई प्रक्रियाओं पर एक ही प्रक्रिया से कई थ्रेड शेड्यूल कर सकता है।

  • यदि किसी प्रक्रिया में एक थ्रेड अवरुद्ध है, तो कर्नेल उसी प्रक्रिया के दूसरे थ्रेड को शेड्यूल कर सकता है।

  • कर्नेल रूटीन खुद को मल्टीथ्रेड किया जा सकता है।

कर्नेल लेवल थ्रेड्स का नुकसान

  • कर्नेल थ्रेड्स आमतौर पर उपयोगकर्ता थ्रेड्स की तुलना में बनाने और प्रबंधित करने के लिए धीमे होते हैं।

  • एक ही प्रक्रिया के भीतर एक धागे से दूसरे में नियंत्रण स्थानांतरित करने के लिए कर्नेल में एक मोड स्विच की आवश्यकता होती है।

थ्रेड कंट्रोल ब्लॉक - टीसीबी

थ्रेड कंट्रोल ब्लॉक (टीसीबी) को ऑपरेटिंग सिस्टम के कर्नेल में डेटा संरचना के रूप में परिभाषित किया जा सकता है जिसमें मुख्य रूप से थ्रेड के बारे में जानकारी होती है। टीसीबी में संग्रहीत थ्रेड-विशिष्ट जानकारी प्रत्येक प्रक्रिया के बारे में कुछ महत्वपूर्ण जानकारी को उजागर करेगी।

टीसीबी में निहित थ्रेड्स से संबंधित निम्नलिखित बिंदुओं पर विचार करें -

  • Thread identification - यह प्रत्येक नए थ्रेड को निर्दिष्ट अद्वितीय थ्रेड आईडी (tid) है।

  • Thread state - इसमें थ्रेड की स्थिति (रनिंग, रननेबल, नॉन-रनिंग, डेड) से संबंधित जानकारी होती है।

  • Program Counter (PC) - यह थ्रेड के वर्तमान कार्यक्रम निर्देश को इंगित करता है।

  • Register set - इसमें अभिकलन के लिए उन्हें सौंपा गया थ्रेड का रजिस्टर मान है।

  • Stack Pointer- यह प्रक्रिया में धागे के ढेर की ओर इशारा करता है। इसमें थ्रेड के दायरे में स्थानीय चर शामिल हैं।

  • Pointer to PCB - इसमें उस थ्रेड को बनाने वाले प्रोसेस का पॉइंटर होता है।

प्रक्रिया और धागे के बीच संबंध

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

निम्न तालिका प्रक्रिया और धागे के बीच तुलना को दर्शाती है -

प्रोसेस थ्रेड
प्रक्रिया भारी वजन या संसाधन गहन है। थ्रेड हल्का होता है जो एक प्रक्रिया से कम संसाधन लेता है।
प्रक्रिया स्विचिंग को ऑपरेटिंग सिस्टम के साथ सहभागिता की आवश्यकता होती है। थ्रेड स्विचिंग को ऑपरेटिंग सिस्टम के साथ सहभागिता करने की आवश्यकता नहीं है।
एकाधिक प्रसंस्करण वातावरणों में, प्रत्येक प्रक्रिया समान कोड निष्पादित करती है, लेकिन इसकी अपनी मेमोरी और फ़ाइल संसाधन होते हैं। सभी थ्रेड्स खुली फ़ाइलों, बाल प्रक्रियाओं का एक ही सेट साझा कर सकते हैं।
यदि एक प्रक्रिया को अवरुद्ध किया जाता है, तो कोई भी अन्य प्रक्रिया तब तक निष्पादित नहीं हो सकती है जब तक कि पहली प्रक्रिया अनब्लॉक न हो जाए। जबकि एक धागा अवरुद्ध है और प्रतीक्षा कर रहा है, उसी कार्य में एक दूसरा धागा चल सकता है।
थ्रेड्स का उपयोग किए बिना कई प्रक्रियाएं अधिक संसाधनों का उपयोग करती हैं। एकाधिक थ्रेडेड प्रक्रियाएँ कम संसाधनों का उपयोग करती हैं।
कई प्रक्रियाओं में, प्रत्येक प्रक्रिया दूसरों के स्वतंत्र रूप से संचालित होती है। एक धागा दूसरे धागे के डेटा को पढ़, लिख या बदल सकता है।
यदि मूल प्रक्रिया में कोई बदलाव होगा तो यह बच्चे की प्रक्रियाओं को प्रभावित नहीं करता है। यदि मुख्य सूत्र में कोई बदलाव होगा तो यह उस प्रक्रिया के अन्य धागों के व्यवहार को प्रभावित कर सकता है।
सिबलिंग प्रक्रियाओं के साथ संवाद करने के लिए, प्रक्रियाओं को अंतर-प्रक्रिया संचार का उपयोग करना चाहिए। थ्रेड्स सीधे उस प्रक्रिया के अन्य थ्रेड्स के साथ संवाद कर सकते हैं।

मल्टीथ्रेडिंग की अवधारणा

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

मल्टीथ्रेडिंग की अवधारणा को निम्नलिखित उदाहरण की मदद से समझा जा सकता है।

उदाहरण

मान लीजिए हम एक प्रक्रिया चला रहे हैं। कुछ लिखने के लिए एमएस शब्द खोलने की प्रक्रिया हो सकती है। ऐसी प्रक्रिया में, एक धागा को एमएस शब्द खोलने के लिए सौंपा जाएगा और लिखने के लिए एक और धागा की आवश्यकता होगी। अब, मान लीजिए कि यदि हम कुछ संपादित करना चाहते हैं, तो संपादन कार्य करने के लिए एक और सूत्र की आवश्यकता होगी।

निम्नलिखित चित्र हमें यह समझने में मदद करते हैं कि स्मृति में कितने धागे मौजूद हैं -

हम उपरोक्त आरेख में देख सकते हैं कि एक प्रक्रिया के भीतर एक से अधिक थ्रेड मौजूद हो सकते हैं, जहां हर थ्रेड में अपना स्वयं का रजिस्टर सेट और स्थानीय चर होते हैं। इसके अलावा, एक प्रक्रिया में सभी धागे वैश्विक चर साझा करते हैं।

मल्टीथ्रेडिंग के पेशेवरों

आइए अब हम मल्टीथ्रेडिंग के कुछ फायदे देखते हैं। फायदे इस प्रकार हैं -

  • Speed of communication - मल्टीथ्रेडिंग गणना की गति में सुधार करता है क्योंकि प्रत्येक कोर या प्रोसेसर अलग-अलग थ्रेड को समवर्ती रूप से संभालता है।

  • Program remains responsive - यह प्रोग्राम को रिस्पॉन्सिव रहने देता है क्योंकि एक थ्रेड इनपुट का इंतजार करता है और दूसरा उसी समय GUI चलाता है।

  • Access to global variables - मल्टीथ्रेडिंग में, किसी विशेष प्रक्रिया के सभी थ्रेड्स ग्लोबल वैरिएबल तक पहुंच सकते हैं और अगर ग्लोबल वैरिएबल में कोई बदलाव है तो यह अन्य थ्रेड्स को भी दिखाई देता है।

  • Utilization of resources - प्रत्येक प्रोग्राम में कई थ्रेड्स चलाने से CPU का बेहतर उपयोग होता है और CPU का निष्क्रिय समय कम हो जाता है।

  • Sharing of data - प्रत्येक थ्रेड के लिए अतिरिक्त स्थान की आवश्यकता नहीं है क्योंकि किसी प्रोग्राम के भीतर थ्रेड्स समान डेटा साझा कर सकते हैं।

मल्टीथ्रेडिंग के विपक्ष

आइए अब हम मल्टीथ्रेडिंग के कुछ नुकसान देखते हैं। नुकसान इस प्रकार हैं -

  • Not suitable for single processor system - मल्टी-प्रोसेसर सिस्टम पर प्रदर्शन की तुलना में मल्टीथ्रेडिंग को एकल प्रोसेसर सिस्टम पर गणना की गति के मामले में प्रदर्शन प्राप्त करना मुश्किल लगता है।

  • Issue of security - जैसा कि हम जानते हैं कि प्रोग्राम के सभी थ्रेड्स समान डेटा साझा करते हैं, इसलिए हमेशा सुरक्षा का मुद्दा होता है क्योंकि कोई भी अज्ञात थ्रेड डेटा को बदल सकता है।

  • Increase in complexity - मल्टीथ्रेडिंग से कार्यक्रम की जटिलता बढ़ सकती है और डिबगिंग मुश्किल हो जाती है।

  • Lead to deadlock state - मल्टीथ्रेडिंग प्रोग्राम को गतिरोध स्थिति को प्राप्त करने के संभावित जोखिम तक ले जा सकता है।

  • Synchronization required- आपसी बहिष्कार से बचने के लिए सिंक्रोनाइज़ेशन की आवश्यकता होती है। यह अधिक मेमोरी और सीपीयू उपयोग की ओर जाता है।

इस अध्याय में, हम सीखेंगे कि पायथन में धागे को कैसे लागू किया जाए।

थ्रेड कार्यान्वयन के लिए पायथन मॉड्यूल

पायथन थ्रेड्स को कभी-कभी हल्की प्रक्रिया कहा जाता है क्योंकि थ्रेड प्रक्रियाओं की तुलना में बहुत कम मेमोरी पर कब्जा कर लेते हैं। थ्रेड्स एक साथ कई कार्य करने की अनुमति देते हैं। पायथन में, हमारे पास निम्नलिखित दो मॉड्यूल हैं जो एक कार्यक्रम में धागे को लागू करते हैं -

  • <_thread>module

  • <threading>module

इन दो मॉड्यूल के बीच मुख्य अंतर यह है कि <_thread> मॉड्यूल एक थ्रेड को एक फ़ंक्शन के रूप में मानता है, जबकि, <threading>मॉड्यूल हर धागे को एक वस्तु के रूप में मानता है और इसे वस्तु उन्मुख तरीके से लागू करता है। इसके अलावा,<_thread>मॉड्यूल निम्न स्तर के थ्रेडिंग में प्रभावी है और इसकी तुलना में कम क्षमताएं हैं <threading> मापांक।

<_थ्रेड> मॉड्यूल

पायथन के पुराने संस्करण में, हमारे पास था <thread>मॉड्यूल लेकिन इसे काफी लंबे समय के लिए "पदावनत" माना जाता है। उपयोगकर्ताओं को उपयोग करने के लिए प्रोत्साहित किया गया है<threading>इसके बजाय मॉड्यूल। इसलिए, पायथन 3 में मॉड्यूल "थ्रेड" अब उपलब्ध नहीं है। इसका नाम बदलकर "<_thread>"पायथन 3 में पीछे की असंगतताओं के लिए।

की सहायता से नया धागा उत्पन्न करना <_thread> मॉड्यूल, हमें कॉल करने की आवश्यकता है start_new_threadइसका तरीका। इस पद्धति के कार्य को निम्नलिखित सिंटैक्स की सहायता से समझा जा सकता है -

_thread.start_new_thread ( function, args[, kwargs] )

यहाँ -

  • args तर्कों का एक तड़का है

  • kwargs कीवर्ड दलीलों का एक वैकल्पिक शब्दकोश है

यदि हम किसी तर्क को पारित किए बिना फ़ंक्शन को कॉल करना चाहते हैं तो हमें तर्कों का एक खाली टपल उपयोग करने की आवश्यकता है args

यह विधि कॉल रिटर्न तुरंत शुरू होती है, बच्चा थ्रेड शुरू होता है, और पास की सूची के साथ फ़ंक्शन को कॉल करता है, यदि कोई हो, तो आर्ग्स का। जब फ़ंक्शन वापस आता है, तो थ्रेड समाप्त हो जाता है।

उदाहरण

निम्नलिखित का उपयोग करके नया धागा उत्पन्न करने के लिए एक उदाहरण है <_thread>मापांक। हम यहां start_new_thread () विधि का उपयोग कर रहे हैं।

import _thread
import time

def print_time( threadName, delay):
   count = 0
   while count < 5:
      time.sleep(delay)
      count += 1
      print ("%s: %s" % ( threadName, time.ctime(time.time()) ))

try:
   _thread.start_new_thread( print_time, ("Thread-1", 2, ) )
   _thread.start_new_thread( print_time, ("Thread-2", 4, ) )
except:
   print ("Error: unable to start thread")
while 1:
   pass

उत्पादन

निम्नलिखित आउटपुट से हमें नए धागों की पीढ़ी को समझने में मदद मिलेगी <_thread> मापांक।

Thread-1: Mon Apr 23 10:03:33 2018
Thread-2: Mon Apr 23 10:03:35 2018
Thread-1: Mon Apr 23 10:03:35 2018
Thread-1: Mon Apr 23 10:03:37 2018
Thread-2: Mon Apr 23 10:03:39 2018
Thread-1: Mon Apr 23 10:03:39 2018
Thread-1: Mon Apr 23 10:03:41 2018
Thread-2: Mon Apr 23 10:03:43 2018
Thread-2: Mon Apr 23 10:03:47 2018
Thread-2: Mon Apr 23 10:03:51 2018

<थ्रेडिंग> मॉड्यूल

<threading>मॉड्यूल ऑब्जेक्ट ओरिएंटेड तरीके से लागू होता है और प्रत्येक थ्रेड को ऑब्जेक्ट के रूप में मानता है। इसलिए, यह <_thread> मॉड्यूल की तुलना में थ्रेड्स के लिए अधिक शक्तिशाली, उच्च-स्तरीय समर्थन प्रदान करता है। यह मॉड्यूल Python 2.4 के साथ शामिल है।

<थ्रेडिंग> मॉड्यूल में अतिरिक्त तरीके

<threading> मॉड्यूल के सभी तरीकों को शामिल करता है <_thread>मॉड्यूल लेकिन यह अतिरिक्त तरीके भी प्रदान करता है। अतिरिक्त तरीके इस प्रकार हैं -

  • threading.activeCount() - यह विधि सक्रिय होने वाले थ्रेड ऑब्जेक्ट्स की संख्या लौटाती है

  • threading.currentThread() - यह विधि कॉलर के थ्रेड कंट्रोल में थ्रेड ऑब्जेक्ट्स की संख्या लौटाती है।

  • threading.enumerate() - यह विधि उन सभी थ्रेड ऑब्जेक्ट्स की सूची देता है जो वर्तमान में सक्रिय हैं।

  • थ्रेडिंग को लागू करने के लिए, <threading> मॉड्यूल है Thread वर्ग जो निम्नलिखित तरीके प्रदान करता है -

    • run() - रन () विधि एक थ्रेड के लिए प्रवेश बिंदु है।

    • start() - रन विधि को कॉल करके स्टार्ट () विधि एक धागा शुरू करती है।

    • join([time]) - द ज्वाइन () थ्रेड्स के समाप्त होने का इंतजार करता है।

    • isAlive() - isAlive () विधि जांचती है कि क्या कोई थ्रेड अभी भी निष्पादित हो रहा है।

    • getName() - getName () विधि एक थ्रेड का नाम देता है।

    • setName() - सेटनाम () विधि एक थ्रेड का नाम सेट करती है।

<थ्रेडिंग> मॉड्यूल का उपयोग करके थ्रेड कैसे बनाएं?

इस भाग में, हम सीखेंगे कि थ्रेड्स का उपयोग कैसे करें <threading>मापांक। <थ्रेडिंग> मॉड्यूल का उपयोग करके एक नया धागा बनाने के लिए इन चरणों का पालन करें -

  • Step 1 - इस चरण में, हमें एक नए उपवर्ग को परिभाषित करने की आवश्यकता है Thread कक्षा।

  • Step 2 - फिर अतिरिक्त तर्क जोड़ने के लिए, हमें ओवरराइड करने की आवश्यकता है __init__(self [,args]) तरीका।

  • Step 3 - इस चरण में, हमें रन (स्वयं [, args]) विधि को ओवरराइड करने की आवश्यकता है ताकि थ्रेड शुरू होने पर क्या करना चाहिए।

  • अब, नया बनाने के बाद Thread उपवर्ग, हम इसका एक उदाहरण बना सकते हैं और फिर शुरू करके एक नया सूत्र शुरू कर सकते हैं start(), जो बदले में कॉल करता है run() तरीका।

उदाहरण

कैसे का उपयोग करके एक नया धागा उत्पन्न करने के लिए सीखने के लिए इस उदाहरण पर विचार करें <threading> मापांक।

import threading
import time
exitFlag = 0

class myThread (threading.Thread):
   def __init__(self, threadID, name, counter):
      threading.Thread.__init__(self)
      self.threadID = threadID
      self.name = name
      self.counter = counter
   def run(self):
      print ("Starting " + self.name)
      print_time(self.name, self.counter, 5)
      print ("Exiting " + self.name)
def print_time(threadName, delay, counter):
   while counter:
      if exitFlag:
         threadName.exit()
      time.sleep(delay)
      print ("%s: %s" % (threadName, time.ctime(time.time())))
      counter -= 1

thread1 = myThread(1, "Thread-1", 1)
thread2 = myThread(2, "Thread-2", 2)

thread1.start()
thread2.start()
thread1.join()
thread2.join()
print ("Exiting Main Thread")
Starting Thread-1
Starting Thread-2

उत्पादन

अब, निम्नलिखित आउटपुट पर विचार करें -

Thread-1: Mon Apr 23 10:52:09 2018
Thread-1: Mon Apr 23 10:52:10 2018
Thread-2: Mon Apr 23 10:52:10 2018
Thread-1: Mon Apr 23 10:52:11 2018
Thread-1: Mon Apr 23 10:52:12 2018
Thread-2: Mon Apr 23 10:52:12 2018
Thread-1: Mon Apr 23 10:52:13 2018
Exiting Thread-1
Thread-2: Mon Apr 23 10:52:14 2018
Thread-2: Mon Apr 23 10:52:16 2018
Thread-2: Mon Apr 23 10:52:18 2018
Exiting Thread-2
Exiting Main Thread

विभिन्न थ्रेड स्टेट्स के लिए पायथन प्रोग्राम

पांच थ्रेड स्टेट्स हैं - नया, रननेबल, रनिंग, वेटिंग और डेड। इन पांच में से पांच में, हम प्रमुख रूप से तीन राज्यों - रनिंग, वेटिंग और डेड पर ध्यान केंद्रित करेंगे। एक थ्रेड चल रहे राज्य में अपने संसाधनों को प्राप्त करता है, प्रतीक्षा स्थिति में संसाधनों की प्रतीक्षा करता है; संसाधन की अंतिम रिलीज, यदि निष्पादित और अधिग्रहित मृत अवस्था में है।

प्रारंभ (), स्लीप () और ज्वाइन () विधियों की मदद से निम्नलिखित पायथन कार्यक्रम क्रमशः दिखाएगा कि कैसे एक धागा क्रमशः चलने, प्रतीक्षा करने और मृत अवस्था में प्रवेश किया।

Step 1 - आवश्यक मॉड्यूल, <सूत्रण> और <समय> आयात करें

import threading
import time

Step 2 - एक फ़ंक्शन को परिभाषित करें, जिसे थ्रेड बनाते समय कहा जाएगा।

def thread_states():
   print("Thread entered in running state")

Step 3 - हम अपने थ्रेड को 2 सेकंड के लिए प्रतीक्षा करने के लिए समय मॉड्यूल की नींद () विधि का उपयोग कर रहे हैं।

time.sleep(2)

Step 4 - अब, हम T1 नाम से एक थ्रेड बना रहे हैं, जो ऊपर परिभाषित फ़ंक्शन का तर्क लेता है।

T1 = threading.Thread(target=thread_states)

Step 5- अब, स्टार्ट () फंक्शन की मदद से हम अपना धागा शुरू कर सकते हैं। यह संदेश का उत्पादन करेगा, जो फ़ंक्शन को परिभाषित करते समय हमारे द्वारा निर्धारित किया गया है।

T1.start()
Thread entered in running state

Step 6 - अब, अंत में हम थ्रेड को मार सकते हैं (ज्वाइन) विधि के बाद जब वह अपना निष्पादन समाप्त कर लेता है।

T1.join()

पायथन में एक धागा शुरू करना

अजगर में, हम अलग-अलग तरीकों से एक नया धागा शुरू कर सकते हैं लेकिन उनमें से सबसे आसान एक एकल फ़ंक्शन के रूप में परिभाषित करना है। फ़ंक्शन को परिभाषित करने के बाद, हम इसे नए के लिए लक्ष्य के रूप में पारित कर सकते हैंthreading.Threadवस्तु वगैरह। फ़ंक्शन कैसे काम करता है यह समझने के लिए निम्नलिखित पायथन कोड निष्पादित करें -

import threading
import time
import random
def Thread_execution(i):
   print("Execution of Thread {} started\n".format(i))
   sleepTime = random.randint(1,4)
   time.sleep(sleepTime)
   print("Execution of Thread {} finished".format(i))
for i in range(4):
   thread = threading.Thread(target=Thread_execution, args=(i,))
   thread.start()
   print("Active Threads:" , threading.enumerate())

उत्पादन

Execution of Thread 0 started
Active Threads:
   [<_MainThread(MainThread, started 6040)>,
      <HistorySavingThread(IPythonHistorySavingThread, started 5968)>,
      <Thread(Thread-3576, started 3932)>]

Execution of Thread 1 started
Active Threads:
   [<_MainThread(MainThread, started 6040)>,
      <HistorySavingThread(IPythonHistorySavingThread, started 5968)>,
      <Thread(Thread-3576, started 3932)>,
      <Thread(Thread-3577, started 3080)>]

Execution of Thread 2 started
Active Threads:
   [<_MainThread(MainThread, started 6040)>,
      <HistorySavingThread(IPythonHistorySavingThread, started 5968)>,
      <Thread(Thread-3576, started 3932)>,
      <Thread(Thread-3577, started 3080)>,
      <Thread(Thread-3578, started 2268)>]

Execution of Thread 3 started
Active Threads:
   [<_MainThread(MainThread, started 6040)>,
      <HistorySavingThread(IPythonHistorySavingThread, started 5968)>,
      <Thread(Thread-3576, started 3932)>,
      <Thread(Thread-3577, started 3080)>,
      <Thread(Thread-3578, started 2268)>,
      <Thread(Thread-3579, started 4520)>]
Execution of Thread 0 finished
Execution of Thread 1 finished
Execution of Thread 2 finished
Execution of Thread 3 finished

पायथन में डेमन थ्रेड्स

पायथन में डेमन थ्रेड्स को लागू करने से पहले, हमें डेमन थ्रेड्स और उनके उपयोग के बारे में जानना होगा। कंप्यूटिंग के संदर्भ में, डेमन एक पृष्ठभूमि प्रक्रिया है, जो विभिन्न सेवाओं जैसे डेटा भेजने, फ़ाइल स्थानांतरण आदि के अनुरोधों को संभालती है, यदि यह अधिक आवश्यक नहीं है तो यह निष्क्रिय हो जाएगा। एक ही कार्य गैर-डेमॉन थ्रेड्स की सहायता से भी किया जा सकता है। हालांकि, इस मामले में, मुख्य धागे को गैर-डेमॉन थ्रेड्स को मैन्युअल रूप से ट्रैक करना होगा। दूसरी ओर, यदि हम डेमन थ्रेड्स का उपयोग कर रहे हैं, तो मुख्य धागा इस बारे में पूरी तरह से भूल सकता है और मुख्य थ्रेड के बाहर निकलने पर इसे मार दिया जाएगा। डेमन थ्रेड्स के बारे में एक और महत्वपूर्ण बात यह है कि हम उन्हें केवल गैर-आवश्यक कार्यों के लिए उपयोग करने का विकल्प चुन सकते हैं जो हमें प्रभावित नहीं करेगा अगर यह पूरा नहीं होता है या बीच में मारा नहीं जाता है। अजगर में डेमन थ्रेड्स का कार्यान्वयन निम्नलिखित है -

import threading
import time

def nondaemonThread():
   print("starting my thread")
   time.sleep(8)
   print("ending my thread")
def daemonThread():
   while True:
   print("Hello")
   time.sleep(2)
if __name__ == '__main__':
   nondaemonThread = threading.Thread(target = nondaemonThread)
   daemonThread = threading.Thread(target = daemonThread)
   daemonThread.setDaemon(True)
   daemonThread.start()
   nondaemonThread.start()

उपरोक्त कोड में, दो कार्य हैं >nondaemonThread() तथा >daemonThread()। पहला फ़ंक्शन अपने राज्य को प्रिंट करता है और 8 सेकंड के बाद सो जाता है, जबकि deamonTread () फ़ंक्शन प्रिंट को हर 2 सेकंड के बाद अनिश्चित काल तक नमस्कार करता है। हम निम्न आउटपुट की मदद से नोंडोमन ​​और डेमन थ्रेड्स के बीच अंतर को समझ सकते हैं -

Hello

starting my thread
Hello
Hello
Hello
Hello
ending my thread
Hello
Hello
Hello
Hello
Hello

थ्रेड सिंक्रोनाइज़ेशन को एक ऐसी विधि के रूप में परिभाषित किया जा सकता है जिसकी मदद से हमें यह आश्वासन दिया जा सकता है कि दो या अधिक समवर्ती धागे एक साथ महत्वपूर्ण खंड के रूप में जाना जाने वाले प्रोग्राम सेगमेंट तक नहीं पहुंच रहे हैं। दूसरी ओर, जैसा कि हम जानते हैं कि महत्वपूर्ण खंड उस कार्यक्रम का हिस्सा है जहां साझा संसाधन तक पहुँचा जाता है। इसलिए हम कह सकते हैं कि सिंक्रनाइजेशन यह सुनिश्चित करने की प्रक्रिया है कि दो या दो से अधिक धागे एक ही समय में संसाधनों तक पहुंच कर एक-दूसरे के साथ इंटरफेस नहीं करते हैं। नीचे दिए गए आरेख से पता चलता है कि चार धागे एक ही समय में एक कार्यक्रम के महत्वपूर्ण खंड तक पहुंचने की कोशिश कर रहे हैं।

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

थ्रेड सिंक्रनाइज़ेशन में समस्याएँ

समवर्ती प्रोग्रामिंग को लागू करते समय या आदिम सिंक्रनाइज़ेशन को लागू करते समय हम समस्याओं का सामना कर सकते हैं। इस भाग में, हम दो प्रमुख मुद्दों पर चर्चा करेंगे। मुद्दे हैं -

  • Deadlock
  • दौड़ की स्थिति

दौड़ की स्थिति

यह समवर्ती प्रोग्रामिंग में प्रमुख मुद्दों में से एक है। साझा संसाधनों के समवर्ती पहुंच से दौड़ की स्थिति बन सकती है। एक दौड़ की स्थिति को एक ऐसी स्थिति के रूप में परिभाषित किया जा सकता है जब दो या अधिक थ्रेड साझा किए गए डेटा तक पहुंच सकते हैं और फिर उसी समय इसके मूल्य को बदलने की कोशिश कर सकते हैं। इसके कारण, चर के मान अप्रत्याशित हो सकते हैं और प्रक्रियाओं के संदर्भ स्विच के समय के आधार पर भिन्न हो सकते हैं।

उदाहरण

दौड़ की स्थिति की अवधारणा को समझने के लिए इस उदाहरण पर विचार करें -

Step 1 - इस चरण में, हमें सूत्रण मॉड्यूल आयात करने की आवश्यकता है -

import threading

Step 2 - अब, एक वैश्विक चर को परिभाषित करें, मान लें कि x, इसके मान के साथ 0 -

x = 0

Step 3 - अब, हमें परिभाषित करने की आवश्यकता है increment_global() फ़ंक्शन, जो इस वैश्विक फ़ंक्शन x में 1 से वेतन वृद्धि करेगा -

def increment_global():

   global x
   x += 1

Step 4 - इस चरण में, हम परिभाषित करेंगे taskofThread()फ़ंक्शन, जो समय की एक निर्दिष्ट संख्या के लिए increment_global () फ़ंक्शन को कॉल करेगा; हमारे उदाहरण के लिए यह 50000 गुना है -

def taskofThread():

   for _ in range(50000):
      increment_global()

Step 5- अब, मुख्य () फ़ंक्शन को परिभाषित करें जिसमें थ्रेड्स t1 और t2 बनाए गए हैं। दोनों को स्टार्ट () फ़ंक्शन की सहायता से शुरू किया जाएगा और जब तक वे जॉइन () फ़ंक्शन की सहायता से अपनी नौकरी समाप्त नहीं करते तब तक प्रतीक्षा करें।

def main():
   global x
   x = 0
   
   t1 = threading.Thread(target= taskofThread)
   t2 = threading.Thread(target= taskofThread)

   t1.start()
   t2.start()

   t1.join()
   t2.join()

Step 6- अब, हमें सीमा देने की आवश्यकता है कि हम कितने पुनरावृत्तियों को मुख्य () फ़ंक्शन को कॉल करना चाहते हैं। यहां, हम इसे 5 बार कॉल कर रहे हैं।

if __name__ == "__main__":
   for i in range(5):
      main()
      print("x = {1} after Iteration {0}".format(i,x))

नीचे दिखाए गए आउटपुट में, हम रेस की स्थिति के प्रभाव को देख सकते हैं क्योंकि प्रत्येक पुनरावृत्ति 100000 की उम्मीद के बाद x का मान है। हालांकि, मूल्य में बहुत भिन्नता है। यह साझा वैश्विक चर x के लिए थ्रेड्स के समवर्ती उपयोग के कारण है।

उत्पादन

x = 100000 after Iteration 0
x = 54034 after Iteration 1
x = 80230 after Iteration 2
x = 93602 after Iteration 3
x = 93289 after Iteration 4

ताले का उपयोग कर दौड़ की स्थिति से निपटना

जैसा कि हमने उपरोक्त कार्यक्रम में दौड़ की स्थिति के प्रभाव को देखा है, हमें एक सिंक्रनाइज़ेशन टूल की आवश्यकता है, जो कई थ्रेड्स के बीच दौड़ की स्थिति से निपट सकता है। पायथन में, द<threading>मॉड्यूल रेस की स्थिति से निपटने के लिए लॉक क्लास प्रदान करता है। इसके अलावा,Lockवर्ग विभिन्न विधियाँ प्रदान करता है जिनकी सहायता से हम कई थ्रेड्स के बीच दौड़ की स्थिति को संभाल सकते हैं। विधियाँ नीचे वर्णित हैं -

अधिग्रहण () विधि

इस विधि का उपयोग ताला प्राप्त करने के लिए किया जाता है, अर्थात। निम्नलिखित सही या गलत मूल्य के आधार पर एक लॉक को अवरुद्ध या गैर-अवरुद्ध किया जा सकता है -

  • With value set to True - यदि अधिग्रहण () विधि ट्रू के साथ लगाई गई है, जो कि डिफ़ॉल्ट तर्क है, तो थ्रेड निष्पादन तब तक अवरुद्ध हो जाता है जब तक कि लॉक अनलॉक न हो जाए।

  • With value set to False - यदि अधिग्रहित () विधि को गलत के साथ लागू किया जाता है, जो डिफ़ॉल्ट तर्क नहीं है, तो थ्रेड निष्पादन तब तक अवरुद्ध नहीं होता है जब तक कि यह सही पर सेट न हो, अर्थात, जब तक कि यह लॉक न हो।

रिलीज () विधि

इस विधि का उपयोग ताला जारी करने के लिए किया जाता है। इस विधि से संबंधित कुछ महत्वपूर्ण कार्य निम्नलिखित हैं -

  • यदि कोई ताला बंद है, तो ताला release()विधि यह अनलॉक होगा। इसका काम वास्तव में एक धागे को आगे बढ़ने की अनुमति देना है यदि एक से अधिक धागे अवरुद्ध हैं और लॉक के अनलॉक होने की प्रतीक्षा कर रहे हैं।

  • यह एक उठाएंगे ThreadError अगर ताला पहले से ही खुला है।

अब, हम दौड़ की स्थिति से बचने के लिए लॉक प्रोग्राम और इसके तरीकों के साथ उपरोक्त कार्यक्रम को फिर से लिख सकते हैं। हमें टास्क लॉफ़्रेड () पद्धति को लॉक तर्क के साथ परिभाषित करने की आवश्यकता है और फिर दौड़ की स्थिति से बचने के लिए लॉक को अवरुद्ध और गैर-अवरुद्ध करने के लिए अधिग्रहण () और रिलीज़ () तरीकों का उपयोग करने की आवश्यकता है।

उदाहरण

नस्ल की स्थिति से निपटने के लिए तालों की अवधारणा को समझने के लिए अजगर कार्यक्रम का उदाहरण निम्नलिखित है -

import threading

x = 0

def increment_global():

   global x
   x += 1

def taskofThread(lock):

   for _ in range(50000):
      lock.acquire()
      increment_global()
      lock.release()

def main():
   global x
   x = 0

   lock = threading.Lock()
   t1 = threading.Thread(target = taskofThread, args = (lock,))
   t2 = threading.Thread(target = taskofThread, args = (lock,))

   t1.start()
   t2.start()

   t1.join()
   t2.join()

if __name__ == "__main__":
   for i in range(5):
      main()
      print("x = {1} after Iteration {0}".format(i,x))

निम्न आउटपुट से पता चलता है कि दौड़ की स्थिति का प्रभाव उपेक्षित है; x के मान के रूप में, प्रत्येक पुनरावृत्ति के बाद, अब 100000 है, जो इस कार्यक्रम की अपेक्षा के अनुसार है।

उत्पादन

x = 100000 after Iteration 0
x = 100000 after Iteration 1
x = 100000 after Iteration 2
x = 100000 after Iteration 3
x = 100000 after Iteration 4

डेडलॉक - द डाइनिंग फिलॉसॉफ़र्स समस्या

डेडलॉक एक परेशानी वाला मुद्दा है जो समवर्ती प्रणालियों को डिजाइन करते समय सामना कर सकता है। हम इस मुद्दे को निम्नानुसार भोजन दार्शनिक समस्या की सहायता से बता सकते हैं -

एड्स्जर डेज्स्ट्रा ने मूल रूप से डाइनिंग दार्शनिक समस्या पेश की, जो गतिरोध नामक समवर्ती प्रणाली की सबसे बड़ी समस्या में से एक के प्रसिद्ध चित्रण में से एक है।

इस समस्या में, एक राउंड टेबल पर पांच प्रसिद्ध दार्शनिक बैठे हैं जो अपने कटोरे से कुछ खा रहे हैं। पांच कांटे हैं जिनका उपयोग पांच दार्शनिक अपने भोजन को खाने के लिए कर सकते हैं। हालांकि, दार्शनिक अपने भोजन को खाने के लिए एक ही समय में दो कांटे का उपयोग करने का निर्णय लेते हैं।

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

समवर्ती प्रणाली में गतिरोध

अब अगर हम देखें तो हमारे समवर्ती प्रणालियों में भी यही मुद्दा उठ सकता है। उपरोक्त उदाहरण में कांटे सिस्टम संसाधन होंगे और प्रत्येक दार्शनिक प्रक्रिया का प्रतिनिधित्व कर सकता है, जो संसाधनों को प्राप्त करने के लिए प्रतिस्पर्धा कर रहा है।

पायथन कार्यक्रम के साथ समाधान

दार्शनिकों को दो प्रकारों में विभाजित करके इस समस्या का हल खोजा जा सकता है - greedy philosophers तथा generous philosophers। मुख्य रूप से एक लालची दार्शनिक बाएं कांटा को लेने की कोशिश करेगा और तब तक इंतजार करेगा जब तक वह वहां न हो। वह तब सही कांटा होने का इंतजार करेगा, उसे उठाएगा, खाएगा और फिर नीचे रख देगा। दूसरी ओर, एक उदार दार्शनिक बाएं कांटे को लेने की कोशिश करेगा और अगर यह नहीं है, तो वह इंतजार करेगा और कुछ समय बाद फिर से प्रयास करेगा। यदि उन्हें बाएं कांटा मिलता है, तो वे दाएं पाने की कोशिश करेंगे। अगर उन्हें सही कांटा भी मिल जाएगा तो वे दोनों कांटे खा लेंगे और छोड़ देंगे। हालांकि, अगर उन्हें सही कांटा नहीं मिलेगा, तो वे बाएं कांटा जारी करेंगे।

उदाहरण

निम्नलिखित पायथन कार्यक्रम हमें भोजन दार्शनिक समस्या का समाधान खोजने में मदद करेगा -

import threading
import random
import time

class DiningPhilosopher(threading.Thread):

   running = True

   def __init__(self, xname, Leftfork, Rightfork):
   threading.Thread.__init__(self)
   self.name = xname
   self.Leftfork = Leftfork
   self.Rightfork = Rightfork

   def run(self):
   while(self.running):
      time.sleep( random.uniform(3,13))
      print ('%s is hungry.' % self.name)
      self.dine()

   def dine(self):
   fork1, fork2 = self.Leftfork, self.Rightfork

   while self.running:
      fork1.acquire(True)
      locked = fork2.acquire(False)
	  if locked: break
      fork1.release()
      print ('%s swaps forks' % self.name)
      fork1, fork2 = fork2, fork1
   else:
      return

   self.dining()
   fork2.release()
   fork1.release()

   def dining(self):
   print ('%s starts eating '% self.name)
   time.sleep(random.uniform(1,10))
   print ('%s finishes eating and now thinking.' % self.name)

def Dining_Philosophers():
   forks = [threading.Lock() for n in range(5)]
   philosopherNames = ('1st','2nd','3rd','4th', '5th')

   philosophers= [DiningPhilosopher(philosopherNames[i], forks[i%5], forks[(i+1)%5]) \
      for i in range(5)]

   random.seed()
   DiningPhilosopher.running = True
   for p in philosophers: p.start()
   time.sleep(30)
   DiningPhilosopher.running = False
   print (" It is finishing.")

Dining_Philosophers()

उपरोक्त कार्यक्रम लालची और उदार दार्शनिकों की अवधारणा का उपयोग करता है। कार्यक्रम का उपयोग भी किया हैacquire() तथा release() के तरीके Lock की कक्षा <threading>मापांक। हम निम्न आउटपुट में समाधान देख सकते हैं -

उत्पादन

4th is hungry.
4th starts eating
1st is hungry.
1st starts eating
2nd is hungry.
5th is hungry.
3rd is hungry.
1st finishes eating and now thinking.3rd swaps forks
2nd starts eating
4th finishes eating and now thinking.
3rd swaps forks5th starts eating
5th finishes eating and now thinking.
4th is hungry.
4th starts eating
2nd finishes eating and now thinking.
3rd swaps forks
1st is hungry.
1st starts eating
4th finishes eating and now thinking.
3rd starts eating
5th is hungry.
5th swaps forks
1st finishes eating and now thinking.
5th starts eating
2nd is hungry.
2nd swaps forks
4th is hungry.
5th finishes eating and now thinking.
3rd finishes eating and now thinking.
2nd starts eating 4th starts eating
It is finishing.

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

थ्रेड इंटर कम्यूनिकेशन से संबंधित निम्नलिखित महत्वपूर्ण बिंदुओं पर विचार करें -

  • No performance gain - यदि हम थ्रेड्स और प्रक्रियाओं के बीच उचित संचार प्राप्त नहीं कर सकते हैं, तो समवर्ती और समानांतरवाद से प्रदर्शन लाभ का कोई फायदा नहीं है।

  • Accomplish task properly - थ्रेड्स के बीच उचित अंतर-तंत्र के बिना, असाइन किए गए कार्य को ठीक से पूरा नहीं किया जा सकता है।

  • More efficient than inter-process communication - अंतर-थ्रेड संचार अंतर-प्रक्रिया संचार की तुलना में अधिक कुशल और उपयोग करने में आसान है क्योंकि एक प्रक्रिया के भीतर सभी थ्रेड्स समान पता स्थान साझा करते हैं और उन्हें साझा मेमोरी का उपयोग करने की आवश्यकता नहीं होती है।

धागा-सुरक्षित संचार के लिए पायथन डेटा संरचनाएं

Multithreaded कोड एक थ्रेड से दूसरे थ्रेड तक जानकारी पास करने की समस्या के साथ आता है। मानक संचार प्राइमेटिव इस समस्या को हल नहीं करते हैं। इसलिए, हमें अपने स्वयं के समग्र ऑब्जेक्ट को लागू करने की आवश्यकता है ताकि संचार धागे को सुरक्षित बनाने के लिए धागे के बीच की वस्तुओं को साझा किया जा सके। निम्नलिखित कुछ डेटा संरचनाएं हैं, जो उनमें कुछ बदलाव करने के बाद थ्रेड-सुरक्षित संचार प्रदान करती हैं -

सेट

थ्रेड-सुरक्षित तरीके से सेट डेटा संरचना का उपयोग करने के लिए, हमें अपने स्वयं के लॉकिंग तंत्र को लागू करने के लिए सेट क्लास का विस्तार करने की आवश्यकता है।

उदाहरण

यहाँ कक्षा को विस्तार देने का एक पायथन उदाहरण है -

class extend_class(set):
   def __init__(self, *args, **kwargs):
      self._lock = Lock()
      super(extend_class, self).__init__(*args, **kwargs)

   def add(self, elem):
      self._lock.acquire()
	  try:
      super(extend_class, self).add(elem)
      finally:
      self._lock.release()
  
   def delete(self, elem):
      self._lock.acquire()
      try:
      super(extend_class, self).delete(elem)
      finally:
      self._lock.release()

उपरोक्त उदाहरण में, एक वर्ग वस्तु जिसका नाम है extend_class परिभाषित किया गया है जो आगे पायथन से विरासत में मिला है set class। इस वर्ग के निर्माता के भीतर एक लॉक ऑब्जेक्ट बनाया जाता है। अब, दो कार्य हैं -add() तथा delete()। ये फ़ंक्शन परिभाषित हैं और थ्रेड-सुरक्षित हैं। वे दोनों पर भरोसा करते हैंsuper एक प्रमुख अपवाद के साथ वर्ग कार्यक्षमता।

डेकोरेटर

यह थ्रेड-सुरक्षित संचार के लिए एक और महत्वपूर्ण विधि है, सज्जाकारों का उपयोग।

उदाहरण

एक पायथन उदाहरण पर विचार करें जो दिखाता है कि सज्जाकार और मिमीिनस का उपयोग कैसे करें;

def lock_decorator(method):

   def new_deco_method(self, *args, **kwargs):
      with self._lock:
         return method(self, *args, **kwargs)
return new_deco_method

class Decorator_class(set):
   def __init__(self, *args, **kwargs):
      self._lock = Lock()
      super(Decorator_class, self).__init__(*args, **kwargs)

   @lock_decorator
   def add(self, *args, **kwargs):
      return super(Decorator_class, self).add(elem)
   @lock_decorator
   def delete(self, *args, **kwargs):
      return super(Decorator_class, self).delete(elem)

उपरोक्त उदाहरण में, lock_decorator नामक एक डेकोरेटर विधि को परिभाषित किया गया है जो आगे पायथन विधि वर्ग से विरासत में मिली है। फिर इस वर्ग के निर्माता के भीतर एक लॉक ऑब्जेक्ट बनाया जाता है। अब, दो कार्य हैं - जोड़ें () और हटाएं ()। ये फ़ंक्शन परिभाषित हैं और थ्रेड-सुरक्षित हैं। वे दोनों एक प्रमुख अपवाद के साथ सुपर क्लास कार्यक्षमता पर भरोसा करते हैं।

सूचियों

सूची डेटा संरचना थ्रेड-सुरक्षित है, साथ ही अस्थायी, इन-मेमोरी मेमोरी के लिए आसान संरचना है। Cpython में, GIL उनके लिए समवर्ती पहुँच से बचाता है। जैसा कि हमें पता चला है कि सूचियाँ थ्रेड-सुरक्षित हैं लेकिन उनमें पड़े डेटा का क्या। दरअसल, सूची का डेटा संरक्षित नहीं है। उदाहरण के लिए,L.append(x)अपेक्षित परिणाम वापस करने की गारंटी नहीं है यदि एक और धागा एक ही काम करने की कोशिश कर रहा है। ऐसा इसलिए है, हालांकिappend() एक परमाणु संचालन और धागा-सुरक्षित है लेकिन दूसरा धागा समवर्ती फैशन में सूची के डेटा को संशोधित करने की कोशिश कर रहा है, इसलिए हम आउटपुट पर दौड़ की स्थिति के दुष्प्रभावों को देख सकते हैं।

इस तरह के मुद्दे को हल करने और डेटा को सुरक्षित रूप से संशोधित करने के लिए, हमें एक उचित लॉकिंग तंत्र को लागू करना चाहिए, जो आगे सुनिश्चित करता है कि कई थ्रेड्स संभवतः दौड़ की स्थिति में नहीं चल सकते। उचित लॉकिंग तंत्र को लागू करने के लिए, हम वर्ग का विस्तार कर सकते हैं जैसा कि हमने पिछले उदाहरणों में किया था।

सूचियों पर कुछ अन्य परमाणु संचालन इस प्रकार हैं -

L.append(x)
L1.extend(L2)
x = L[i]
x = L.pop()
L1[i:j] = L2
L.sort()
x = y
x.field = y
D[x] = y
D1.update(D2)
D.keys()

यहाँ -

  • एल, एल 1, एल 2 सभी सूचियां हैं
  • डी, डी 1, डी 2 डिकेट हैं
  • x, y ऑब्जेक्ट हैं
  • मैं, जे ints हैं

कतारों

यदि सूची का डेटा सुरक्षित नहीं है, तो हमें इसके परिणाम भुगतने पड़ सकते हैं। हम गलत डेटा आइटम प्राप्त कर सकते हैं, या हटा सकते हैं। यही कारण है कि कतार डेटा संरचना का उपयोग करने की सिफारिश की जाती है। कतार की वास्तविक दुनिया का उदाहरण सिंगल-लेन वन-वे रोड हो सकता है, जहां वाहन पहले प्रवेश करता है, पहले बाहर निकलता है। अधिक वास्तविक दुनिया के उदाहरण टिकट खिड़कियों और बस-स्टॉप पर कतारों के देखे जा सकते हैं।

कतारें डिफ़ॉल्ट रूप से, थ्रेड-सुरक्षित डेटा संरचना हैं और हमें जटिल लॉकिंग तंत्र को लागू करने के बारे में चिंता करने की आवश्यकता नहीं है। अजगर हमें प्रदान करता है हमारे आवेदन में विभिन्न प्रकार की कतारों का उपयोग करने के लिए मॉड्यूल।

कतारों के प्रकार

इस भाग में, हम विभिन्न प्रकार की कतारों के बारे में अर्जित करेंगे। पायथन से उपयोग करने के लिए कतारों के तीन विकल्प प्रदान करता है<queue> मॉड्यूल -

  • सामान्य कतारें (फीफो, फर्स्ट आउट इन फर्स्ट)
  • LIFO, लास्ट में फर्स्ट आउट
  • Priority

हम बाद के वर्गों में विभिन्न कतारों के बारे में जानेंगे।

सामान्य कतारें (फीफो, फर्स्ट आउट इन फर्स्ट)

यह पायथन द्वारा की पेशकश की सबसे अधिक इस्तेमाल किया कतार कार्यान्वयन है। इस कतार तंत्र में जो भी पहले आएगा, पहले सेवा प्राप्त करेगा। FIFO को सामान्य कतार भी कहा जाता है। FIFO कतारों का प्रतिनिधित्व इस प्रकार किया जा सकता है -

फीफो कतार के पायथन कार्यान्वयन

अजगर में, एफआईएफओ कतार को एकल धागे के साथ-साथ मल्टीथ्रेड्स के साथ लागू किया जा सकता है।

एकल धागे के साथ फीफो कतार

एकल धागे के साथ फीफो कतार को लागू करने के लिए, Queueक्लास एक पहले फर्स्ट-इन-आउट कंटेनर को लागू करेगा। तत्वों का उपयोग करके अनुक्रम के एक "अंत" में जोड़ा जाएगाput(), और दूसरे छोर से उपयोग करके हटा दिया गया get()

उदाहरण

निम्नलिखित एकल धागा के साथ फीफो कतार के कार्यान्वयन के लिए पायथन कार्यक्रम है -

import queue

q = queue.Queue()

for i in range(8):
   q.put("item-" + str(i))

while not q.empty():
   print (q.get(), end = " ")

उत्पादन

item-0 item-1 item-2 item-3 item-4 item-5 item-6 item-7

आउटपुट से पता चलता है कि उपरोक्त प्रोग्राम यह बताने के लिए एक एकल धागे का उपयोग करता है कि तत्वों को कतार से उसी क्रम में हटा दिया जाता है जिस क्रम में उन्हें डाला जाता है।

कई धागों के साथ फीफो कतार

कई थ्रेड्स के साथ FIFO को लागू करने के लिए, हमें myqueue () फ़ंक्शन को परिभाषित करने की आवश्यकता है, जो कतार मॉड्यूल से विस्तारित है। सिंगल थ्रेड के साथ फीफो कतार को लागू करते समय प्राप्त () और पुट () विधियों की कार्यप्रणाली ऊपर चर्चा की गई है। फिर इसे मल्टीथ्रेडेड बनाने के लिए, हमें थ्रेड्स को घोषित करने और उन्हें तत्काल करने की आवश्यकता है। ये धागे FIFO तरीके से कतार का उपभोग करेंगे।

उदाहरण

निम्नलिखित कई धागे के साथ फीफो कतार के कार्यान्वयन के लिए पायथन कार्यक्रम है

import threading
import queue
import random
import time
def myqueue(queue):
   while not queue.empty():
   item = queue.get()
   if item is None:
   break
   print("{} removed {} from the queue".format(threading.current_thread(), item))
   queue.task_done()
   time.sleep(2)
q = queue.Queue()
for i in range(5):
   q.put(i)
threads = []
for i in range(4):
   thread = threading.Thread(target=myqueue, args=(q,))
   thread.start()
   threads.append(thread)
for thread in threads:
   thread.join()

उत्पादन

<Thread(Thread-3654, started 5044)> removed 0 from the queue
<Thread(Thread-3655, started 3144)> removed 1 from the queue
<Thread(Thread-3656, started 6996)> removed 2 from the queue
<Thread(Thread-3657, started 2672)> removed 3 from the queue
<Thread(Thread-3654, started 5044)> removed 4 from the queue

LIFO, पहले आउट कतार में अंतिम

यह कतार FIFO (फर्स्ट इन फर्स्ट आउट) कतारों की तुलना में बिलकुल विपरीत उपमा का उपयोग करती है। इस कतार तंत्र में, जो अंतिम आएगा, उसे पहले सेवा मिलेगी। यह स्टैक डेटा संरचना को लागू करने के समान है। कृत्रिम बुद्धिमत्ता के एल्गोरिदम की तरह गहराई-पहली खोज को लागू करते समय LIFO कतार उपयोगी साबित होती है।

LIFO कतार का पायथन कार्यान्वयन

अजगर में, LIFO कतार को एकल धागे के साथ-साथ मल्टीथ्रेड के साथ लागू किया जा सकता है।

एकल धागे के साथ LIFO कतार

एकल धागे के साथ LIFO कतार को लागू करने के लिए, Queue क्लास संरचना का उपयोग करके एक बुनियादी अंतिम-इन, पहले-आउट कंटेनर को लागू करेगा Queue.LifoQueue। अब, बुलाने परput()तत्वों को कंटेनर के सिर में जोड़ा जाता है और उपयोग करने पर सिर से हटा दिया जाता है get()

उदाहरण

निम्नलिखित एकल धागा के साथ LIFO कतार के कार्यान्वयन के लिए पायथन कार्यक्रम है -

import queue

q = queue.LifoQueue()

for i in range(8):
   q.put("item-" + str(i))

while not q.empty():
   print (q.get(), end=" ")
Output:
item-7 item-6 item-5 item-4 item-3 item-2 item-1 item-0

आउटपुट से पता चलता है कि उपरोक्त प्रोग्राम यह बताने के लिए एकल थ्रेड का उपयोग करता है कि तत्वों को कतार में उस विपरीत क्रम से हटा दिया जाता है जिसे वे सम्मिलित करते हैं।

कई धागों के साथ LIFO कतार

कार्यान्वयन समान है जैसा कि हमने कई थ्रेड्स के साथ एफआईएफओ कतारों का कार्यान्वयन किया है। अंतर केवल इतना है कि हमें इसका उपयोग करने की आवश्यकता हैQueue वर्ग जो संरचना का उपयोग करके एक बुनियादी अंतिम-इन, पहले-आउट कंटेनर को लागू करेगा Queue.LifoQueue

उदाहरण

निम्नलिखित कई धागों के साथ LIFO कतार के कार्यान्वयन के लिए पायथन कार्यक्रम है -

import threading
import queue
import random
import time
def myqueue(queue):
   while not queue.empty():
      item = queue.get()
      if item is None:
      break
	  print("{} removed {} from the queue".format(threading.current_thread(), item))
      queue.task_done()
      time.sleep(2)
q = queue.LifoQueue()
for i in range(5):
   q.put(i)
threads = []
for i in range(4):
   thread = threading.Thread(target=myqueue, args=(q,))
   thread.start()
   threads.append(thread)
for thread in threads:
   thread.join()

उत्पादन

<Thread(Thread-3882, started 4928)> removed 4 from the queue
<Thread(Thread-3883, started 4364)> removed 3 from the queue
<Thread(Thread-3884, started 6908)> removed 2 from the queue
<Thread(Thread-3885, started 3584)> removed 1 from the queue
<Thread(Thread-3882, started 4928)> removed 0 from the queue

प्राथमिकता कतार

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

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

प्राथमिकता कतार का पायथन कार्यान्वयन

अजगर में, प्राथमिकता कतार को एकल धागे के साथ-साथ मल्टीथ्रेड्स के साथ लागू किया जा सकता है।

एकल धागे के साथ प्राथमिकता कतार

एकल धागे के साथ प्राथमिकता कतार को लागू करने के लिए, Queue वर्ग संरचना का उपयोग करके प्राथमिकता कंटेनर पर एक कार्य को लागू करेगा Queue।प्राथमिकता कतार। अब, बुलाने परput()तत्वों को एक मूल्य के साथ जोड़ा जाता है जहां सबसे कम मूल्य की सर्वोच्च प्राथमिकता होगी और इसलिए पहले उपयोग करके पुनर्प्राप्त किया जाएगा get()

उदाहरण

एकल धागे के साथ प्राथमिकता कतार के कार्यान्वयन के लिए निम्नलिखित पायथन कार्यक्रम पर विचार करें -

import queue as Q
p_queue = Q.PriorityQueue()

p_queue.put((2, 'Urgent'))
p_queue.put((1, 'Most Urgent'))
p_queue.put((10, 'Nothing important'))
prio_queue.put((5, 'Important'))

while not p_queue.empty():
   item = p_queue.get()
   print('%s - %s' % item)

उत्पादन

1 – Most Urgent
2 - Urgent
5 - Important
10 – Nothing important

उपरोक्त आउटपुट में, हम देख सकते हैं कि कतार ने प्राथमिकता के आधार पर वस्तुओं को संग्रहीत किया है - कम मूल्य में उच्च प्राथमिकता है।

बहु सूत्र के साथ प्राथमिकता कतार

कार्यान्वयन कई धागों के साथ FIFO और LIFO कतारों के कार्यान्वयन के समान है। अंतर केवल इतना है कि हमें इसका उपयोग करने की आवश्यकता हैQueue संरचना का उपयोग करके प्राथमिकता को प्रारंभिक करने के लिए वर्ग Queue.PriorityQueue। एक और अंतर कतार के उत्पन्न होने के तरीके के साथ है। नीचे दिए गए उदाहरण में, यह दो समान डेटा सेट के साथ उत्पन्न होगा।

उदाहरण

निम्नलिखित पायथन कार्यक्रम कई धागों के साथ प्राथमिकता कतार के कार्यान्वयन में मदद करता है -

import threading
import queue
import random
import time
def myqueue(queue):
   while not queue.empty():
      item = queue.get()
      if item is None:
      break
      print("{} removed {} from the queue".format(threading.current_thread(), item))
      queue.task_done()
      time.sleep(1)
q = queue.PriorityQueue()
for i in range(5):
   q.put(i,1)

for i in range(5):
   q.put(i,1)

threads = []
for i in range(2):
   thread = threading.Thread(target=myqueue, args=(q,))
   thread.start()
   threads.append(thread)
for thread in threads:
   thread.join()

उत्पादन

<Thread(Thread-4939, started 2420)> removed 0 from the queue
<Thread(Thread-4940, started 3284)> removed 0 from the queue
<Thread(Thread-4939, started 2420)> removed 1 from the queue
<Thread(Thread-4940, started 3284)> removed 1 from the queue
<Thread(Thread-4939, started 2420)> removed 2 from the queue
<Thread(Thread-4940, started 3284)> removed 2 from the queue
<Thread(Thread-4939, started 2420)> removed 3 from the queue
<Thread(Thread-4940, started 3284)> removed 3 from the queue
<Thread(Thread-4939, started 2420)> removed 4 from the queue
<Thread(Thread-4940, started 3284)> removed 4 from the queue

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

टेस्ट क्यों?

इससे पहले कि हम परीक्षण के महत्व के बारे में चर्चा में गोता लगाएँ, हमें यह जानना होगा कि परीक्षण क्या है। सामान्य शब्दों में, परीक्षण यह पता लगाने की तकनीक है कि कोई चीज़ कितनी अच्छी तरह काम कर रही है। दूसरी ओर, विशेष रूप से अगर हम कंप्यूटर प्रोग्राम या सॉफ्टवेयर के बारे में बात करते हैं तो परीक्षण एक सॉफ्टवेयर प्रोग्राम की कार्यक्षमता तक पहुंचने की तकनीक है।

इस खंड में, हम सॉफ्टवेयर परीक्षण के महत्व पर चर्चा करेंगे। सॉफ़्टवेयर डेवलपमेंट में, क्लाइंट को सॉफ़्टवेयर रिलीज़ करने से पहले डबल-चेकिंग होनी चाहिए। इसलिए अनुभवी परीक्षण टीम द्वारा सॉफ्टवेयर का परीक्षण करना बहुत महत्वपूर्ण है। सॉफ्टवेयर परीक्षण के महत्व को समझने के लिए निम्नलिखित बिंदुओं पर विचार करें -

सॉफ्टवेयर की गुणवत्ता में सुधार

निश्चित रूप से, कोई भी कंपनी कम गुणवत्ता वाला सॉफ्टवेयर नहीं देना चाहती है और कोई भी ग्राहक कम गुणवत्ता वाला सॉफ्टवेयर नहीं खरीदना चाहता है। परीक्षण में बग को खोजने और ठीक करने से सॉफ्टवेयर की गुणवत्ता में सुधार होता है।

ग्राहकों की संतुष्टि

किसी भी व्यवसाय का सबसे महत्वपूर्ण हिस्सा उनके ग्राहकों की संतुष्टि है। बग मुक्त और अच्छी गुणवत्ता वाले सॉफ़्टवेयर प्रदान करके कंपनियां ग्राहकों की संतुष्टि प्राप्त कर सकती हैं।

नई सुविधाओं के प्रभाव को कम

मान लीजिए कि हमने 10000 लाइनों का एक सॉफ्टवेयर सिस्टम बनाया है और हमें एक नई सुविधा जोड़ने की आवश्यकता है तो विकास टीम को इस नए फीचर के पूरे सॉफ्टवेयर पर प्रभाव की चिंता होगी। यहां, परीक्षण भी एक महत्वपूर्ण भूमिका निभाता है क्योंकि यदि परीक्षण टीम ने परीक्षणों का एक अच्छा सूट बनाया है तो यह हमें किसी भी संभावित विनाशकारी विराम से बचा सकता है।

प्रयोगकर्ता का अनुभव

किसी भी व्यवसाय का एक और सबसे महत्वपूर्ण हिस्सा उस उत्पाद के उपयोगकर्ताओं का अनुभव है। केवल परीक्षण यह आश्वासन दे सकता है कि अंतिम उपयोगकर्ता को उत्पाद का उपयोग करना सरल और आसान लगता है।

खर्चों में कटौती

परीक्षण सॉफ्टवेयर की कुल लागत को डिलीवरी के बाद ठीक करने के बजाय इसके विकास के परीक्षण चरण में बग्स को खोजने और तय करने में कटौती कर सकता है। यदि सॉफ्टवेयर की डिलीवरी के बाद एक प्रमुख बग है तो यह खर्चों के संदर्भ में अपनी मूर्त लागत को बढ़ाएगा और ग्राहक की असंतोष, कंपनी की नकारात्मक प्रतिष्ठा आदि के संदर्भ में अमूर्त लागत कहेगा।

क्या टेस्ट करें?

परीक्षण करने के लिए हमेशा उपयुक्त ज्ञान रखने की सलाह दी जाती है। इस खंड में, हम पहले किसी भी सॉफ़्टवेयर का परीक्षण करते समय परीक्षक के मुख्य उद्देश्य को समझेंगे। कोड कवरेज, यानी, परीक्षण करते समय हमारे परीक्षण सूट हिट कोड की कितनी पंक्तियाँ, से बचा जाना चाहिए। ऐसा इसलिए है क्योंकि परीक्षण करते समय, केवल कोड की लाइनों की संख्या पर ध्यान केंद्रित करना हमारे सिस्टम में कोई वास्तविक मूल्य नहीं जोड़ता है। कुछ कीड़े रह सकते हैं, जो बाद में तैनाती के बाद भी बाद के चरण में प्रतिबिंबित होते हैं।

परीक्षण करने के लिए निम्नलिखित महत्वपूर्ण बिंदुओं पर विचार करें -

  • हमें कोड कवरेज की बजाय कोड की कार्यक्षमता के परीक्षण पर ध्यान देने की आवश्यकता है।

  • हमें पहले कोड के सबसे महत्वपूर्ण भागों का परीक्षण करने की आवश्यकता है और फिर कोड के कम महत्वपूर्ण भागों की ओर बढ़ना चाहिए। इससे समय जरूर बचेगा।

  • परीक्षक के पास विभिन्न परीक्षण होने चाहिए, जो सॉफ़्टवेयर को उसकी सीमा तक धकेल सकते हैं।

समवर्ती सॉफ्टवेयर प्रोग्राम के परीक्षण के लिए अनुमोदन

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

समवर्ती सॉफ्टवेयर प्रोग्राम के लिए परीक्षण तकनीक व्यापक रूप से उन इंटरलेइंग को चुनने पर ध्यान केंद्रित कर रही है जो दौड़ की स्थिति, गतिरोध और परमाणु के उल्लंघन जैसे संभावित हानिकारक पैटर्न को उजागर करते हैं। समवर्ती सॉफ्टवेयर प्रोग्राम के परीक्षण के लिए दो दृष्टिकोण निम्नलिखित हैं -

व्यवस्थित अन्वेषण

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

संपत्ति पर ही आधारित

प्रॉपर्टी से चलने वाले दृष्टिकोण इस बात पर निर्भर करते हैं कि कंसीलरिटी दोष के कारण कंसीव करने में अधिक संभावना है जो कि संदिग्ध मेमोरी एक्सेस पैटर्न जैसे विशिष्ट गुणों को उजागर करता है। विभिन्न संपत्ति-चालित दृष्टिकोण विभिन्न दोषों जैसे कि दौड़ की स्थिति, गतिरोध और परमाणु क्षमता के उल्लंघन को लक्षित करते हैं, जो आगे एक या अन्य विशिष्ट गुणों पर निर्भर करता है।

परीक्षण रणनीतियाँ

टेस्ट रणनीति को टेस्ट दृष्टिकोण के रूप में भी जाना जाता है। रणनीति निर्धारित करती है कि परीक्षण कैसे किया जाएगा। टेस्ट दृष्टिकोण में दो तकनीकें हैं -

सक्रिय

एक दृष्टिकोण जिसमें निर्माण के पहले दोषों को खोजने और उन्हें ठीक करने के लिए परीक्षण डिजाइन प्रक्रिया जितनी जल्दी हो सके शुरू की जाती है।

रिएक्टिव

एक दृष्टिकोण जिसमें विकास प्रक्रिया के पूरा होने तक परीक्षण शुरू नहीं होता है।

अजगर कार्यक्रम पर किसी भी परीक्षण रणनीति या दृष्टिकोण को लागू करने से पहले, हमारे पास एक सॉफ्टवेयर प्रोग्राम की तरह की त्रुटियों के बारे में एक मूल विचार होना चाहिए। त्रुटियां इस प्रकार हैं -

वाक्यविन्यास त्रुटियाँ

कार्यक्रम के विकास के दौरान, कई छोटी त्रुटियां हो सकती हैं। गलतियाँ ज्यादातर टाइपिंग गलतियों के कारण होती हैं। उदाहरण के लिए, बृहदान्त्र गुम होना या किसी कीवर्ड की गलत वर्तनी, आदि ऐसी त्रुटियाँ प्रोग्राम सिंटैक्स में गलती के कारण होती हैं न कि तर्क में। इसलिए, इन त्रुटियों को सिंटैक्टिकल त्रुटियाँ कहा जाता है।

शब्दार्थ त्रुटियाँ

शब्दार्थ त्रुटियों को तार्किक त्रुटियां भी कहा जाता है। यदि सॉफ़्टवेयर प्रोग्राम में कोई तार्किक या शब्दार्थ त्रुटि है, तो कथन सही ढंग से संकलित और चलेगा, लेकिन यह वांछित आउटपुट नहीं देगा क्योंकि तर्क सही नहीं है।

इकाई का परीक्षण

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

हमारे बाद के अनुभागों में, हम यूनिट परीक्षण के लिए विभिन्न पायथन मॉड्यूल के बारे में जानेंगे।

सबसे अच्छा मॉड्यूल

यूनिट परीक्षण के लिए सबसे पहला मॉड्यूल सबसे बेकार मॉड्यूल है। यह JUnit से प्रेरित है और डिफ़ॉल्ट रूप से Python3.6 में शामिल है। यह परीक्षण स्वचालन, परीक्षणों के लिए सेटअप और शटडाउन कोड साझा करने, संग्रह में परीक्षणों के एकत्रीकरण और रिपोर्टिंग ढांचे से परीक्षणों की स्वतंत्रता का समर्थन करता है।

निम्नलिखित कुछ महत्वपूर्ण अवधारणाएं हैं जिन्हें यूनिटेस्ट मॉड्यूल द्वारा समर्थित किया गया है

पाठ का निर्धारण

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

परीक्षण का मामला

परीक्षण मामला जांचता है कि इनपुट के विशिष्ट सेट से एक आवश्यक प्रतिक्रिया आ रही है या नहीं। यूनिस्टेस्ट मॉड्यूल में टेस्टकेस नाम का एक बेस क्लास शामिल है जिसका उपयोग नए परीक्षण मामलों को बनाने के लिए किया जा सकता है। इसमें दो डिफ़ॉल्ट तरीके शामिल हैं -

  • setUp()- व्यायाम करने से पहले परीक्षण स्थिरता स्थापित करने के लिए एक हुक विधि। इसे लागू परीक्षण विधियों को कॉल करने से पहले कहा जाता है।

  • tearDown( - कक्षा में सभी परीक्षणों को चलाने के बाद कक्षा की स्थिरता को कम करने के लिए एक हुक विधि।

परीक्षण सूट

यह परीक्षण सूट, परीक्षण मामलों या दोनों का एक संग्रह है।

टेस्ट रनर

यह परीक्षण मामलों या सूट के चलने को नियंत्रित करता है और उपयोगकर्ता को परिणाम प्रदान करता है। यह परिणाम प्रदान करने के लिए GUI या सरल पाठ इंटरफ़ेस का उपयोग कर सकता है।

Example

निम्नलिखित पायथन कार्यक्रम नामांकित मॉड्यूल का परीक्षण करने के लिए सबसे उपयुक्त मॉड्यूल का उपयोग करता है Fibonacci। कार्यक्रम किसी संख्या की फाइबोनैचि श्रृंखला की गणना करने में मदद करता है। इस उदाहरण में, हमने विभिन्न तरीकों का उपयोग करके परीक्षण मामलों को परिभाषित करने के लिए, Fibo_test नामक एक वर्ग बनाया है। ये विधियाँ unittest.TestCase से विरासत में मिली हैं। हम डिफ़ॉल्ट तरीकों से दो का उपयोग कर रहे हैं - सेटअप () और टियरडाउन ()। हम टेस्टफिबोकल विधि को भी परिभाषित करते हैं। परीक्षण के नाम को पत्र परीक्षण के साथ शुरू किया जाना चाहिए। अंतिम ब्लॉक में, unittest.main () परीक्षण स्क्रिप्ट को कमांड-लाइन इंटरफ़ेस प्रदान करता है।

import unittest
def fibonacci(n):
   a, b = 0, 1
   for i in range(n):
   a, b = b, a + b
   return a
class Fibo_Test(unittest.TestCase):
   def setUp(self):
   print("This is run before our tests would be executed")
   def tearDown(self):
   print("This is run after the completion of execution of our tests")

   def testfibocal(self):
   self.assertEqual(fib(0), 0)
   self.assertEqual(fib(1), 1)
   self.assertEqual(fib(5), 5)
   self.assertEqual(fib(10), 55)
   self.assertEqual(fib(20), 6765)

if __name__ == "__main__":
   unittest.main()

जब कमांड लाइन से चलाया जाता है, तो उपरोक्त स्क्रिप्ट एक आउटपुट उत्पन्न करती है जो इस तरह दिखता है -

उत्पादन

This runs before our tests would be executed.
This runs after the completion of execution of our tests.
.
----------------------------------------------------------------------
Ran 1 test in 0.006s
OK

अब, इसे स्पष्ट करने के लिए, हम अपना कोड बदल रहे हैं, जिससे फाइबोनैचि मॉड्यूल को परिभाषित करने में मदद मिली।

एक उदाहरण के रूप में निम्नलिखित कोड ब्लॉक पर विचार करें -

def fibonacci(n):
   a, b = 0, 1
   for i in range(n):
   a, b = b, a + b
   return a

कोड ब्लॉक में कुछ बदलाव निम्नानुसार किए गए हैं -

def fibonacci(n):
   a, b = 1, 1
   for i in range(n):
   a, b = b, a + b
   return a

अब, बदले हुए कोड के साथ स्क्रिप्ट चलाने के बाद, हम निम्नलिखित आउटपुट प्राप्त करेंगे -

This runs before our tests would be executed.
This runs after the completion of execution of our tests.
F
======================================================================
FAIL: testCalculation (__main__.Fibo_Test)
----------------------------------------------------------------------
Traceback (most recent call last):
File "unitg.py", line 15, in testCalculation
self.assertEqual(fib(0), 0)
AssertionError: 1 != 0
----------------------------------------------------------------------
Ran 1 test in 0.007s

FAILED (failures = 1)

उपरोक्त आउटपुट से पता चलता है कि मॉड्यूल वांछित आउटपुट देने में विफल रहा है।

डॉकटेस्ट मॉड्यूल

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

यदि हमारे कोड में सब कुछ ठीक है, तो डॉकस्टेस्ट मॉड्यूल से कोई आउटपुट नहीं होगा; अन्यथा, यह आउटपुट प्रदान करेगा।

उदाहरण

निम्नलिखित पायथन उदाहरण फाइबोनैचि नामक मॉड्यूल का परीक्षण करने के लिए डॉकस्टेस्ट मॉड्यूल का उपयोग करता है, जो किसी संख्या की फाइबोनैचि श्रृंखला की गणना करने में मदद करता है।

import doctest
def fibonacci(n):
   """
   Calculates the Fibonacci number

   >>> fibonacci(0)
   0
   >>> fibonacci(1)
   1
   >>> fibonacci(10)
   55
   >>> fibonacci(20)
   6765
   >>>

   """
   a, b = 1, 1
   for i in range(n):
   a, b = b, a + b
   return a
      if __name__ == "__main__":
   doctest.testmod()

हम देख सकते हैं कि फ़ॉब नाम के संबंधित फ़ंक्शन के डॉकस्ट्रिंग में आउटपुट के साथ इंटरेक्टिव पायथन सत्र था। यदि हमारा कोड ठीक है, तो doctest मॉड्यूल से कोई आउटपुट नहीं होगा। लेकिन यह देखने के लिए कि यह कैसे काम करता है हम इसे -v विकल्प के साथ चला सकते हैं।

(base) D:\ProgramData>python dock_test.py -v
Trying:
   fibonacci(0)
Expecting:
   0
ok
Trying:
   fibonacci(1)
Expecting:
   1
ok
Trying:
   fibonacci(10)
Expecting:
   55
ok
Trying:
   fibonacci(20)
Expecting:
   6765
ok
1 items had no tests:
   __main__
1 items passed all tests:
4 tests in __main__.fibonacci
4 tests in 2 items.
4 passed and 0 failed.
Test passed.

अब, हम उस कोड को बदल देंगे जिसने फाइबोनैचि मॉड्यूल को परिभाषित करने में मदद की

एक उदाहरण के रूप में निम्नलिखित कोड ब्लॉक पर विचार करें -

def fibonacci(n):
   a, b = 0, 1
   for i in range(n):
   a, b = b, a + b
   return a

निम्नलिखित कोड ब्लॉक परिवर्तनों के साथ मदद करता है -

def fibonacci(n):
   a, b = 1, 1
   for i in range(n):
   a, b = b, a + b
   return a

स्क्रिप्ट को बिना -v विकल्प के भी चलाने के बाद, बदले हुए कोड के साथ, हम आउटपुट को नीचे दिखाए गए अनुसार प्राप्त करेंगे।

उत्पादन

(base) D:\ProgramData>python dock_test.py
**********************************************************************
File "unitg.py", line 6, in __main__.fibonacci
Failed example:
   fibonacci(0)
Expected:
   0
Got:
   1
**********************************************************************
File "unitg.py", line 10, in __main__.fibonacci
Failed example:
   fibonacci(10)
Expected:
   55
Got:
   89
**********************************************************************
File "unitg.py", line 12, in __main__.fibonacci
Failed example:
   fibonacci(20)
Expected:
   6765
Got:
   10946
**********************************************************************
1 items had failures:
   3 of 4 in __main__.fibonacci
***Test Failed*** 3 failures.

हम उपरोक्त आउटपुट में देख सकते हैं कि तीन परीक्षण विफल हो गए हैं।

इस अध्याय में, हम थ्रेड अनुप्रयोगों को डीबग करना सीखेंगे। हम डिबगिंग के महत्व को भी जानेंगे।

डीबगिंग क्या है?

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

पायथन डिबगर

पायथन डिबगर या pdbपायथन मानक पुस्तकालय का एक हिस्सा है। यह हार्ड-टू-फाइंड बग को ट्रैक करने के लिए एक अच्छा कमबैक टूल है और हमें दोषपूर्ण कोड को जल्दी और मज़बूती से ठीक करने की अनुमति देता है। अनुवर्ती दो सबसे महत्वपूर्ण कार्य हैंpdp डिबगर -

  • यह हमें रनटाइम पर चर के मूल्यों की जांच करने की अनुमति देता है।
  • हम कोड के माध्यम से कदम रख सकते हैं और ब्रेकप्वाइंट भी सेट कर सकते हैं।

हम निम्नलिखित दो तरीकों से pdb के साथ काम कर सकते हैं -

  • कमांड-लाइन के माध्यम से; इसे पोस्टमॉर्टम डिबगिंग भी कहा जाता है।
  • अंतःक्रियात्मक रूप से pdb चलाकर।

पीडीबी के साथ काम करना

पायथन डिबगर के साथ काम करने के लिए, हमें उस स्थान पर निम्नलिखित कोड का उपयोग करने की आवश्यकता है, जहां हम डीबग में तोड़ना चाहते हैं -

import pdb;
pdb.set_trace()

कमांड-लाइन के माध्यम से pdb के साथ काम करने के लिए निम्न आदेशों पर विचार करें।

  • h(help)
  • d(down)
  • u(up)
  • b(break)
  • cl(clear)
  • l(list))
  • n(next))
  • c(continue)
  • s(step)
  • r(return))
  • b(break)

निम्नलिखित पायथन डीबगर के एच (मदद) कमांड का एक डेमो है -

import pdb

pdb.set_trace()
--Call--
>d:\programdata\lib\site-packages\ipython\core\displayhook.py(247)__call__()
-> def __call__(self, result = None):
(Pdb) h

Documented commands (type help <topic>):
========================================
EOF   c         d       h        list     q       rv      undisplay
a     cl        debug   help     ll       quit    s       unt
alias clear     disable ignore   longlist r       source  until
args  commands  display interact n        restart step    up
b     condition down    j        next     return  tbreak  w
break cont      enable  jump     p        retval  u       whatis
bt    continue  exit    l        pp       run     unalias where

Miscellaneous help topics:
==========================
exec pdb

उदाहरण

पायथन डिबगर के साथ काम करते हुए, हम स्क्रिप्ट को कहीं भी निम्न पंक्तियों का उपयोग करके सेट कर सकते हैं -

import pdb;
pdb.set_trace()

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

import pdb;
a = "aaa"
pdb.set_trace()
b = "bbb"
c = "ccc"
final = a + b + c
print (final)

जब उपरोक्त स्क्रिप्ट चलती है, तो यह प्रोग्राम को = "आ" तक निष्पादित करेगा, हम इसे निम्न आउटपुट में जांच सकते हैं।

उत्पादन

--Return--
> <ipython-input-7-8a7d1b5cc854>(3)<module>()->None
-> pdb.set_trace()
(Pdb) p a
'aaa'
(Pdb) p b
*** NameError: name 'b' is not defined
(Pdb) p c
*** NameError: name 'c' is not defined

Pdb में कमांड 'p (प्रिंट)' का उपयोग करने के बाद, यह स्क्रिप्ट केवल 'आ' प्रिंट कर रही है। इसके बाद एक त्रुटि होती है क्योंकि हमने ब्रेकप्वाइंट को = "आआ" तक निर्धारित किया है।

इसी प्रकार, हम ब्रेकपॉइंट बदलकर स्क्रिप्ट को चला सकते हैं और आउटपुट में अंतर देख सकते हैं -

import pdb
a = "aaa"
b = "bbb"
c = "ccc"
pdb.set_trace()
final = a + b + c
print (final)

उत्पादन

--Return--
> <ipython-input-9-a59ef5caf723>(5)<module>()->None
-> pdb.set_trace()
(Pdb) p a
'aaa'
(Pdb) p b
'bbb'
(Pdb) p c
'ccc'
(Pdb) p final
*** NameError: name 'final' is not defined
(Pdb) exit

निम्नलिखित स्क्रिप्ट में, हम प्रोग्राम की अंतिम पंक्ति में ब्रेकपॉइंट सेट कर रहे हैं -

import pdb
a = "aaa"
b = "bbb"
c = "ccc"
final = a + b + c
pdb.set_trace()
print (final)

आउटपुट इस प्रकार है -

--Return--
> <ipython-input-11-8019b029997d>(6)<module>()->None
-> pdb.set_trace()
(Pdb) p a
'aaa'
(Pdb) p b
'bbb'
(Pdb) p c
'ccc'
(Pdb) p final
'aaabbbccc'
(Pdb)

इस अध्याय में, हम सीखेंगे कि प्रदर्शन के मुद्दों को सुलझाने में बेंचमार्किंग और प्रोफाइलिंग कैसे मदद करती है।

माना कि हमने एक कोड लिखा था और यह वांछित परिणाम भी दे रहा है लेकिन क्या होगा अगर हम इस कोड को थोड़ा और तेजी से चलाना चाहते हैं क्योंकि जरूरतें बदल गई हैं। इस मामले में, हमें यह पता लगाने की आवश्यकता है कि हमारे कोड के कौन से हिस्से पूरे कार्यक्रम को धीमा कर रहे हैं। इस मामले में, बेंचमार्किंग और प्रोफाइलिंग उपयोगी हो सकती है।

बेंचमार्किंग क्या है?

मानक के साथ तुलना करके बेंचमार्किंग का उद्देश्य किसी वस्तु का मूल्यांकन करना है। हालांकि, यह सवाल यहां उठता है कि सॉफ्टवेयर प्रोग्रामिंग के मामले में बेंचमार्किंग क्या होगी और हमें इसकी आवश्यकता क्यों है। कोड को बेंचमार्क करने का मतलब है कि कोड कितनी तेजी से निष्पादित हो रहा है और कहां अड़चन है। बेंचमार्किंग का एक बड़ा कारण यह है कि यह कोड का अनुकूलन करता है।

बेंचमार्किंग कैसे काम करता है?

अगर हम बेंचमार्किंग के काम के बारे में बात करते हैं, तो हमें पूरे कार्यक्रम को एक वर्तमान स्थिति के रूप में बेंचमार्किंग करके शुरू करना होगा, फिर हम माइक्रो बेंचमार्क को जोड़ सकते हैं और फिर एक प्रोग्राम को छोटे कार्यक्रमों में बदल सकते हैं। हमारे कार्यक्रम के भीतर बाधाओं को खोजने और इसे अनुकूलित करने के लिए। दूसरे शब्दों में, हम इसे बड़ी और कठिन समस्या को छोटी श्रृंखला में तोड़कर और उन्हें अनुकूलित करने के लिए थोड़ी आसान समस्याओं के रूप में समझ सकते हैं।

बेंचमार्किंग के लिए पायथन मॉड्यूल

पायथन में, हमारे पास बेंचमार्किंग के लिए एक डिफ़ॉल्ट मॉड्यूल है जिसे कहा जाता है timeit। की मदद सेtimeit मॉड्यूल, हम अपने मुख्य कार्यक्रम के भीतर छोटे से पायथन कोड के प्रदर्शन को माप सकते हैं।

उदाहरण

निम्नलिखित पायथन लिपि में, हम आयात कर रहे हैं timeit मॉड्यूल, जो आगे दो कार्यों को निष्पादित करने में लगने वाले समय को मापता है - functionA तथा functionB -

import timeit
import time
def functionA():
   print("Function A starts the execution:")
   print("Function A completes the execution:")
def functionB():
   print("Function B starts the execution")
   print("Function B completes the execution")
start_time = timeit.default_timer()
functionA()
print(timeit.default_timer() - start_time)
start_time = timeit.default_timer()
functionB()
print(timeit.default_timer() - start_time)

उपरोक्त स्क्रिप्ट को चलाने के बाद, हम नीचे दिखाए गए अनुसार दोनों कार्यों का निष्पादन समय प्राप्त करेंगे।

उत्पादन

Function A starts the execution:
Function A completes the execution:
0.0014599495514175942
Function B starts the execution
Function B completes the execution
0.0017024724827479076

डेकोरेटर फ़ंक्शन का उपयोग करके हमारे अपने टाइमर को लिखना

पायथन में, हम अपना खुद का टाइमर बना सकते हैं, जो ठीक उसी तरह काम करेगा timeitमापांक। की सहायता से किया जा सकता हैdecoratorसमारोह। निम्नलिखित कस्टम टाइमर का एक उदाहरण है -

import random
import time

def timer_func(func):

   def function_timer(*args, **kwargs):
   start = time.time()
   value = func(*args, **kwargs)
   end = time.time()
   runtime = end - start
   msg = "{func} took {time} seconds to complete its execution."
      print(msg.format(func = func.__name__,time = runtime))
   return value
   return function_timer

@timer_func
def Myfunction():
   for x in range(5):
   sleep_time = random.choice(range(1,3))
   time.sleep(sleep_time)

if __name__ == '__main__':
   Myfunction()

उपरोक्त अजगर स्क्रिप्ट यादृच्छिक समय मॉड्यूल को आयात करने में मदद करती है। हमने टाइमर_फंच () डेकोरेटर फ़ंक्शन बनाया है। इसके अंदर function_timer () फ़ंक्शन है। अब, नेस्टेड फ़ंक्शन पास किए गए फ़ंक्शन को कॉल करने से पहले समय को पकड़ लेगा। फिर यह फ़ंक्शन के लौटने का इंतजार करता है और अंतिम समय पकड़ लेता है। इस तरह, हम अंत में अजगर स्क्रिप्ट को निष्पादन समय प्रिंट कर सकते हैं। स्क्रिप्ट आउटपुट दिखाएगी जैसा कि नीचे दिखाया गया है।

उत्पादन

Myfunction took 8.000457763671875 seconds to complete its execution.

प्रोफाइलिंग क्या है?

कभी-कभी प्रोग्रामर कुछ विशेषताओं को मापना चाहता है जैसे कि मेमोरी का उपयोग, समय की जटिलता या उस कार्यक्रम की वास्तविक क्षमता को मापने के लिए कार्यक्रमों के बारे में विशेष निर्देशों का उपयोग। प्रोग्राम के बारे में इस तरह के मापन को प्रोफाइलिंग कहा जाता है। प्रोफाइलिंग ऐसे मापने के लिए गतिशील कार्यक्रम विश्लेषण का उपयोग करता है।

बाद के अनुभागों में, हम प्रोफाइलिंग के लिए विभिन्न पायथन मॉड्यूल्स के बारे में जानेंगे।

cProfile - इनबिल्ट मॉड्यूल

cProfileएक पायथन में निर्मित मॉड्यूल है जो प्रोफाइलिंग के लिए है। मॉड्यूल उचित ओवरहेड के साथ एक सी-एक्सटेंशन है जो इसे लंबे समय तक चलने वाले कार्यक्रमों की रूपरेखा के लिए उपयुक्त बनाता है। इसे चलाने के बाद, यह सभी फ़ंक्शन और निष्पादन समय को लॉग करता है। यह बहुत शक्तिशाली है लेकिन कभी-कभी इसकी व्याख्या करना और कार्य करना थोड़ा कठिन होता है। निम्नलिखित उदाहरण में, हम नीचे दिए गए कोड पर cProfile का उपयोग कर रहे हैं -

उदाहरण

def increment_global():

   global x
   x += 1

def taskofThread(lock):

   for _ in range(50000):
   lock.acquire()
   increment_global()
   lock.release()

def main():
   global x
   x = 0

   lock = threading.Lock()

   t1 = threading.Thread(target=taskofThread, args=(lock,))
   t2 = threading.Thread(target= taskofThread, args=(lock,))

   t1.start()
   t2.start()

   t1.join()
   t2.join()

if __name__ == "__main__":
   for i in range(5):
      main()
   print("x = {1} after Iteration {0}".format(i,x))

उपरोक्त कोड में सहेजा गया है thread_increment.pyफ़ाइल। अब, कमांड लाइन पर cProfile के साथ कोड निष्पादित करें -

(base) D:\ProgramData>python -m cProfile thread_increment.py
x = 100000 after Iteration 0
x = 100000 after Iteration 1
x = 100000 after Iteration 2
x = 100000 after Iteration 3
x = 100000 after Iteration 4
      3577 function calls (3522 primitive calls) in 1.688 seconds

   Ordered by: standard name

   ncalls tottime percall cumtime percall filename:lineno(function)

   5 0.000 0.000 0.000 0.000 <frozen importlib._bootstrap>:103(release)
   5 0.000 0.000 0.000 0.000 <frozen importlib._bootstrap>:143(__init__)
   5 0.000 0.000 0.000 0.000 <frozen importlib._bootstrap>:147(__enter__)
   … … … …

उपरोक्त आउटपुट से, यह स्पष्ट है कि cProfile सभी 3577 फ़ंक्शनों को प्रिंट करता है, जिनमें से प्रत्येक में खर्च किए गए समय और उन्हें कॉल किए जाने की संख्या के साथ। अनुगमन हम उत्पादन में प्राप्त कॉलम हैं -

  • ncalls - यह किए गए कॉल की संख्या है।

  • tottime - यह दिए गए फ़ंक्शन में बिताया गया कुल समय है।

  • percall - यह एनसीएल द्वारा विभाजित टोटाइम के भागफल को संदर्भित करता है।

  • cumtime- यह इस और सभी उपक्षेत्रों में खर्च किया गया संचयी समय है। यह पुनरावर्ती कार्यों के लिए भी सटीक है।

  • percall - यह आदिम कॉल द्वारा विभाजित सह-काल का भागफल है।

  • filename:lineno(function) - यह मूल रूप से प्रत्येक फ़ंक्शन का संबंधित डेटा प्रदान करता है।

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

  • यदि एक थ्रेड पूल में एक धागा अपने निष्पादन को पूरा करता है तो उस धागे का पुन: उपयोग किया जा सकता है।

  • यदि एक थ्रेड समाप्त हो जाता है, तो उस थ्रेड को बदलने के लिए एक और थ्रेड बनाया जाएगा।

पायथन मॉड्यूल - समवर्ती.फुट्स

पायथन मानक पुस्तकालय में शामिल हैं concurrent.futuresमापांक। इस मॉड्यूल को पायथन 3.2 में जोड़ा गया था ताकि डेवलपर्स को अतुल्यकालिक कार्यों को लॉन्च करने के लिए एक उच्च-स्तरीय इंटरफ़ेस प्रदान किया जा सके। यह थ्रेड या प्रक्रियाओं के पूल का उपयोग करके कार्यों को चलाने के लिए इंटरफ़ेस प्रदान करने के लिए पायथन के थ्रेडिंग और मल्टीप्रोसेसिंग मॉड्यूल के शीर्ष पर एक अमूर्त परत है।

हमारे बाद के अनुभागों में, हम समवर्ती.फुट्स मॉड्यूल के विभिन्न वर्गों के बारे में जानेंगे।

एग्जीक्यूटिव क्लास

Executorका एक अमूर्त वर्ग है concurrent.futuresपायथन मॉड्यूल। इसे सीधे इस्तेमाल नहीं किया जा सकता है और हमें निम्नलिखित ठोस उपवर्गों में से एक का उपयोग करने की आवश्यकता है -

  • ThreadPoolExecutor
  • ProcessPoolExecutor

ThreadPoolExecutor - एक ठोस उपवर्ग

यह एक्ज़ीक्यूटर क्लास के ठोस उपवर्गों में से एक है। उपवर्ग बहु-सूत्रण का उपयोग करता है और हमें कार्यों को प्रस्तुत करने के लिए धागे का एक पूल मिलता है। यह पूल उपलब्ध थ्रेड्स को कार्य सौंपता है और उन्हें चलाने के लिए शेड्यूल करता है।

कैसे एक ThreadPoolExecutor बनाने के लिए?

की मदद से concurrent.futures मॉड्यूल और उसके ठोस उपवर्ग Executor, हम आसानी से धागे का एक पूल बना सकते हैं। इसके लिए, हमें निर्माण करने की आवश्यकता हैThreadPoolExecutorथ्रेड्स की संख्या के साथ हम पूल में चाहते हैं। डिफ़ॉल्ट रूप से, संख्या 5 है। फिर हम थ्रेड पूल को एक कार्य सबमिट कर सकते हैं। जब हमsubmit() एक कार्य, हम वापस आ जाओ Future। फ्यूचर ऑब्जेक्ट में एक विधि होती है जिसे कहा जाता हैdone(), जो बताता है कि भविष्य सुलझ गया है या नहीं। इसके साथ, उस विशेष भविष्य की वस्तु के लिए एक मूल्य निर्धारित किया गया है। जब कोई कार्य पूरा होता है, तो थ्रेड पूल निष्पादक मूल्य को भविष्य की वस्तु पर सेट करता है।

उदाहरण

from concurrent.futures import ThreadPoolExecutor
from time import sleep
def task(message):
   sleep(2)
   return message

def main():
   executor = ThreadPoolExecutor(5)
   future = executor.submit(task, ("Completed"))
   print(future.done())
   sleep(2)
   print(future.done())
   print(future.result())
if __name__ == '__main__':
main()

उत्पादन

False
True
Completed

उपरोक्त उदाहरण में, ए ThreadPoolExecutor5 धागे के साथ निर्माण किया गया है। फिर एक कार्य, जो संदेश देने से पहले 2 सेकंड के लिए प्रतीक्षा करेगा, थ्रेड पूल निष्पादक को प्रस्तुत किया जाता है। जैसा कि आउटपुट से देखा जाता है, कार्य 2 सेकंड तक पूरा नहीं होता है, इसलिए पहला कॉलdone()झूठा लौटेगा। 2 सेकंड के बाद, कार्य किया जाता है और हम कॉल करके भविष्य का परिणाम प्राप्त करते हैंresult() इस पर विधि।

Instantiating ThreadPoolExecutor - प्रसंग प्रबंधक

त्वरित करने का दूसरा तरीका ThreadPoolExecutorसंदर्भ प्रबंधक की सहायता से होता है। यह उपरोक्त उदाहरण में प्रयुक्त विधि के समान काम करता है। संदर्भ प्रबंधक का उपयोग करने का मुख्य लाभ यह है कि यह वाक्यात्मक रूप से अच्छा दिखता है। तत्काल कोड निम्नलिखित कोड की मदद से किया जा सकता है -

with ThreadPoolExecutor(max_workers = 5) as executor

उदाहरण

निम्नलिखित उदाहरण पायथन डॉक्स से उधार लिया गया है। इस उदाहरण में, सबसे पहलेconcurrent.futuresमॉड्यूल आयात करना पड़ता है। फिर एक समारोह का नामload_url()बनाया गया है जो अनुरोधित url को लोड करेगा। फ़ंक्शन तब बनाता हैThreadPoolExecutorपूल में 5 धागे के साथ। ThreadPoolExecutorसंदर्भ प्रबंधक के रूप में उपयोग किया गया है। हम भविष्य के परिणाम को कॉल करके प्राप्त कर सकते हैंresult() इस पर विधि।

import concurrent.futures
import urllib.request

URLS = ['http://www.foxnews.com/',
   'http://www.cnn.com/',
   'http://europe.wsj.com/',
   'http://www.bbc.co.uk/',
   'http://some-made-up-domain.com/']

def load_url(url, timeout):
   with urllib.request.urlopen(url, timeout = timeout) as conn:
   return conn.read()

with concurrent.futures.ThreadPoolExecutor(max_workers = 5) as executor:

   future_to_url = {executor.submit(load_url, url, 60): url for url in URLS}
   for future in concurrent.futures.as_completed(future_to_url):
   url = future_to_url[future]
   try:
      data = future.result()
   except Exception as exc:
      print('%r generated an exception: %s' % (url, exc))
   else:
      print('%r page is %d bytes' % (url, len(data)))

उत्पादन

निम्नलिखित पायथन लिपि का उत्पादन होगा -

'http://some-made-up-domain.com/' generated an exception: <urlopen error [Errno 11004] getaddrinfo failed>
'http://www.foxnews.com/' page is 229313 bytes
'http://www.cnn.com/' page is 168933 bytes
'http://www.bbc.co.uk/' page is 283893 bytes
'http://europe.wsj.com/' page is 938109 bytes

Executor.map () फ़ंक्शन का उपयोग

अजगर map()फ़ंक्शन को व्यापक रूप से कई कार्यों में उपयोग किया जाता है। ऐसा एक कार्य पुनरावृत्तियों के भीतर प्रत्येक तत्व के लिए एक निश्चित कार्य लागू करना है। इसी तरह, हम किसी फ़ंक्शन के लिए इट्रेटर के सभी तत्वों को मैप कर सकते हैं और इन्हें बाहर निकालने के लिए स्वतंत्र नौकरियों के रूप में सबमिट कर सकते हैंThreadPoolExecutor। फ़ंक्शन कैसे काम करता है यह समझने के लिए पायथन स्क्रिप्ट के निम्नलिखित उदाहरण पर विचार करें।

उदाहरण

नीचे दिए गए उदाहरण में, मानचित्र फ़ंक्शन का उपयोग करने के लिए किया जाता है square() मान सरणी में प्रत्येक मान पर कार्य करें।

from concurrent.futures import ThreadPoolExecutor
from concurrent.futures import as_completed
values = [2,3,4,5]
def square(n):
   return n * n
def main():
   with ThreadPoolExecutor(max_workers = 3) as executor:
      results = executor.map(square, values)
for result in results:
      print(result)
if __name__ == '__main__':
   main()

उत्पादन

उपरोक्त पायथन लिपि निम्न आउटपुट उत्पन्न करती है -

4
9
16
25

पूल का निर्माण उसी तरह से किया और उपयोग किया जा सकता है जैसा कि हमने थ्रेड के पूल का निर्माण और उपयोग किया है। प्रक्रिया पूल को पूर्व-तात्कालिक और निष्क्रिय प्रक्रियाओं के समूह के रूप में परिभाषित किया जा सकता है, जो काम देने के लिए तैयार हैं। जब हम बड़ी संख्या में कार्यों को करने की आवश्यकता होती है, तो प्रत्येक कार्य के लिए नई प्रक्रियाओं को तत्काल शुरू करने पर प्रक्रिया पूल बनाना पसंद किया जाता है।

पायथन मॉड्यूल - समवर्ती.फुट्स

पायथन मानक पुस्तकालय में एक मॉड्यूल होता है जिसे कहा जाता है concurrent.futures। इस मॉड्यूल को पायथन 3.2 में जोड़ा गया था ताकि डेवलपर्स को अतुल्यकालिक कार्यों को लॉन्च करने के लिए एक उच्च-स्तरीय इंटरफ़ेस प्रदान किया जा सके। यह थ्रेड या प्रक्रियाओं के पूल का उपयोग करके कार्यों को चलाने के लिए इंटरफ़ेस प्रदान करने के लिए पायथन के थ्रेडिंग और मल्टीप्रोसेसिंग मॉड्यूल के शीर्ष पर एक अमूर्त परत है।

अपने बाद के अनुभागों में, हम समवर्ती.फुट्स मॉड्यूल के विभिन्न उपवर्गों को देखेंगे।

एग्जीक्यूटिव क्लास

Executor का एक अमूर्त वर्ग है concurrent.futuresपायथन मॉड्यूल। इसे सीधे इस्तेमाल नहीं किया जा सकता है और हमें निम्नलिखित ठोस उपवर्गों में से एक का उपयोग करने की आवश्यकता है -

  • ThreadPoolExecutor
  • ProcessPoolExecutor

ProcessPoolExecutor - एक ठोस उपवर्ग

यह एक्ज़ीक्यूटर क्लास के ठोस उपवर्गों में से एक है। यह बहु-प्रसंस्करण का उपयोग करता है और हमें कार्यों को प्रस्तुत करने के लिए प्रक्रियाओं का एक पूल मिलता है। यह पूल उपलब्ध प्रक्रियाओं को कार्य सौंपता है और उन्हें चलाने के लिए शेड्यूल करता है।

ProcessPoolExecutor कैसे बनाये?

की मदद से concurrent.futures मॉड्यूल और उसके ठोस उपवर्ग Executor, हम आसानी से प्रक्रिया का एक पूल बना सकते हैं। इसके लिए, हमें निर्माण करने की आवश्यकता हैProcessPoolExecutorहम पूल में जितनी प्रक्रियाएँ चाहते हैं। डिफ़ॉल्ट रूप से, संख्या 5. यह प्रक्रिया पूल में एक कार्य सबमिट करने के बाद है।

उदाहरण

अब हम उसी उदाहरण पर विचार करेंगे जिसका उपयोग हमने थ्रेड पूल बनाते समय किया था, एकमात्र अंतर यह है कि अब हम उपयोग करेंगे ProcessPoolExecutor के बजाय ThreadPoolExecutor

from concurrent.futures import ProcessPoolExecutor
from time import sleep
def task(message):
   sleep(2)
   return message

def main():
   executor = ProcessPoolExecutor(5)
   future = executor.submit(task, ("Completed"))
   print(future.done())
   sleep(2)
   print(future.done())
   print(future.result())
if __name__ == '__main__':
main()

उत्पादन

False
False
Completed

उपरोक्त उदाहरण में, एक प्रक्रियाPoolExecutor5 धागे के साथ निर्माण किया गया है। फिर एक कार्य, जो संदेश देने से पहले 2 सेकंड तक प्रतीक्षा करेगा, प्रक्रिया पूल निष्पादक को प्रस्तुत की जाती है। जैसा कि आउटपुट से देखा जाता है, कार्य 2 सेकंड तक पूरा नहीं होता है, इसलिए पहला कॉलdone()झूठा लौटेगा। 2 सेकंड के बाद, कार्य किया जाता है और हम कॉल करके भविष्य का परिणाम प्राप्त करते हैंresult() इस पर विधि।

इंस्टेंटिअटिंग प्रोसेसपूलएक्सक्यूटोर - प्रसंग प्रबंधक

ProcessPoolExecutor को त्वरित करने का एक अन्य तरीका संदर्भ प्रबंधक की मदद से है। यह उपरोक्त उदाहरण में प्रयुक्त विधि के समान काम करता है। संदर्भ प्रबंधक का उपयोग करने का मुख्य लाभ यह है कि यह वाक्यात्मक रूप से अच्छा दिखता है। तत्काल कोड निम्नलिखित कोड की मदद से किया जा सकता है -

with ProcessPoolExecutor(max_workers = 5) as executor

उदाहरण

बेहतर समझ के लिए, हम उसी उदाहरण का उपयोग कर रहे हैं जैसे कि थ्रेड पूल बनाते समय। इस उदाहरण में, हमें आयात करके शुरू करने की आवश्यकता हैconcurrent.futuresमापांक। फिर एक समारोह का नामload_url()बनाया गया है जो अनुरोधित url को लोड करेगा। ProcessPoolExecutorइसके बाद पूल में 5 नंबरों के धागे बनाए जाते हैं। प्रक्रियाPoolExecutorसंदर्भ प्रबंधक के रूप में उपयोग किया गया है। हम भविष्य के परिणाम को कॉल करके प्राप्त कर सकते हैंresult() इस पर विधि।

import concurrent.futures
from concurrent.futures import ProcessPoolExecutor
import urllib.request

URLS = ['http://www.foxnews.com/',
   'http://www.cnn.com/',
   'http://europe.wsj.com/',
   'http://www.bbc.co.uk/',
   'http://some-made-up-domain.com/']

def load_url(url, timeout):
   with urllib.request.urlopen(url, timeout = timeout) as conn:
      return conn.read()

def main():
   with concurrent.futures.ProcessPoolExecutor(max_workers=5) as executor:
      future_to_url = {executor.submit(load_url, url, 60): url for url in URLS}
      for future in concurrent.futures.as_completed(future_to_url):
      url = future_to_url[future]
      try:
         data = future.result()
      except Exception as exc:
         print('%r generated an exception: %s' % (url, exc))
      else:
         print('%r page is %d bytes' % (url, len(data)))

if __name__ == '__main__':
   main()

उत्पादन

उपरोक्त पायथन लिपि निम्न आउटपुट उत्पन्न करेगी -

'http://some-made-up-domain.com/' generated an exception: <urlopen error [Errno 11004] getaddrinfo failed>
'http://www.foxnews.com/' page is 229476 bytes
'http://www.cnn.com/' page is 165323 bytes
'http://www.bbc.co.uk/' page is 284981 bytes
'http://europe.wsj.com/' page is 967575 bytes

Executor.map () फ़ंक्शन का उपयोग

अजगर map()फ़ंक्शन का उपयोग कई कार्यों को करने के लिए व्यापक रूप से किया जाता है। ऐसा एक कार्य पुनरावृत्तियों के भीतर प्रत्येक तत्व के लिए एक निश्चित कार्य लागू करना है। इसी तरह, हम एक इट्रेटर के सभी तत्वों को एक फंक्शन में मैप कर सकते हैं और इन्हें स्वतंत्र जॉब्स के रूप में सबमिट कर सकते हैंProcessPoolExecutor। इसे समझने के लिए पायथन लिपि के निम्नलिखित उदाहरण पर विचार करें।

उदाहरण

हम उसी उदाहरण पर विचार करेंगे जिसका उपयोग हमने थ्रेड पूल बनाते समय किया था Executor.map()समारोह। दिए गए उदाहरण में, मानचित्र फ़ंक्शन का उपयोग लागू करने के लिए किया जाता हैsquare() मान सरणी में प्रत्येक मान पर कार्य करें।

from concurrent.futures import ProcessPoolExecutor
from concurrent.futures import as_completed
values = [2,3,4,5]
def square(n):
   return n * n
def main():
   with ProcessPoolExecutor(max_workers = 3) as executor:
      results = executor.map(square, values)
   for result in results:
      print(result)
if __name__ == '__main__':
   main()

उत्पादन

उपरोक्त पायथन लिपि निम्न आउटपुट उत्पन्न करेगी

4
9
16
25

ProcessPoolExecutor और ThreadPoolExecutor का उपयोग कब करें?

अब जब हमने एक्ज़ीक्यूटर क्लासेस - थ्रेडपूल एक्ज़ीक्यूटर और प्रोसेसपूल एक्सक्यूटोर दोनों के बारे में अध्ययन किया है, तो हमें यह जानना होगा कि किस एग्ज़ॅक्टर को कब इस्तेमाल करना है। हमें CPU / बाउंड वर्कलोड के मामले में ProcessPoolExecutor और I / O- बाउंड वर्कलोड के मामले में ThreadPoolExecutor चुनने की आवश्यकता है।

अगर हम उपयोग करते हैं ProcessPoolExecutor, तब हमें जीआईएल के बारे में चिंता करने की आवश्यकता नहीं है क्योंकि यह मल्टीप्रोसेसिंग का उपयोग करता है। इसके अलावा, निष्पादन समय कम होगा जब तुलना की जाएगीThreadPoolExecution। इसे समझने के लिए निम्नलिखित पायथन लिपि उदाहरण पर विचार करें।

उदाहरण

import time
import concurrent.futures

value = [8000000, 7000000]

def counting(n):
   start = time.time()
   while n > 0:
      n -= 1
   return time.time() - start

def main():
   start = time.time()
   with concurrent.futures.ProcessPoolExecutor() as executor:
      for number, time_taken in zip(value, executor.map(counting, value)):
         print('Start: {} Time taken: {}'.format(number, time_taken))
   print('Total time taken: {}'.format(time.time() - start))

if __name__ == '__main__':
main()

उत्पादन

Start: 8000000 Time taken: 1.5509998798370361
Start: 7000000 Time taken: 1.3259999752044678
Total time taken: 2.0840001106262207

Example- Python script with ThreadPoolExecutor:
import time
import concurrent.futures

value = [8000000, 7000000]

def counting(n):
   start = time.time()
   while n > 0:
      n -= 1
   return time.time() - start

def main():
   start = time.time()
   with concurrent.futures.ThreadPoolExecutor() as executor:
      for number, time_taken in zip(value, executor.map(counting, value)):
         print('Start: {} Time taken: {}'.format(number, time_taken))
      print('Total time taken: {}'.format(time.time() - start))

if __name__ == '__main__':
main()

उत्पादन

Start: 8000000 Time taken: 3.8420000076293945
Start: 7000000 Time taken: 3.6010000705718994
Total time taken: 3.8480000495910645

उपरोक्त दोनों कार्यक्रमों के आउटपुट से, हम उपयोग करते समय निष्पादन समय का अंतर देख सकते हैं ProcessPoolExecutor तथा ThreadPoolExecutor

इस अध्याय में, हम मल्टीप्रोसेसिंग और मल्टीथ्रेडिंग के बीच तुलना पर अधिक ध्यान केंद्रित करेंगे।

बहु

यह एक एकल कंप्यूटर प्रणाली के भीतर दो या अधिक CPU इकाइयों का उपयोग है। हमारे कंप्यूटर सिस्टम में उपलब्ध CPU कोर की पूरी संख्या का उपयोग करके हमारे हार्डवेयर से पूरी क्षमता प्राप्त करना सबसे अच्छा तरीका है।

बहु सूत्रण

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

निम्न तालिका उनके बीच कुछ महत्वपूर्ण अंतर दिखाती है -

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

वैश्विक दुभाषिया लॉक (GIL) के प्रभाव को खत्म करना

समवर्ती अनुप्रयोगों के साथ काम करते समय, पायथन में मौजूद एक सीमा होती है जिसे कहा जाता है GIL (Global Interpreter Lock)। जीआईएल हमें सीपीयू के कई कोर का उपयोग करने की अनुमति नहीं देता है और इसलिए हम कह सकते हैं कि पायथन में कोई सच्चे धागे नहीं हैं। जीआईएल म्यूटेक्स - म्यूचुअल एक्सक्लूजन लॉक है, जो चीजों को सुरक्षित रखता है। दूसरे शब्दों में, हम कह सकते हैं कि जीआईएल कई धागों को समानांतर में पायथन कोड को निष्पादित करने से रोकता है। ताला को एक समय में केवल एक धागे से रखा जा सकता है और अगर हम किसी धागे को निष्पादित करना चाहते हैं तो उसे पहले ताला प्राप्त करना होगा।

मल्टीप्रोसेसिंग के उपयोग के साथ, हम GIL द्वारा सीमित सीमा को प्रभावी ढंग से बायपास कर सकते हैं -

  • मल्टीप्रोसेसिंग का उपयोग करके, हम कई प्रक्रियाओं की क्षमता का उपयोग कर रहे हैं और इसलिए हम GIL के कई उदाहरणों का उपयोग कर रहे हैं।

  • इसके कारण, किसी भी समय हमारे कार्यक्रमों के भीतर एक धागे के बाइटकोड को निष्पादित करने का कोई प्रतिबंध नहीं है।

पायथन में प्रक्रियाएं शुरू करना

मल्टीप्रोसेसिंग मॉड्यूल के भीतर पायथन में एक प्रक्रिया शुरू करने के लिए निम्नलिखित तीन विधियों का उपयोग किया जा सकता है -

  • Fork
  • Spawn
  • Forkserver

फोर्क के साथ एक प्रक्रिया बनाना

फोर्क कमांड UNIX में पाया जाने वाला एक मानक कमांड है। इसका उपयोग बाल प्रक्रिया नामक नई प्रक्रियाओं को बनाने के लिए किया जाता है। इस बच्चे की प्रक्रिया समवर्ती प्रक्रिया के साथ चलती है जिसे मूल प्रक्रिया कहा जाता है। ये बच्चे प्रक्रियाएं अपनी मूल प्रक्रियाओं के समान होती हैं और माता-पिता के लिए उपलब्ध सभी संसाधनों को प्राप्त करती हैं। फोर्क के साथ एक प्रक्रिया बनाते समय निम्नलिखित सिस्टम कॉल का उपयोग किया जाता है -

  • fork()- यह एक सिस्टम कॉल है जिसे आम तौर पर कर्नेल में लागू किया जाता है। इसका उपयोग प्रक्रिया की एक प्रति बनाने के लिए किया जाता है। p>

  • getpid() - यह सिस्टम कॉल कॉलिंग प्रक्रिया की प्रक्रिया आईडी (PID) लौटाता है।

उदाहरण

निम्नलिखित पायथन लिपि का उदाहरण आपको समझने में मदद करेगा कि एक नई बाल प्रक्रिया कैसे बनाई जाए और बच्चे और माता-पिता की प्रक्रिया के पीआईडी ​​प्राप्त करें।

import os

def child():
   n = os.fork()
   
   if n > 0:
      print("PID of Parent process is : ", os.getpid())

   else:
      print("PID of Child process is : ", os.getpid())
child()

उत्पादन

PID of Parent process is : 25989
PID of Child process is : 25990

स्पॉन के साथ एक प्रक्रिया बनाना

स्पॉन का मतलब है कुछ नया शुरू करना। इसलिए, एक प्रक्रिया को जन्म देने का अर्थ है एक मूल प्रक्रिया द्वारा एक नई प्रक्रिया का निर्माण। मूल प्रक्रिया अपने निष्पादन को अतुल्यकालिक रूप से जारी रखती है या तब तक इंतजार करती है जब तक कि बच्चे की प्रक्रिया उसके निष्पादन को समाप्त नहीं कर देती। एक प्रक्रिया पैदा करने के लिए इन चरणों का पालन करें -

  • मल्टीप्रोसेसिंग मॉड्यूल आयात करना।

  • वस्तु प्रक्रिया बनाना।

  • कॉल करके प्रक्रिया गतिविधि शुरू करना start() तरीका।

  • तब तक इंतजार करना जब तक कि प्रक्रिया ने अपना काम पूरा नहीं कर लिया हो और कॉल करके बाहर नहीं निकल गया join() तरीका।

उदाहरण

पायथन लिपि का निम्नलिखित उदाहरण तीन प्रक्रियाओं को पैदा करने में मदद करता है

import multiprocessing

def spawn_process(i):
   print ('This is process: %s' %i)
   return

if __name__ == '__main__':
   Process_jobs = []
   for i in range(3):
   p = multiprocessing.Process(target = spawn_process, args = (i,))
      Process_jobs.append(p)
   p.start()
   p.join()

उत्पादन

This is process: 0
This is process: 1
This is process: 2

Forkserver के साथ एक प्रक्रिया बनाना

Forkserver तंत्र केवल उन चयनित UNIX प्लेटफार्मों पर उपलब्ध है जो Unix पाइप्स पर फ़ाइल डिस्क्रिप्टर को पास करने का समर्थन करते हैं। Forkserver तंत्र के कार्य को समझने के लिए निम्नलिखित बिंदुओं पर विचार करें -

  • नई प्रक्रिया शुरू करने के लिए Forkserver तंत्र का उपयोग करने पर एक सर्वर को इंस्टेंट किया जाता है।

  • सर्वर तब कमांड प्राप्त करता है और नई प्रक्रियाएं बनाने के लिए सभी अनुरोधों को संभालता है।

  • एक नई प्रक्रिया बनाने के लिए, हमारा अजगर कार्यक्रम फोर्स्कवर को एक अनुरोध भेजेगा और यह हमारे लिए एक प्रक्रिया तैयार करेगा।

  • अंत में, हम अपने कार्यक्रमों में इस नई बनाई गई प्रक्रिया का उपयोग कर सकते हैं।

पायथन में डेमन प्रक्रियाएं

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

उदाहरण

यहां, हम उसी उदाहरण का उपयोग कर रहे हैं जैसा कि डेमन थ्रेड्स में उपयोग किया जाता है। एकमात्र अंतर मॉड्यूल के परिवर्तन से हैmultithreading सेवा multiprocessingऔर डेमोनिक झंडे को सच करना। हालाँकि, आउटपुट में बदलाव होगा जैसा कि नीचे दिखाया गया है -

import multiprocessing
import time

def nondaemonProcess():
   print("starting my Process")
   time.sleep(8)
   print("ending my Process")
def daemonProcess():
   while True:
   print("Hello")
   time.sleep(2)
if __name__ == '__main__':
   nondaemonProcess = multiprocessing.Process(target = nondaemonProcess)
   daemonProcess = multiprocessing.Process(target = daemonProcess)
   daemonProcess.daemon = True
   nondaemonProcess.daemon = False
   daemonProcess.start()
   nondaemonProcess.start()

उत्पादन

starting my Process
ending my Process

डेमन थ्रेड्स द्वारा उत्पन्न एक की तुलना में आउटपुट अलग है, क्योंकि नो डेमन मोड में प्रक्रिया का आउटपुट होता है। इसलिए, मुख्य प्रक्रियाओं के चलने की प्रक्रियाओं की दृढ़ता से बचने के लिए डेमोनिक प्रक्रिया स्वतः समाप्त हो जाती है।

पायथन में प्रक्रियाओं को समाप्त करना

हम किसी प्रक्रिया का उपयोग करके तुरंत उसे मार सकते हैं या समाप्त कर सकते हैं terminate()तरीका। हम इस पद्धति का उपयोग बच्चे की प्रक्रिया को समाप्त करने के लिए करेंगे, जो कि इसके निष्पादन को पूरा करने से ठीक पहले फ़ंक्शन की मदद से बनाया गया है।

उदाहरण

import multiprocessing
import time
def Child_process():
   print ('Starting function')
   time.sleep(5)
   print ('Finished function')
P = multiprocessing.Process(target = Child_process)
P.start()
print("My Process has terminated, terminating main thread")
print("Terminating Child Process")
P.terminate()
print("Child Process successfully terminated")

उत्पादन

My Process has terminated, terminating main thread
Terminating Child Process
Child Process successfully terminated

आउटपुट दिखाता है कि प्रोग्राम चाइल्ड प्रोसेस के निष्पादन से पहले समाप्त हो जाता है जिसे चाइल्डप्रोसेस () फ़ंक्शन की सहायता से बनाया गया है। इसका तात्पर्य यह है कि बच्चे की प्रक्रिया को सफलतापूर्वक समाप्त कर दिया गया है।

पायथन में वर्तमान प्रक्रिया की पहचान करना

ऑपरेटिंग सिस्टम की प्रत्येक प्रक्रिया में PID के रूप में ज्ञात प्रक्रिया पहचान है। पायथन में, हम निम्नलिखित कमांड की मदद से वर्तमान प्रक्रिया के पीआईडी ​​का पता लगा सकते हैं -

import multiprocessing
print(multiprocessing.current_process().pid)

उदाहरण

पायथन लिपि का निम्नलिखित उदाहरण मुख्य प्रक्रिया के पीआईडी ​​के साथ-साथ बाल प्रक्रिया के पीआईडी ​​का पता लगाने में मदद करता है -

import multiprocessing
import time
def Child_process():
   print("PID of Child Process is: {}".format(multiprocessing.current_process().pid))
print("PID of Main process is: {}".format(multiprocessing.current_process().pid))
P = multiprocessing.Process(target=Child_process)
P.start()
P.join()

उत्पादन

PID of Main process is: 9401
PID of Child Process is: 9402

उपवर्ग में एक प्रक्रिया का उपयोग करना

हम उप-वर्गीकृत करके धागे बना सकते हैं threading.Threadकक्षा। इसके अतिरिक्त, हम सब-क्लासिंग करके भी प्रक्रियाएँ बना सकते हैंmultiprocessing.Processकक्षा। उपवर्ग में एक प्रक्रिया का उपयोग करने के लिए, हमें निम्नलिखित बिंदुओं पर विचार करने की आवश्यकता है -

  • हमें एक नए उपवर्ग को परिभाषित करने की आवश्यकता है Process कक्षा।

  • हमें इससे आगे निकलने की जरूरत है _init_(self [,args] ) कक्षा।

  • हमें ओवरराइड करने की जरूरत है run(self [,args] ) क्या लागू करने के लिए विधि Process

  • हमें इनवाइट करके प्रक्रिया शुरू करने की जरूरत हैstart() तरीका।

उदाहरण

import multiprocessing
class MyProcess(multiprocessing.Process):
   def run(self):
   print ('called run method in process: %s' %self.name)
   return
if __name__ == '__main__':
   jobs = []
   for i in range(5):
   P = MyProcess()
   jobs.append(P)
   P.start()
   P.join()

उत्पादन

called run method in process: MyProcess-1
called run method in process: MyProcess-2
called run method in process: MyProcess-3
called run method in process: MyProcess-4
called run method in process: MyProcess-5

पायथन मल्टीप्रोसेसिंग मॉड्यूल - पूल क्लास

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

लागू () विधि

यह विधि के समान है.submit()उसकि विधि .ThreadPoolExecutor.यह परिणाम तैयार होने तक अवरुद्ध रहता है।

apply_async () विधि

जब हमें अपने कार्यों के समानांतर निष्पादन की आवश्यकता होती है तो हमें इसका उपयोग करने की आवश्यकता होती हैapply_async()पूल में कार्य प्रस्तुत करने की विधि। यह एक अतुल्यकालिक ऑपरेशन है जो मुख्य थ्रेड को तब तक लॉक नहीं करेगा जब तक कि सभी बच्चे की प्रक्रिया निष्पादित न हो जाए।

नक्शा () विधि

ठीक वैसे ही apply()विधि, यह परिणाम तैयार होने तक भी अवरुद्ध करता है। यह बिल्ट-इन के बराबर हैmap() फ़ंक्शन जो विभिन्न डेटा को विभिन्न कार्यों के रूप में प्रक्रिया पूल में पुनरावृत्त डेटा को विभाजित करता है और सबमिट करता है।

map_async () विधि

यह इसका एक प्रकार है map() विधि के रूप में apply_async() को है apply()तरीका। यह एक परिणाम वस्तु देता है। जब परिणाम तैयार हो जाता है, तो उस पर एक कॉल करने योग्य लागू होता है। कॉल करने योग्य को तुरंत पूरा किया जाना चाहिए; अन्यथा, परिणाम को संभालने वाला धागा अवरुद्ध हो जाएगा।

उदाहरण

निम्नलिखित उदाहरण आपको समानांतर निष्पादन करने के लिए एक प्रक्रिया पूल को लागू करने में मदद करेगा। संख्या के वर्ग की एक सरल गणना को लागू करके किया गया हैsquare() के माध्यम से कार्य करते हैं multiprocessing.Poolतरीका। फिरpool.map() 5 का उपयोग करने के लिए उपयोग किया गया है, क्योंकि इनपुट 0 से 4 तक पूर्णांकों की एक सूची है। परिणाम में संग्रहीत किया जाएगा p_outputs और यह छपा है।

def square(n):
   result = n*n
   return result
if __name__ == '__main__':
   inputs = list(range(5))
   p = multiprocessing.Pool(processes = 4)
   p_outputs = pool.map(function_square, inputs)
   p.close()
   p.join()
   print ('Pool :', p_outputs)

उत्पादन

Pool : [0, 1, 4, 9, 16]

प्रोसेस इंटरकॉम्यूनिकेशन का अर्थ है प्रक्रियाओं के बीच डेटा का आदान-प्रदान। समानांतर अनुप्रयोग के विकास के लिए प्रक्रियाओं के बीच डेटा का आदान-प्रदान करना आवश्यक है। निम्नलिखित आरेख कई उप प्रक्रियाओं के बीच सिंक्रनाइज़ेशन के लिए विभिन्न संचार तंत्र दिखाता है -

विभिन्न संचार तंत्र

इस खंड में, हम विभिन्न संचार तंत्रों के बारे में जानेंगे। तंत्र नीचे वर्णित हैं -

कतारों

बहु-प्रक्रिया कार्यक्रमों के साथ कतारों का उपयोग किया जा सकता है। की कतार वर्गmultiprocessing मॉड्यूल के समान है Queue.Queueकक्षा। इसलिए, एक ही एपीआई का उपयोग किया जा सकता है।Multiprocessing.Queue हमें एक थ्रेड प्रदान करता है और प्रक्रियाओं के बीच संचार का सुरक्षित फीफो (फर्स्ट-इन फर्स्ट-आउट) तंत्र प्रदान करता है।

उदाहरण

मल्टीप्रोसेसिंग के क्यू क्लास की अवधारणा को समझने के लिए मल्टीप्रोसेसिंग पर अजगर आधिकारिक डॉक्स से लिया गया एक सरल उदाहरण है।

from multiprocessing import Process, Queue
import queue
import random
def f(q):
   q.put([42, None, 'hello'])
def main():
   q = Queue()
   p = Process(target = f, args = (q,))
   p.start()
   print (q.get())
if __name__ == '__main__':
   main()

उत्पादन

[42, None, 'hello']

पाइप्स

यह एक डेटा संरचना है, जिसका उपयोग बहु-प्रक्रिया कार्यक्रमों में प्रक्रियाओं के बीच संवाद करने के लिए किया जाता है। पाइप () फ़ंक्शन पाइप से जुड़े कनेक्शन ऑब्जेक्ट की एक जोड़ी देता है जो डिफ़ॉल्ट रूप से द्वैध (दो तरह से) है। यह निम्नलिखित तरीके से काम करता है -

  • यह कनेक्शन ऑब्जेक्ट्स की एक जोड़ी देता है जो पाइप के दो सिरों का प्रतिनिधित्व करते हैं।

  • प्रत्येक वस्तु के दो तरीके हैं - send() तथा recv(), प्रक्रियाओं के बीच संवाद करने के लिए।

उदाहरण

निम्नलिखित अवधारणा को समझने के लिए मल्टीप्रोसेसिंग पर अजगर आधिकारिक डॉक्स से लिया गया एक सरल उदाहरण है Pipe() मल्टीप्रोसेसिंग का कार्य।

from multiprocessing import Process, Pipe

def f(conn):
   conn.send([42, None, 'hello'])
   conn.close()

if __name__ == '__main__':
   parent_conn, child_conn = Pipe()
   p = Process(target = f, args = (child_conn,))
   p.start()
   print (parent_conn.recv())
   p.join()

उत्पादन

[42, None, 'hello']

मैनेजर

प्रबंधक एक मल्टीप्रोसेसिंग मॉड्यूल का एक वर्ग है जो अपने सभी उपयोगकर्ताओं के बीच साझा जानकारी को समन्वयित करने का एक तरीका प्रदान करता है। एक प्रबंधक ऑब्जेक्ट एक सर्वर प्रक्रिया को नियंत्रित करता है, जो साझा वस्तुओं का प्रबंधन करता है और अन्य प्रक्रियाओं को उन्हें हेरफेर करने की अनुमति देता है। दूसरे शब्दों में, प्रबंधक डेटा बनाने का एक तरीका प्रदान करते हैं जिसे विभिन्न प्रक्रियाओं के बीच साझा किया जा सकता है। प्रबंधक वस्तु के विभिन्न गुण निम्नलिखित हैं -

  • प्रबंधक की मुख्य संपत्ति एक सर्वर प्रक्रिया को नियंत्रित करना है, जो साझा वस्तुओं का प्रबंधन करती है।

  • एक अन्य महत्वपूर्ण संपत्ति सभी साझा किए गए ऑब्जेक्ट को अपडेट करना है जब कोई भी प्रक्रिया इसे संशोधित करती है।

उदाहरण

निम्नलिखित एक उदाहरण है जो सर्वर प्रक्रिया में एक सूची रिकॉर्ड बनाने और फिर उस सूची में एक नया रिकॉर्ड जोड़ने के लिए प्रबंधक ऑब्जेक्ट का उपयोग करता है।

import multiprocessing

def print_records(records):
   for record in records:
      print("Name: {0}\nScore: {1}\n".format(record[0], record[1]))

def insert_record(record, records):
   records.append(record)
      print("A New record is added\n")

if __name__ == '__main__':
   with multiprocessing.Manager() as manager:

      records = manager.list([('Computers', 1), ('Histoty', 5), ('Hindi',9)])
      new_record = ('English', 3)

      p1 = multiprocessing.Process(target = insert_record, args = (new_record, records))
      p2 = multiprocessing.Process(target = print_records, args = (records,))
	  p1.start()
      p1.join()
      p2.start()
      p2.join()

उत्पादन

A New record is added

Name: Computers
Score: 1

Name: Histoty
Score: 5

Name: Hindi
Score: 9

Name: English
Score: 3

प्रबंधक में नाम स्थान की अवधारणा

प्रबंधक वर्ग नेमस्पेस की अवधारणा के साथ आता है, जो कई प्रक्रियाओं में कई विशेषताओं को साझा करने का एक त्वरित तरीका है। नाम स्थान में कोई सार्वजनिक पद्धति नहीं है, जिसे कहा जा सकता है, लेकिन उनके पास लिखने योग्य गुण हैं।

उदाहरण

निम्नलिखित पायथन लिपि उदाहरण हमें मुख्य प्रक्रिया और बाल प्रक्रिया में डेटा साझा करने के लिए नामस्थान का उपयोग करने में मदद करता है -

import multiprocessing

def Mng_NaSp(using_ns):

   using_ns.x +=5
   using_ns.y *= 10

if __name__ == '__main__':
   manager = multiprocessing.Manager()
   using_ns = manager.Namespace()
   using_ns.x = 1
   using_ns.y = 1

   print ('before', using_ns)
   p = multiprocessing.Process(target = Mng_NaSp, args = (using_ns,))
   p.start()
   p.join()
   print ('after', using_ns)

उत्पादन

before Namespace(x = 1, y = 1)
after Namespace(x = 6, y = 10)

Ctypes- ऐरे और वैल्यू

मल्टीप्रोसेसिंग मॉड्यूल साझा मेमोरी मैप में डेटा को संग्रहीत करने के लिए एरे और वैल्यू ऑब्जेक्ट प्रदान करता है। Array साझा मेमोरी से आवंटित एक ctypes सरणी है और Value साझा मेमोरी से आवंटित एक ctypes ऑब्जेक्ट है।

मल्टीप्रोसेसिंग से आयात प्रक्रिया, मूल्य, एरियर के साथ होना।

उदाहरण

पायथन स्क्रिप्ट निम्नलिखित प्रक्रियाओं के बीच कुछ डेटा साझा करने के लिए Ctypes सरणी और मान का उपयोग करने के लिए अजगर डॉक्स से लिया गया एक उदाहरण है।

def f(n, a):
   n.value = 3.1415927
   for i in range(len(a)):
   a[i] = -a[i]

if __name__ == '__main__':
   num = Value('d', 0.0)
   arr = Array('i', range(10))

   p = Process(target = f, args = (num, arr))
   p.start()
   p.join()
   print (num.value)
   print (arr[:])

उत्पादन

3.1415927
[0, -1, -2, -3, -4, -5, -6, -7, -8, -9]

संचारी प्रक्रियाएं (CSP)

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

पायथन लाइब्रेरी - PyCSP

सीएसपी में पाए जाने वाले मुख्य आदिम को लागू करने के लिए, पायथन में एक पुस्तकालय है जिसे PyCSP कहा जाता है। यह कार्यान्वयन को बहुत छोटा और पठनीय रखता है ताकि इसे बहुत आसानी से समझा जा सके। निम्नलिखित PyCSP का मूल प्रक्रिया नेटवर्क है -

उपरोक्त PyCSP प्रक्रिया नेटवर्क में, दो प्रक्रियाएँ हैं - Process1 और प्रक्रिया 2. ये प्रक्रियाएँ दो चैनलों - चैनल 1 और चैनल 2 के माध्यम से संदेश भेजकर संवाद करती हैं।

PyCSP स्थापित करना

निम्नलिखित आदेश की मदद से, हम पायथन लाइब्रेरी PyCSP स्थापित कर सकते हैं -

pip install PyCSP

उदाहरण

पायथन लिपि का अनुसरण करना एक दूसरे के समानांतर दो प्रक्रियाओं को चलाने के लिए एक सरल उदाहरण है। यह PyCSP पाइथन लिबरी की मदद से किया जाता है -

from pycsp.parallel import *
import time
@process
def P1():
   time.sleep(1)
   print('P1 exiting')
@process
def P2():
   time.sleep(1)
   print('P2 exiting')
def main():
   Parallel(P1(), P2())
   print('Terminating')
if __name__ == '__main__':
   main()

उपरोक्त लिपि में, दो कार्य अर्थात् P1 तथा P2 बनाया गया है और फिर से सजाया गया है @process उन्हें प्रक्रियाओं में परिवर्तित करने के लिए।

उत्पादन

P2 exiting
P1 exiting
Terminating

इवेंट-संचालित प्रोग्रामिंग घटनाओं पर केंद्रित है। आखिरकार, कार्यक्रम का प्रवाह घटनाओं पर निर्भर करता है। अब तक, हम या तो अनुक्रमिक या समानांतर निष्पादन मॉडल के साथ काम कर रहे थे लेकिन इवेंट-संचालित प्रोग्रामिंग की अवधारणा वाले मॉडल को एसिंक्रोनस मॉडल कहा जाता है। ईवेंट-चालित प्रोग्रामिंग एक इवेंट लूप पर निर्भर करती है जो हमेशा नए आने वाले ईवेंट के लिए सुन रही है। इवेंट-संचालित प्रोग्रामिंग का कार्य घटनाओं पर निर्भर है। एक बार जब कोई घटना घटती है, तो घटनाएं तय करती हैं कि किस पर अमल करना है और किस क्रम में। फ्लोचार्ट का अनुसरण करने से आपको यह समझने में मदद मिलेगी कि यह कैसे काम करता है -

पायथन मॉड्यूल - एसिंसीओ

Asyncio मॉड्यूल Python 3.4 में जोड़ा गया था और यह सह-दिनचर्या का उपयोग करके एकल-थ्रेडेड समवर्ती कोड लिखने के लिए बुनियादी ढांचा प्रदान करता है। Asyncio मॉड्यूल द्वारा उपयोग की जाने वाली विभिन्न अवधारणाएँ निम्नलिखित हैं -

घटना पाश

इवेंट-लूप एक कम्प्यूटेशनल कोड में सभी घटनाओं को संभालने के लिए एक कार्यक्षमता है। यह पूरे कार्यक्रम के निष्पादन के दौरान रास्ते का काम करता है और आने वाली घटनाओं को ट्रैक करता है। Asyncio मॉड्यूल प्रति प्रक्रिया एक एकल ईवेंट लूप की अनुमति देता है। इवेंट लूप का प्रबंधन करने के लिए असिनसियो मॉड्यूल द्वारा कुछ तरीके दिए गए हैं -

  • loop = get_event_loop() - यह विधि वर्तमान संदर्भ के लिए ईवेंट लूप प्रदान करेगी।

  • loop.call_later(time_delay,callback,argument) - यह विधि कॉलबैक के लिए व्यवस्था करती है जिसे दिए गए time_delay सेकंड के बाद कॉल किया जाना है।

  • loop.call_soon(callback,argument)- यह विधि एक कॉलबैक के लिए व्यवस्था करती है जिसे जल्द से जल्द बुलाया जाना है। कॉलबैक को call_soon () रिटर्न के बाद कॉल किया जाता है और जब कंट्रोल ईवेंट लूप में लौटता है।

  • loop.time() - इस विधि का उपयोग इवेंट लूप की आंतरिक घड़ी के अनुसार वर्तमान समय को वापस करने के लिए किया जाता है।

  • asyncio.set_event_loop() - यह विधि वर्तमान लूप के लिए ईवेंट लूप को लूप पर सेट करेगी।

  • asyncio.new_event_loop() - यह विधि एक नया ईवेंट लूप ऑब्जेक्ट बनाएगी और लौटाएगी।

  • loop.run_forever() - यह विधि तब तक चलेगी जब तक कि स्टॉप () विधि को नहीं कहा जाता है।

उदाहरण

इवेंट लूप का निम्न उदाहरण मुद्रण में मदद करता है hello worldget_event_loop () विधि का उपयोग करके। यह उदाहरण पायथन आधिकारिक डॉक्स से लिया गया है।

import asyncio

def hello_world(loop):
   print('Hello World')
   loop.stop()

loop = asyncio.get_event_loop()

loop.call_soon(hello_world, loop)

loop.run_forever()
loop.close()

उत्पादन

Hello World

फ्यूचर्स

यह समवर्ती.फुटर्स के साथ संगत है। सीवन वर्ग जो एक अभिकलन का प्रतिनिधित्व करता है जिसे पूरा नहीं किया गया है। Asyncio.futures.Future और concurrent.futures.Future के बीच अंतर निम्नलिखित हैं।

  • परिणाम () और अपवाद () विधियाँ समयबाह्य तर्क नहीं लेती हैं और जब भविष्य अभी तक पूरा नहीं हुआ है तो एक अपवाद बढ़ाते हैं।

  • Add_done_callback () के साथ पंजीकृत कॉलबैक को हमेशा इवेंट लूप के call_soon () के माध्यम से कहा जाता है।

  • asyncio.futures। सीवन वर्ग प्रतीक्षा के साथ संगत नहीं है () और as_completed () कार्यों समवर्ती.फुट्स पैकेज में।

उदाहरण

निम्नलिखित एक उदाहरण है जो आपको यह समझने में मदद करेगा कि asyncio.futures.future वर्ग का उपयोग कैसे करें।

import asyncio

async def Myoperation(future):
   await asyncio.sleep(2)
   future.set_result('Future Completed')

loop = asyncio.get_event_loop()
future = asyncio.Future()
asyncio.ensure_future(Myoperation(future))
try:
   loop.run_until_complete(future)
   print(future.result())
finally:
   loop.close()

उत्पादन

Future Completed

Coroutines

Asyncio में कोरआउट की अवधारणा थ्रेडिंग मॉड्यूल के तहत मानक थ्रेड ऑब्जेक्ट की अवधारणा के समान है। यह सबरूटीन अवधारणा का सामान्यीकरण है। निष्पादन के दौरान एक कोरटाइन को निलंबित किया जा सकता है ताकि यह बाहरी प्रसंस्करण की प्रतीक्षा करे और उस बिंदु से वापस लौटे जिस पर बाहरी प्रसंस्करण किया गया था। निम्नलिखित दो तरीके हमें कोरटाइन लागू करने में मदद करते हैं -

async डीईएफ़ फ़ंक्शन ()

यह Asyncio मॉड्यूल के तहत कोरटाइन के कार्यान्वयन के लिए एक विधि है। निम्नलिखित के लिए एक पायथन स्क्रिप्ट है -

import asyncio

async def Myoperation():
   print("First Coroutine")

loop = asyncio.get_event_loop()
try:
   loop.run_until_complete(Myoperation())

finally:
   loop.close()

उत्पादन

First Coroutine

@ asyncio.coroutine डेकोरेटर

कोरटाइन के कार्यान्वयन के लिए एक अन्य विधि @ asyncio.coroutine डेकोरेटर के साथ जनरेटर का उपयोग करना है। निम्नलिखित के लिए एक पायथन स्क्रिप्ट है -

import asyncio

@asyncio.coroutine
def Myoperation():
   print("First Coroutine")

loop = asyncio.get_event_loop()
try:
   loop.run_until_complete(Myoperation())

finally:
   loop.close()

उत्पादन

First Coroutine

कार्य

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

import asyncio
import time
async def Task_ex(n):
   time.sleep(1)
   print("Processing {}".format(n))
async def Generator_task():
   for i in range(10):
      asyncio.ensure_future(Task_ex(i))
   int("Tasks Completed")
   asyncio.sleep(2)

loop = asyncio.get_event_loop()
loop.run_until_complete(Generator_task())
loop.close()

उत्पादन

Tasks Completed
Processing 0
Processing 1
Processing 2
Processing 3
Processing 4
Processing 5
Processing 6
Processing 7
Processing 8
Processing 9

परिवहन

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

बेसट्रांसपोर्ट से विरासत में प्राप्त विभिन्न प्रकार के परिवहन निम्नलिखित हैं -

  • ReadTransport - यह केवल-पढ़ने के लिए एक इंटरफ़ेस है।

  • WriteTransport - यह केवल-लिखने के लिए एक इंटरफ़ेस है।

  • DatagramTransport - यह डेटा भेजने के लिए एक इंटरफ़ेस है।

  • BaseSubprocessTransport - बेसट्रांसपोर्ट वर्ग के समान।

बेसट्रांसपोर्ट क्लास के पांच अलग-अलग तरीके हैं, जो बाद में चार परिवहन प्रकारों में क्षणिक हैं -

  • close() - यह परिवहन बंद कर देता है।

  • is_closing() - यदि ट्रांसपोर्ट बंद हो रहा है या पहले से ही बंद है। ट्रांसपोर्ट्स।

  • get_extra_info(name, default = none) - यह हमें परिवहन के बारे में कुछ अतिरिक्त जानकारी देगा।

  • get_protocol() - यह विधि वर्तमान प्रोटोकॉल को वापस कर देगी।

प्रोटोकॉल

Asyncio मॉड्यूल आधार वर्ग प्रदान करता है जिसे आप अपने नेटवर्क प्रोटोकॉल को लागू करने के लिए उपवर्ग कर सकते हैं। उन वर्गों को परिवहन के साथ संयोजन में उपयोग किया जाता है; प्रोटोकॉल आने वाले डेटा को पार्स करता है और आउटगोइंग डेटा के लेखन के लिए पूछता है, जबकि परिवहन वास्तविक I / O और बफरिंग के लिए जिम्मेदार है। प्रोटोकॉल के तीन वर्ग निम्नलिखित हैं -

  • Protocol - यह टीसीपी और एसएसएल ट्रांसपोर्ट के साथ स्ट्रीमिंग प्रोटोकॉल को लागू करने के लिए बेस क्लास है।

  • DatagramProtocol - यह UDP ट्रांसपोर्ट के साथ उपयोग के लिए डेटाग्राम प्रोटोकॉल को लागू करने के लिए आधार वर्ग है।

  • SubprocessProtocol - यह यूनिडायरेक्शनल पाइप के एक सेट के माध्यम से बाल प्रक्रियाओं के साथ संचार करने वाले प्रोटोकॉल को लागू करने के लिए आधार वर्ग है।

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

प्रतिक्रियाशील प्रोग्रामिंग के लिए ReactiveX या RX

ReactiveX या Raective Extension रिएक्टिव प्रोग्रामिंग का सबसे प्रसिद्ध कार्यान्वयन है। ReactiveX का काम निम्न दो वर्गों पर निर्भर करता है -

अवलोकनीय वर्ग

यह वर्ग डेटा स्ट्रीम या घटनाओं का स्रोत है और यह आने वाले डेटा को पैक करता है ताकि डेटा को एक थ्रेड से दूसरे में पास किया जा सके। यह तब तक डेटा नहीं देगा जब तक कि कुछ पर्यवेक्षक इसकी सदस्यता नहीं लेते हैं।

प्रेक्षक वर्ग

यह वर्ग द्वारा उत्सर्जित डेटा स्ट्रीम का उपभोग करता है observable। पर्यवेक्षित के साथ कई पर्यवेक्षक हो सकते हैं और प्रत्येक पर्यवेक्षक प्रत्येक डेटा आइटम प्राप्त करेगा जो उत्सर्जित होता है। पर्यवेक्षक अवलोकन योग्य सदस्यता लेकर तीन प्रकार के कार्यक्रम प्राप्त कर सकता है -

  • on_next() event - इसका मतलब है कि डेटा स्ट्रीम में एक तत्व है।

  • on_completed() event - इसका मतलब है कि उत्सर्जन समाप्त हो रहा है और कोई और वस्तु नहीं आ रही है।

  • on_error() event - यह उत्सर्जन के अंत का भी तात्पर्य है लेकिन जब कोई त्रुटि होती है observable

आरएक्सपीवाई - रिएक्टिव प्रोग्रामिंग के लिए पायथन मॉड्यूल

आरएक्सपीवाई एक पायथन मॉड्यूल है जिसका उपयोग प्रतिक्रियाशील प्रोग्रामिंग के लिए किया जा सकता है। हमें यह सुनिश्चित करने की आवश्यकता है कि मॉड्यूल स्थापित है। RxPY मॉड्यूल को स्थापित करने के लिए निम्नलिखित कमांड का उपयोग किया जा सकता है -

pip install RxPY

उदाहरण

निम्नलिखित एक पायथन स्क्रिप्ट है, जो उपयोग करती है RxPY मॉड्यूल और उसकी कक्षाएं Observable तथा Observe forप्रतिक्रियाशील प्रोग्रामिंग। मूल रूप से दो वर्ग हैं -

  • get_strings() - प्रेक्षक से तार पाने के लिए।

  • PrintObserver()- आब्जर्वर से स्ट्रिंग्स को प्रिंट करने के लिए। इसमें पर्यवेक्षक वर्ग की सभी तीन घटनाओं का उपयोग किया गया है। यह भी सदस्यता () वर्ग का उपयोग करता है।

from rx import Observable, Observer
def get_strings(observer):
   observer.on_next("Ram")
   observer.on_next("Mohan")
   observer.on_next("Shyam")
      observer.on_completed()
class PrintObserver(Observer):
   def on_next(self, value):
      print("Received {0}".format(value))
   def on_completed(self):
   print("Finished")
   def on_error(self, error):
      print("Error: {0}".format(error))
source = Observable.create(get_strings)
source.subscribe(PrintObserver())

उत्पादन

Received Ram
Received Mohan
Received Shyam
Finished

प्रतिक्रियाशील प्रोग्रामिंग के लिए PyFunctional पुस्तकालय

PyFunctionalएक और पायथन लाइब्रेरी है जिसका उपयोग रिएक्टिव प्रोग्रामिंग के लिए किया जा सकता है। यह हमें पायथन प्रोग्रामिंग भाषा का उपयोग करके कार्यात्मक कार्यक्रम बनाने में सक्षम बनाता है। यह उपयोगी है क्योंकि यह हमें जंजीर कार्यात्मक ऑपरेटरों का उपयोग करके डेटा पाइपलाइन बनाने की अनुमति देता है।

RxPY और PyFunctional के बीच अंतर

दोनों पुस्तकालयों का उपयोग प्रतिक्रियाशील प्रोग्रामिंग के लिए किया जाता है और समान फैशन में स्ट्रीम को संभालता है लेकिन दोनों के बीच मुख्य अंतर डेटा की हैंडलिंग पर निर्भर करता है। RxPY सिस्टम में डेटा और घटनाओं को संभालता है PyFunctional कार्यात्मक प्रोग्रामिंग प्रतिमानों का उपयोग करके डेटा के परिवर्तन पर केंद्रित है।

PyFunctional मॉड्यूल स्थापित करना

हमें इसका उपयोग करने से पहले इस मॉड्यूल को स्थापित करना होगा। इसे निम्नानुसार पाइप कमांड की सहायता से स्थापित किया जा सकता है -

pip install pyfunctional

उदाहरण

निम्नलिखित उदाहरण का उपयोग करता है the PyFunctional मॉड्यूल और इसके seqवह वर्ग जो स्ट्रीम ऑब्जेक्ट के रूप में कार्य करता है जिसके साथ हम पुनरावृत्ति और हेरफेर कर सकते हैं। इस कार्यक्रम में, यह लैम्डा फ़ंक्शन का उपयोग करके अनुक्रम को मैप करता है जो हर मूल्य को दोगुना करता है, फिर उस मान को फ़िल्टर करता है जहां x 4 से अधिक है और अंत में यह शेष सभी मानों के योग में अनुक्रम को कम कर देता है।

from functional import seq

result = seq(1,2,3).map(lambda x: x*2).filter(lambda x: x > 4).reduce(lambda x, y: x + y)

print ("Result: {}".format(result))

उत्पादन

Result: 6