प्रक्रिया निर्माण और समाप्ति
अब तक हम जानते हैं कि जब भी हम किसी कार्यक्रम को निष्पादित करते हैं तो एक प्रक्रिया बनाई जाती है और निष्पादन के पूरा होने के बाद समाप्त की जाएगी। क्या होगा अगर हमें कार्यक्रम के भीतर एक प्रक्रिया बनाने की आवश्यकता है और इसके लिए एक अलग कार्य शेड्यूल करना चाहता है। क्या इसे हासिल किया जा सकता है? हां, जाहिर तौर पर प्रक्रिया निर्माण के माध्यम से। बेशक, काम पूरा हो जाने के बाद यह अपने आप समाप्त हो जाएगा या आप इसे आवश्यकतानुसार समाप्त कर सकते हैं।
प्रक्रिया निर्माण के माध्यम से प्राप्त किया जाता है fork() system call। नई बनाई गई प्रक्रिया को चाइल्ड प्रोसेस कहा जाता है और इसे शुरू करने वाली प्रक्रिया (या निष्पादन शुरू होने पर प्रक्रिया) को पेरेंट प्रक्रिया कहा जाता है। कांटा () सिस्टम कॉल के बाद, अब हमारे पास दो प्रक्रियाएं हैं - माता-पिता और बच्चे की प्रक्रिया। उन्हें कैसे अंतर करना है? बहुत सरल है, यह उनकी वापसी मूल्यों के माध्यम से है।
बच्चे की प्रक्रिया के निर्माण के बाद, हमें फोर्क () सिस्टम कॉल विवरण देखें।
#include <sys/types.h>
#include <unistd.h>
pid_t fork(void);
बच्चे की प्रक्रिया बनाता है। इस कॉल के बाद, दो प्रक्रियाएं होती हैं, मौजूदा एक को मूल प्रक्रिया कहा जाता है और नव निर्मित को बाल प्रक्रिया कहा जाता है।
कांटा () सिस्टम कॉल रिटर्न तीनों मानों में से एक है -
एक त्रुटि को इंगित करने के लिए नकारात्मक मूल्य, अर्थात, बच्चे की प्रक्रिया बनाने में असफल।
बाल प्रक्रिया के लिए एक शून्य देता है।
मूल प्रक्रिया के लिए सकारात्मक मान देता है। यह मान नव निर्मित चाइल्ड प्रक्रिया की प्रक्रिया आईडी है।
आइए हम एक साधारण कार्यक्रम पर विचार करें।
File name: basicfork.c
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
int main() {
fork();
printf("Called fork() system call\n");
return 0;
}
निष्पादन के चरण
संकलन
gcc basicfork.c -o basicfork
निष्पादन / आउटपुट
Called fork() system call
Called fork() system call
Note- आमतौर पर कांटा () कॉल के बाद, बच्चे की प्रक्रिया और माता-पिता की प्रक्रिया अलग-अलग कार्य करती है। यदि एक ही कार्य को चलाने की आवश्यकता है, तो प्रत्येक कांटा () कॉल के लिए यह 2 पावर n बार चलेगा, जहांn कई बार कांटा () की संख्या होती है।
उपरोक्त मामले में, फोर्क () को एक बार कहा जाता है, इसलिए आउटपुट दो बार मुद्रित होता है (2 शक्ति 1)। यदि फोर्क () को 3 बार कहा जाता है, तो आउटपुट 8 गुना (2 पावर 3) प्रिंट होगा। यदि इसे 5 बार कहा जाता है, तो यह 32 बार और इसी तरह और आगे प्रिंट करता है।
कांटा देखा () बच्चे की प्रक्रिया बनाते हैं, यह माता-पिता और बच्चे की प्रक्रियाओं का विवरण देखने का समय है।
फ़ाइल का नाम: pids_after_fork.c
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
int main() {
pid_t pid, mypid, myppid;
pid = getpid();
printf("Before fork: Process id is %d\n", pid);
pid = fork();
if (pid < 0) {
perror("fork() failure\n");
return 1;
}
// Child process
if (pid == 0) {
printf("This is child process\n");
mypid = getpid();
myppid = getppid();
printf("Process id is %d and PPID is %d\n", mypid, myppid);
} else { // Parent process
sleep(2);
printf("This is parent process\n");
mypid = getpid();
myppid = getppid();
printf("Process id is %d and PPID is %d\n", mypid, myppid);
printf("Newly created process id or child pid is %d\n", pid);
}
return 0;
}
संकलन और निष्पादन कदम
Before fork: Process id is 166629
This is child process
Process id is 166630 and PPID is 166629
Before fork: Process id is 166629
This is parent process
Process id is 166629 and PPID is 166628
Newly created process id or child pid is 166630
एक प्रक्रिया दो तरह से समाप्त हो सकती है -
असामान्य रूप से, कुछ संकेतों के वितरण पर होता है, कहते हैं कि सिग्नल समाप्त करें।
सामान्य रूप से, _exit () सिस्टम कॉल (या _Exit () सिस्टम कॉल) या निकास () लाइब्रेरी फ़ंक्शन का उपयोग करते हुए।
_Exit () और निकास () के बीच का अंतर मुख्य रूप से सफाई गतिविधि है। exit() नियंत्रण वापस कर्नेल में लौटने से पहले कुछ सफाई करता है, जबकि _exit() (या _Exit ()) नियंत्रण को वापस कर्नेल में तुरंत लौटा देगा।
बाहर निकलने के साथ निम्नलिखित उदाहरण कार्यक्रम पर विचार करें ()।
फ़ाइल नाम: atexit_sample.c
#include <stdio.h>
#include <stdlib.h>
void exitfunc() {
printf("Called cleanup function - exitfunc()\n");
return;
}
int main() {
atexit(exitfunc);
printf("Hello, World!\n");
exit (0);
}
संकलन और निष्पादन कदम
Hello, World!
Called cleanup function - exitfunc()
_Exit () के साथ निम्नलिखित उदाहरण कार्यक्रम पर विचार करें।
फ़ाइल नाम: at_exit_sample.c
#include <stdio.h>
#include <unistd.h>
void exitfunc() {
printf("Called cleanup function - exitfunc()\n");
return;
}
int main() {
atexit(exitfunc);
printf("Hello, World!\n");
_exit (0);
}
संकलन और निष्पादन कदम
Hello, World!