संदेश कतार
जब हमें पहले से ही साझा मेमोरी है तो हमें संदेश कतारों की आवश्यकता क्यों है? यह कई कारणों से होगा, आइए हम इसे सरलीकरण के लिए कई बिंदुओं में तोड़ने की कोशिश करें -
जैसा कि समझा जाता है, एक बार एक प्रक्रिया द्वारा संदेश प्राप्त होने के बाद यह किसी अन्य प्रक्रिया के लिए उपलब्ध नहीं होगा। जबकि साझा मेमोरी में, डेटा एक्सेस करने के लिए कई प्रक्रियाओं के लिए उपलब्ध है।
यदि हम छोटे संदेश स्वरूपों के साथ संवाद करना चाहते हैं।
साझा मेमोरी डेटा को एक ही समय में कई प्रक्रियाओं को संप्रेषित करते समय सिंक्रनाइज़ेशन के साथ संरक्षित करने की आवश्यकता होती है।
साझा मेमोरी का उपयोग करके लिखने और पढ़ने की आवृत्ति अधिक है, फिर कार्यक्षमता को लागू करना बहुत जटिल होगा। इस तरह के मामलों में उपयोग के संबंध में नहीं।
क्या होगा यदि सभी प्रक्रियाओं को साझा मेमोरी तक पहुंचने की आवश्यकता नहीं है, लेकिन बहुत कम प्रक्रियाओं को केवल इसकी आवश्यकता है, संदेश कतारों के साथ लागू करना बेहतर होगा।
यदि हम अलग-अलग डेटा पैकेट के साथ संवाद करना चाहते हैं, तो प्रक्रिया ए को बी टाइप करने के लिए संदेश टाइप 1, सी को प्रोसेस करने के लिए संदेश टाइप 10 और सी टाइप करने के लिए संदेश टाइप 20 भेजा जा रहा है। इस मामले में, संदेश कतारों के साथ लागू करना सरल है। दिए गए संदेश प्रकार को 1, 10, 20 के रूप में सरल बनाने के लिए, यह या तो 0 या + ve हो सकता है या जैसा कि नीचे चर्चा की गई है।
बेशक, संदेश कतार का क्रम FIFO (फर्स्ट इन फर्स्ट आउट) है। कतार में डाला गया पहला संदेश पुनर्प्राप्त किया जाने वाला पहला संदेश है।
साझा मेमोरी या संदेश कतारों का उपयोग करना आवेदन की आवश्यकता पर निर्भर करता है और इसका प्रभावी ढंग से उपयोग किया जा सकता है।
संदेश कतारों का उपयोग करते हुए संचार निम्नलिखित तरीकों से हो सकता है -
एक प्रक्रिया द्वारा साझा मेमोरी में लिखना और दूसरी मेमोरी द्वारा साझा मेमोरी से पढ़ना। जैसा कि हम जानते हैं, पढ़ना कई प्रक्रियाओं के साथ भी किया जा सकता है।
अलग-अलग डेटा पैकेट के साथ एक प्रक्रिया द्वारा साझा मेमोरी में लिखना और संदेश प्रक्रियाओं के अनुसार कई प्रक्रियाओं द्वारा पढ़ना।
संदेश कतारों पर कुछ जानकारी देखने के बाद, अब सिस्टम कॉल (सिस्टम V) के लिए जाँच करने का समय है जो संदेश कतारों का समर्थन करता है।
संदेश कतारों का उपयोग करते हुए संचार करने के लिए, निम्नलिखित कदम हैं -
Step 1 - एक संदेश कतार बनाएं या पहले से मौजूद संदेश कतार (msgget ()) से कनेक्ट करें
Step 2 - संदेश कतार में लिखें (संदेश) ()
Step 3 - संदेश कतार से पढ़ें (msgrcv ())
Step 4 - संदेश कतार पर संदेश संचालन करें (msgctl ())
अब, ऊपर दिए गए कॉल्स पर वाक्य रचना और कुछ जानकारी की जाँच करते हैं।
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
int msgget(key_t key, int msgflg)
यह सिस्टम कॉल सिस्टम V संदेश कतार बनाता है या आवंटित करता है। निम्नलिखित तर्कों को पारित करने की आवश्यकता है -
पहला तर्क, कुंजी, संदेश कतार को पहचानता है। कुंजी या तो एक मनमाना मूल्य हो सकती है या एक जो लाइब्रेरी फंक्शन ftok () से ली जा सकती है।
दूसरा तर्क, shmflg, आवश्यक संदेश कतार ध्वज / s निर्दिष्ट करता है जैसे IPC_CREAT (यदि संदेश कतार मौजूद नहीं है) या IPC_EXCL (संदेश कतार बनाने के लिए IPC_CREAT के साथ उपयोग किया जाता है और कॉल विफल रहता है, यदि संदेश कतार पहले से मौजूद है)। साथ ही परमिशन पास करने की जरूरत है।
Note - अनुमतियों के विवरण के लिए पहले के खंड देखें।
यह कॉल सफलता और -1 पर विफलता के मामले में एक मान्य संदेश कतार पहचानकर्ता (संदेश कतार के आगे कॉल के लिए उपयोग किया गया) लौटाएगा। विफलता का कारण जानने के लिए, गलत चर या पेरर () फ़ंक्शन के साथ जांचें।
इस कॉल के संबंध में विभिन्न त्रुटियां हैं EACCESS (अनुमति अस्वीकृत), EEXIST (कतार पहले से मौजूद नहीं है), ENOENT (कतार मौजूद नहीं है), ENOMEM (कतार बनाने के लिए पर्याप्त मेमोरी नहीं), आदि।
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
int msgsnd(int msgid, const void *msgp, size_t msgsz, int msgflg)
यह सिस्टम कॉल संदेश कतार (सिस्टम V) में एक संदेश भेजता / अपेंड करता है। निम्नलिखित तर्कों को पारित करने की आवश्यकता है -
पहला तर्क, msgstr "संदेश कतार पहचानता है, संदेश कतार पहचानकर्ता। संदेशदाता की सफलता पर पहचानकर्ता का मान प्राप्त होता है ()
दूसरा तर्क, msgp, मैसेज का पॉइंटर है, जिसे कॉलर को भेजा जाता है, जो निम्न फॉर्म की संरचना में परिभाषित होता है -
struct msgbuf {
long mtype;
char mtext[1];
};
चर mtype का उपयोग विभिन्न संदेश प्रकारों के साथ संचार करने के लिए किया जाता है, जिसे msgrcv () कॉल में विस्तार से बताया गया है। चर mtext एक सरणी या अन्य संरचना है जिसका आकार msgsz (सकारात्मक मूल्य) द्वारा निर्दिष्ट किया गया है। यदि mtext फ़ील्ड का उल्लेख नहीं किया जाता है, तो इसे शून्य आकार संदेश माना जाता है, जिसकी अनुमति है।
तीसरा तर्क, संदेश, संदेश का आकार है (संदेश एक अशक्त चरित्र के साथ समाप्त होना चाहिए)
चौथा तर्क, मेसफ्लग, IPC_NOWAIT जैसे कुछ झंडों को इंगित करता है (जब कुँजी या MSG_NOERROR में कोई संदेश नहीं मिलता है, तो संदेश पाठ काटता है, यदि msgsz बाइट्स से अधिक है)
यह कॉल सफलता पर 0 और असफलता के मामले में -1 होगी। विफलता का कारण जानने के लिए, गलत चर या पेरर () फ़ंक्शन के साथ जांचें।
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
int msgrcv(int msgid, const void *msgp, size_t msgsz, long msgtype, int msgflg)
यह सिस्टम कॉल संदेश कतार (सिस्टम V) से संदेश को पुनर्प्राप्त करता है। निम्नलिखित तर्कों को पारित करने की आवश्यकता है -
पहला तर्क,।, संदेश कतार पहचानता है, यानी संदेश कतार पहचानकर्ता। संदेशदाता की सफलता पर पहचानकर्ता का मान प्राप्त होता है ()
दूसरा तर्क, संदेश, कॉल करने वाले से प्राप्त संदेश का सूचक है। इसे निम्नलिखित रूप की संरचना में परिभाषित किया गया है -
struct msgbuf {
long mtype;
char mtext[1];
};
चर mtype का उपयोग विभिन्न संदेश प्रकारों के साथ संचार करने के लिए किया जाता है। चर mtext एक सरणी या अन्य संरचना है जिसका आकार msgsz (सकारात्मक मूल्य) द्वारा निर्दिष्ट किया गया है। यदि mtext फ़ील्ड का उल्लेख नहीं किया जाता है, तो इसे शून्य आकार संदेश माना जाता है, जिसकी अनुमति है।
तीसरा तर्क, संदेश, प्राप्त संदेश का आकार है (संदेश एक अशक्त चरित्र के साथ समाप्त होना चाहिए)
मुँह तर्क, संदेश, संदेश के प्रकार को इंगित करता है -
If msgtype is 0 - कतार में पहला प्राप्त संदेश पढ़ता है
If msgtype is +ve - msgtype की कतार में पहला संदेश पढ़ता है (यदि msgtype 10 है, तो केवल टाइप 10 का पहला संदेश पढ़ता है, भले ही शुरुआत में अन्य प्रकार कतार में हों)
If msgtype is –ve - संदेश प्रकार के पूर्ण मान से कम या उसके बराबर के पहले संदेश को पढ़ता है (जैसे, यदि संदेश -५ है, तो यह ५ से कम प्रकार का पहला संदेश पढ़ता है, संदेश का प्रकार १ से ५ तक)
पांचवा तर्क, मेसफ्लग, IPC_NOWAIT जैसे कुछ झंडों को इंगित करता है (जब कुँजी या MSG_NOERROR में कोई संदेश नहीं मिलता है तो तुरंत संदेश भेज देता है (यदि संदेश संदेश बट्स से अधिक हो तो)
यह कॉल वास्तव में सफलता पर mtext सरणी में प्राप्त बाइट्स की संख्या और -1 को विफलता के मामले में लौटाएगा। विफलता का कारण जानने के लिए, गलत चर या पेरर () फ़ंक्शन के साथ जांचें।
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
int msgctl(int msgid, int cmd, struct msqid_ds *buf)
यह सिस्टम कॉल संदेश कतार (सिस्टम V) का नियंत्रण संचालन करता है। निम्नलिखित तर्कों को पारित करने की आवश्यकता है -
पहला तर्क,।, संदेश कतार पहचानता है, यानी संदेश कतार पहचानकर्ता। संदेशदाता की सफलता पर पहचानकर्ता का मान प्राप्त होता है ()
दूसरा तर्क, cmd, संदेश कतार पर आवश्यक नियंत्रण ऑपरेशन करने के लिए कमांड है। Cmd के लिए मान्य मान हैं -
IPC_STAT- बुफ़ द्वारा बताई गई संरचना को msqid_ds संरचना के प्रत्येक सदस्य के वर्तमान मूल्यों की जानकारी। इस आदेश को संदेश कतार पर पढ़ने की अनुमति की आवश्यकता है।
IPC_SET - यूजर आईडी, मालिक की ग्रुप आईडी, संरचनाएं आदि द्वारा बताई गई अनुमतियां आदि को सेट करता है।
IPC_RMID - संदेश कतार को तुरंत हटा देता है।
IPC_INFO - buf द्वारा बताई गई संरचना में संदेश कतार की सीमाओं और मापदंडों के बारे में जानकारी देता है, जो कि प्रकार संरचना msginfo की है
MSG_INFO - संदेश कतार द्वारा भस्म किए गए सिस्टम संसाधनों के बारे में जानकारी युक्त एक msginfo संरचना लौटाता है।
तीसरा तर्क, bf, संदेश कतार संरचना का एक संकेतक है जिसका नाम संरचना msqid_ds है। इस संरचना के मूल्यों का उपयोग या तो सेट के लिए किया जाएगा या सेमी के अनुसार प्राप्त किया जाएगा।
यह कॉल पारित आदेश के आधार पर मान लौटाएगा। IPC_INFO और MSG_INFO या MSG_STAT की सफलता संदेश कतार के सूचकांक या पहचानकर्ता या अन्य संचालन के लिए 0 और विफलता के मामले में -1 लौटाती है। विफलता का कारण जानने के लिए, गलत चर या पेरर () फ़ंक्शन के साथ जांचें।
संदेश कतारों के संबंध में बुनियादी जानकारी और सिस्टम कॉल को देखने के बाद, अब एक कार्यक्रम के साथ जांच करने का समय है।
कार्यक्रम को देखने से पहले हमें विवरण देखें -
Step 1 - दो प्रक्रियाएँ बनाएँ, एक संदेश कतार (msgq_send.c) में भेजने के लिए है और दूसरा संदेश कतार से संदेश भेजने के लिए है (msgq_recv.c)
Step 2- फॉक () फ़ंक्शन का उपयोग करके कुंजी बनाना। इसके लिए, शुरुआत में msgq.txt फ़ाइल बनाई जाती है ताकि एक अनूठी कुंजी प्राप्त की जा सके।
Step 3 - भेजने की प्रक्रिया निम्न कार्य करती है।
उपयोगकर्ता से स्ट्रिंग इनपुट पढ़ता है
नई लाइन हटाता है, अगर यह मौजूद है
संदेश कतार में भेजता है
इनपुट के अंत तक प्रक्रिया को दोहराता है (CTRL + D)
इनपुट का अंत प्राप्त होने के बाद, प्रक्रिया के अंत को सूचित करने के लिए "अंत" संदेश भेजता है
Step 4 - प्राप्त करने की प्रक्रिया में, निम्नलिखित कार्य करता है।
- कतार से संदेश पढ़ता है
- आउटपुट प्रदर्शित करता है
- यदि प्राप्त संदेश "अंत" है, तो प्रक्रिया को समाप्त करता है और बाहर निकलता है
सरल बनाने के लिए, हम इस नमूने के लिए संदेश प्रकार का उपयोग नहीं कर रहे हैं। साथ ही, एक प्रक्रिया कतार में लिख रही है और दूसरी प्रक्रिया कतार से पढ़ रही है। इसे आवश्यकतानुसार बढ़ाया जा सकता है, आदर्श रूप से एक प्रक्रिया कतार में लिखी जाएगी और कई प्रक्रियाएँ कतार से पढ़ी जाएंगी।
अब, हम प्रक्रिया की जाँच करते हैं (कतार में संदेश भेजना) - फाइल: msgq_send.c
/* Filename: msgq_send.c */
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <errno.h>
#include <string.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#define PERMS 0644
struct my_msgbuf {
long mtype;
char mtext[200];
};
int main(void) {
struct my_msgbuf buf;
int msqid;
int len;
key_t key;
system("touch msgq.txt");
if ((key = ftok("msgq.txt", 'B')) == -1) {
perror("ftok");
exit(1);
}
if ((msqid = msgget(key, PERMS | IPC_CREAT)) == -1) {
perror("msgget");
exit(1);
}
printf("message queue: ready to send messages.\n");
printf("Enter lines of text, ^D to quit:\n");
buf.mtype = 1; /* we don't really care in this case */
while(fgets(buf.mtext, sizeof buf.mtext, stdin) != NULL) {
len = strlen(buf.mtext);
/* remove newline at end, if it exists */
if (buf.mtext[len-1] == '\n') buf.mtext[len-1] = '\0';
if (msgsnd(msqid, &buf, len+1, 0) == -1) /* +1 for '\0' */
perror("msgsnd");
}
strcpy(buf.mtext, "end");
len = strlen(buf.mtext);
if (msgsnd(msqid, &buf, len+1, 0) == -1) /* +1 for '\0' */
perror("msgsnd");
if (msgctl(msqid, IPC_RMID, NULL) == -1) {
perror("msgctl");
exit(1);
}
printf("message queue: done sending messages.\n");
return 0;
}
संकलन और निष्पादन कदम
message queue: ready to send messages.
Enter lines of text, ^D to quit:
this is line 1
this is line 2
message queue: done sending messages.
संदेश प्राप्त करने की प्रक्रिया का कोड निम्नलिखित है (कतार से संदेश प्राप्त करना) - फाइल: msgq_recv.c
/* Filename: msgq_recv.c */
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#define PERMS 0644
struct my_msgbuf {
long mtype;
char mtext[200];
};
int main(void) {
struct my_msgbuf buf;
int msqid;
int toend;
key_t key;
if ((key = ftok("msgq.txt", 'B')) == -1) {
perror("ftok");
exit(1);
}
if ((msqid = msgget(key, PERMS)) == -1) { /* connect to the queue */
perror("msgget");
exit(1);
}
printf("message queue: ready to receive messages.\n");
for(;;) { /* normally receiving never ends but just to make conclusion
/* this program ends wuth string of end */
if (msgrcv(msqid, &buf, sizeof(buf.mtext), 0, 0) == -1) {
perror("msgrcv");
exit(1);
}
printf("recvd: \"%s\"\n", buf.mtext);
toend = strcmp(buf.mtext,"end");
if (toend == 0)
break;
}
printf("message queue: done receiving messages.\n");
system("rm msgq.txt");
return 0;
}
संकलन और निष्पादन कदम
message queue: ready to receive messages.
recvd: "this is line 1"
recvd: "this is line 2"
recvd: "end"
message queue: done receiving messages.