C ++ बहु फैलाव

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

प्रक्रिया-आधारित मल्टीटास्किंग कार्यक्रमों के समवर्ती निष्पादन को संभालती है। थ्रेड-आधारित मल्टीटास्किंग एक ही कार्यक्रम के टुकड़ों के समवर्ती निष्पादन से संबंधित है।

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

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

यह ट्यूटोरियल मानता है कि आप लिनक्स ओएस पर काम कर रहे हैं और हम POSIX का उपयोग करके बहु-थ्रेडेड C ++ प्रोग्राम लिखने जा रहे हैं। POSIX थ्रेड्स या Pthreads एपीआई प्रदान करता है जो कई यूनिक्स-जैसे POSIX सिस्टम जैसे FreeBSD, NetBSD, GNU / Linux, Mac OS X और Solaris पर उपलब्ध हैं।

सूत्र बनाना

POSIX थ्रेड बनाने के लिए निम्न दिनचर्या का उपयोग किया जाता है -

#include <pthread.h>
pthread_create (thread, attr, start_routine, arg)

यहाँ, pthread_createएक नया धागा बनाता है और इसे निष्पादन योग्य बनाता है। इस दिनचर्या को आपके कोड के भीतर कहीं से भी किसी भी समय कहा जा सकता है। यहाँ मापदंडों का वर्णन है -

अनु क्रमांक पैरामीटर और विवरण
1

thread

एक अपारदर्शी, नए धागे के लिए अद्वितीय पहचानकर्ता सबरूटीन द्वारा लौट आया।

2

attr

एक अपारदर्शी विशेषता ऑब्जेक्ट जिसका उपयोग थ्रेड विशेषताएँ सेट करने के लिए किया जा सकता है। आप डिफ़ॉल्ट विशेषताओं के लिए एक थ्रेड विशेषताएँ ऑब्जेक्ट या NULL निर्दिष्ट कर सकते हैं।

3

start_routine

C ++ रूटीन जो थ्रेड को बनाते समय निष्पादित करेगा।

4

arg

एक तर्क जो start_routine को पास किया जा सकता है। इसे संदर्भ द्वारा टाइप शून्य के सूचक कलाकारों के रूप में पारित किया जाना चाहिए। यदि कोई तर्क पारित नहीं करना है तो NULL का उपयोग किया जा सकता है।

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

थ्रेडिंग समाप्त करना

निम्नलिखित दिनचर्या है जो हम एक POSIX थ्रेड को समाप्त करने के लिए उपयोग करते हैं -

#include <pthread.h>
pthread_exit (status)

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

यदि मुख्य () उसके द्वारा बनाए गए थ्रेड्स से पहले खत्म हो जाता है, और pthread_exit () के साथ बाहर निकल जाता है, तो अन्य थ्रेड निष्पादित होते रहेंगे। अन्यथा, वे स्वचालित रूप से समाप्त हो जाएंगे जब मुख्य () खत्म हो जाएगा।

Example

यह सरल उदाहरण कोड pthread_create () दिनचर्या के साथ 5 धागे बनाता है। प्रत्येक थ्रेड प्रिंट करता है "हैलो वर्ल्ड!" संदेश, और फिर pthread_exit () को कॉल के साथ समाप्त होता है।

#include <iostream>
#include <cstdlib>
#include <pthread.h>

using namespace std;

#define NUM_THREADS 5

void *PrintHello(void *threadid) {
   long tid;
   tid = (long)threadid;
   cout << "Hello World! Thread ID, " << tid << endl;
   pthread_exit(NULL);
}

int main () {
   pthread_t threads[NUM_THREADS];
   int rc;
   int i;
   
   for( i = 0; i < NUM_THREADS; i++ ) {
      cout << "main() : creating thread, " << i << endl;
      rc = pthread_create(&threads[i], NULL, PrintHello, (void *)i);
      
      if (rc) {
         cout << "Error:unable to create thread," << rc << endl;
         exit(-1);
      }
   }
   pthread_exit(NULL);
}

निम्न प्रोग्राम का उपयोग कर संकलन करें- लाइब्रेरी का निम्नानुसार उपयोग करें -

$gcc test.cpp -lpthread

अब, अपने प्रोग्राम को निष्पादित करें जो निम्नलिखित आउटपुट देता है -

main() : creating thread, 0
main() : creating thread, 1
main() : creating thread, 2
main() : creating thread, 3
main() : creating thread, 4
Hello World! Thread ID, 0
Hello World! Thread ID, 1
Hello World! Thread ID, 2
Hello World! Thread ID, 3
Hello World! Thread ID, 4

थ्रेड्स के लिए तर्क पारित करना

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

#include <iostream>
#include <cstdlib>
#include <pthread.h>

using namespace std;

#define NUM_THREADS 5

struct thread_data {
   int  thread_id;
   char *message;
};

void *PrintHello(void *threadarg) {
   struct thread_data *my_data;
   my_data = (struct thread_data *) threadarg;

   cout << "Thread ID : " << my_data->thread_id ;
   cout << " Message : " << my_data->message << endl;

   pthread_exit(NULL);
}

int main () {
   pthread_t threads[NUM_THREADS];
   struct thread_data td[NUM_THREADS];
   int rc;
   int i;

   for( i = 0; i < NUM_THREADS; i++ ) {
      cout <<"main() : creating thread, " << i << endl;
      td[i].thread_id = i;
      td[i].message = "This is message";
      rc = pthread_create(&threads[i], NULL, PrintHello, (void *)&td[i]);
      
      if (rc) {
         cout << "Error:unable to create thread," << rc << endl;
         exit(-1);
      }
   }
   pthread_exit(NULL);
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

main() : creating thread, 0
main() : creating thread, 1
main() : creating thread, 2
main() : creating thread, 3
main() : creating thread, 4
Thread ID : 3 Message : This is message
Thread ID : 2 Message : This is message
Thread ID : 0 Message : This is message
Thread ID : 1 Message : This is message
Thread ID : 4 Message : This is message

ज्वाइनिंग और डिटैक्शन थ्रेड्स

निम्नलिखित दो दिनचर्याएं हैं जिनका उपयोग हम धागे से जुड़ने या अलग करने के लिए कर सकते हैं -

pthread_join (threadid, status) 
pthread_detach (threadid)

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

यह उदाहरण दर्शाता है कि Pthread join दिनचर्या का उपयोग करके थ्रेड पूर्णता की प्रतीक्षा कैसे करें।

#include <iostream>
#include <cstdlib>
#include <pthread.h>
#include <unistd.h>

using namespace std;

#define NUM_THREADS 5

void *wait(void *t) {
   int i;
   long tid;

   tid = (long)t;

   sleep(1);
   cout << "Sleeping in thread " << endl;
   cout << "Thread with id : " << tid << "  ...exiting " << endl;
   pthread_exit(NULL);
}

int main () {
   int rc;
   int i;
   pthread_t threads[NUM_THREADS];
   pthread_attr_t attr;
   void *status;

   // Initialize and set thread joinable
   pthread_attr_init(&attr);
   pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_JOINABLE);

   for( i = 0; i < NUM_THREADS; i++ ) {
      cout << "main() : creating thread, " << i << endl;
      rc = pthread_create(&threads[i], &attr, wait, (void *)i );
      if (rc) {
         cout << "Error:unable to create thread," << rc << endl;
         exit(-1);
      }
   }

   // free attribute and wait for the other threads
   pthread_attr_destroy(&attr);
   for( i = 0; i < NUM_THREADS; i++ ) {
      rc = pthread_join(threads[i], &status);
      if (rc) {
         cout << "Error:unable to join," << rc << endl;
         exit(-1);
      }
      cout << "Main: completed thread id :" << i ;
      cout << "  exiting with status :" << status << endl;
   }

   cout << "Main: program exiting." << endl;
   pthread_exit(NULL);
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

main() : creating thread, 0
main() : creating thread, 1
main() : creating thread, 2
main() : creating thread, 3
main() : creating thread, 4
Sleeping in thread
Thread with id : 0 .... exiting
Sleeping in thread
Thread with id : 1 .... exiting
Sleeping in thread
Thread with id : 2 .... exiting
Sleeping in thread
Thread with id : 3 .... exiting
Sleeping in thread
Thread with id : 4 .... exiting
Main: completed thread id :0  exiting with status :0
Main: completed thread id :1  exiting with status :0
Main: completed thread id :2  exiting with status :0
Main: completed thread id :3  exiting with status :0
Main: completed thread id :4  exiting with status :0
Main: program exiting.