पायथन में कंसीडर - थ्रेड्स का पूल

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

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

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

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

पायथन मानक पुस्तकालय में शामिल हैं 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