Day06(上) Liunx高級系統設計6-消息隊列

概述

消息隊列是消息的鏈表,存放在內存中,由內核維護

特點

1 、消息隊列中的消息是有類型的。
2 、消息隊列中的消息是有格式的。
3 、消息隊列可以實現消息的隨機查詢。消息不一定要以先進先出的次序讀取,編程時可以按消息的類型讀取。
4 、消息隊列允許一個或多個進程向它寫入或者讀取消息。
5 、與無名管道、命名管道一樣,從消息隊列中讀出消息,消息隊列中對應的數據都會被刪除。
6 、每個消息隊列都有消息隊列標識符,消息隊列的標識符在整個系統中是唯一的。
7 、只有內核重啟或人工刪除消息隊列時,該消息隊列才會被刪除。若不人工刪除消息隊列,消息隊列會一直存在于系統中
注意 :
ubuntu 某些版本中消息隊列限制值如下 :
每個消息內容最多為 8K 字節
每個消息隊列容量最多為 16K 字節
系統中消息隊列個數最多為 1609
系統中消息個數最多為 16384

消息隊列·操作命令

ipcs -q 獲取消息隊列信息

ipcrm -q? msgid 刪除消息隊列id為msgid的消息隊列

獲取key值

概述
System V 提供的進程間通訊機制 (IPC 通信機制 ) 需要一個 key
key 值可以是人為指定的,也可以通過 ftok 函數獲得。
ftok 函數
所需頭文件 :
#include <sys/types.h>
#include <sys/ipc.h>
函數 :
key_t ftok(const char *pathname, int proj_id)
參數:
????????pathname:路徑名 , 這個路徑必須是存在的而且可以訪問的
????????????????//~ 家目錄
????????????????///? 根目錄
????????????????//./ 當前目錄
????????proj_id:項目 ID ,非 0 整數 ( 只有低 8 位有效 )
返回值:
????????成功返回 key
????????失敗返回 -1
注意 :
????????當調用該函數時傳入的參數一致獲取到的key 值也將相同
????????多進程通訊時要保證路徑名相同,id 也相同
示例:
#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>
int main(int argc, char const *argv[])
{
key_t key01 = ftok("./",2023);
key_t key02 = ftok("./",2023);
key_t key03 = ftok("./",2022);
printf("key1=%u\n",key01);
printf("key2=%u\n",key02);
printf("key3=%u\n",key03);
return 0;
}

創建和獲取消息隊列-msgget函數

作用:

創建一個新的或打開一個已經存在的消息隊列。不同的進程調用此函數,只要用相同的
key 值就能得到同一個消息隊列的標識符
函數:
所需頭文件
????????#include <sys/msg.h>
函數
????????int msgget(key_t key, int msgflg);
參數:
????????key: IPC 鍵值。
????????msgflg:標識函數的行為及消息隊列的權限。
????????msgflg 的取值:
????????IPC_CREAT:創建消息隊列。
????????IPC_EXCL:檢測消息隊列是否存在。
位或權限位:消息隊列位或權限位后可以設置消息隊列的訪問權限,格式和open 函數的 mode_t 一樣,但可執行權限未使用。
返回值:
????????成功:消息隊列的標識符(msgid) ,失敗:返回 -1
#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>
int main(int argc, char const *argv[])
{key_t key = ftok("./",12);printf("ipc鍵值是:%d\n",key);int msgid = msgget(key,IPC_CREAT |0666);printf("消息隊列id:%d\n",msgid);key_t key2 = ftok("./",13);int msgid2 = msgget(key2,IPC_EXCL);//查看消息隊列是否存在printf("消息隊列id2:%d\n",msgid2);return 0;
}

消息格式

typedef struct _msg
{
????????long mtype ; /* 消息類型 , 必須是第一個成員 , 必須是 long , 就是該消息的 id*/
????????char mtext [ 100 ]; /* 消息正文 , 用戶自定義 */
????????... /* 消息的正文可以有多個成員 */
} MSG ;

發送消息-msgsnd函數

作用 : 將新消息添加到消息隊列。
函數:
所需頭文件 :
????????#include <sys/msg.h>
函數 :
????????int msgsnd(int msqid, const void *msgp,size_t msgsz, int msgflg);
參數 :
????????msqid:消息隊列的標識符。
????????msgp:待發送消息結構體的地址。
????????msgsz:消息正文的字節數。
????????msgflg:函數的控制屬性
????????????????0:msgsnd 調用阻塞直到條件滿足為止。 ( 推薦 )
????????????????IPC_NOWAIT: 若消息沒有立即發送則調用該函數的進程會立即返回。
返回值 :
????????成功:0
????????失敗:-1

接收消息-msgrcv

作用 : 從標識符為 msqid 的消息隊列中接收一個消息。一旦接收消息成功,則消息 在消息隊列中被刪除。
函數
所需頭文件
????????#include <sys/msg.h>
函數 :
????????ssize_t msgrcv(int msqid, void *msgp, size_t msgsz, long msgtyp,int msgflg);
參數 :
????????msqid:消息隊列的標識符,代表要從哪個消息列中獲取消息。
????????msgp:存放消息結構體的地址。
????????msgsz:消息正文的字節數。
????????msgtyp:消息的類型、可以有以下幾種類型
????????msgtyp=0:返回隊列中的第一個消息
????????msgtyp>0:返回隊列中消息類型為 msgtyp 的消息
????????msgtyp<0:返回隊列中消息類型值小于或等于 msgtyp 絕對值的消息 , 如果這種消息有若干個, 則取類型值最小的消息。
注意:
若消息隊列中有多種類型的消息 ,msgrcv 獲取消息的時候按消息類型獲取, 不是先進先出的。
在獲取某類型消息的時候 , 若隊列中有多條此類型的消息 , 則獲取最先添加的消息, 即先進先出原則
msgflg: 函數的控制屬性
????????0:msgrcv調用阻塞直到接收消息成功為止。
????????MSG_NOERROR:若返回的消息字節數比 nbytes 字節數多 , 則消息就會截短到nbytes字節 , 且不通知消息發送進程
IPC_NOWAIT: 調用進程會立即返回。若沒有收到消息則立即返回 -1
返回值 :
????????成功返回讀取消息的長度
????????失敗返回-1

案例1:簡單的消息收發

需要 :
張三發出消息類型為 10 20 的消息
李四接收消息類型為 10 的消息
王五接收消息類型為 20 的消息
代碼 02_zs.c
#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <string.h>
typedef struct struct_msg{
long msgType;
char name[20];
char text[128];
}Msg;
int main(int argc, char const *argv[])
{
//獲取key值
key_t key = ftok("./",258);
//創建或獲取消息對象
int msgid = msgget(key,IPC_CREAT| 0666);
//準備要發送的消息
Msg msg01 = {10,"張三","你好 李四"};
Msg msg02;
msg02.msgType=20;
strcpy(msg02.name,"張三");
strcpy(msg02.text,"你好 王五");
//發送消息給李四
msgsnd(msgid,&msg01,sizeof(msg01) - sizeof(long),0);
//發現消息給王五
msgsnd(msgid,&msg02,sizeof(msg01) - sizeof(long),0);
return 0;
}
代碼 02_ls.c
#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <string.h>
typedef struct struct_msg{
long msgType;
char name[20];
char text[128];
}Msg;
int main(int argc, char const *argv[])
{
key_t key = ftok("./",258);
int msqid = msgget(key,IPC_CREAT|0666);
Msg msg;
msgrcv(msqid,&msg,sizeof(msg)-sizeof(long),10,0);
printf("李四接收到的消息為\n");
printf("姓名:%s說%s\n",msg.name,msg.text);
return 0;
}

代碼03——ww.c

#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <string.h>
typedef struct struct_msg{
long msgType;
char name[20];
char text[128];
}Msg;
int main(int argc, char const *argv[])
{
key_t key = ftok("./",258);
int msqid = msgget(key,IPC_CREAT|0666);
Msg msg;
msgrcv(msqid,&msg,sizeof(msg)-sizeof(long),20,0);
printf("王五接收到的消息為\n");
printf("姓名:%s說%s\n",msg.name,msg.text);
return 0;
}

消息隊列控制(了解)

作用 : 對消息隊列刪除 , 獲取 , 修改等操作
語法 :
所需頭文件 :
????????#include <sys/msg.h>
函數 :
????????int msgctl(int msqid, int cmd, struct msqid_ds *buf);
參數:
????????msqid:消息隊列的標識符。
????????cmd:函數功能的控制。
????????????????IPC_RMID:刪除由 msqid 指示的消息隊列,將它從系統中刪除并破壞相關數據結構。 ????????????????IPC_STAT:將 msqid 相關的數據結構中各個元素的當前值存入到由 buf 指向的結構中。
????????????????IPC_SET:將 msqid 相關的數據結構中的元素設置為由 buf 指向的結構中的對應值。
????????buf:msqid_ds 數據類型的地址,用來存放或更改消息隊列的屬性。
返回值:
????????成功: 返回 0
????????失敗: 返回 -1
struct msqid_ds類型:
struct msqid_ds {
struct ipc_perm msg_perm; /* 所有者與權限 */
time_t msg_stime; /* 最后一條消息發送的時間 */
time_t msg_rtime; /* 最后一條消息接收的時間 */
time_t msg_ctime; /* 最后一次更改的時間 */
unsigned long __msg_cbytes; /* 隊列中的當前字節數(非標準) */
msgqnum_t msg_qnum; /* 隊列中的當前消息數 */
msglen_t msg_qbytes; /* 隊列中允許的最大字節數 */
pid_t msg_lspid; /* 最后一次發送的進程 id */
pid_t msg_lrpid; /* 最后一次接收的進程 id */
};
struct ipc_perm類型:
struct ipc_perm {
key_t __key; /* 提供給 msgget 的密鑰 (2) */
uid_t uid; /* 所有者的有效 UID */
gid_t gid; /* 所有者的有效 GID */
uid_t cuid; /* 創建者的有效 UID */
gid_t cgid; /* 創建者的有效 GID */
unsigned short mode; /* 權限 */
unsigned short __seq; /* 序列號 */
};

案例2:多人聊天程序

#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <unistd.h>
#include <sys/wait.h>
#include <string.h>
typedef struct INFO
{long type;char name[20];char text[128];        
}info;
int main(int argc, char const *argv[])
{printf("請輸入用戶姓名:\n");char name[50] = {0};scanf("%s",name);int stype = 0;printf("請輸入要發送的類型:\n");scanf("%d",&stype);int rtype = 0;printf("請輸入要接受的數據類型:\n");scanf("%d",&rtype);//創建消息隊列key_t key = ftok("./",11) ;int mesid = msgget(key,IPC_CREAT | 0666);printf("mesid=%d\n",mesid);int i = 0;for(i = 0;i < 2;i++){int pid = fork();if(pid == 0){break;}}if(i == 0){while(1){info inf;msgrcv(mesid,&inf,sizeof(info)-sizeof(long),rtype,0);printf("%s:%s\n",inf.name,inf.text);if(strcmp("886",inf.text) == 0){break;}printf("111\n");}_exit(0);}else if(i == 1){while(1){printf("222\n");          info inf;scanf("%s",inf.text);inf.type = stype;strcpy(inf.name,name);msgsnd(mesid,&inf,sizeof(info)-sizeof(long),0);if(strcmp("886",inf.text) == 0){break;}}_exit(0);}else if(i == 2){while(1){int pid = waitpid(-1,NULL,WNOHANG);if(pid == -1){break;}}}return 0;
}

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/214842.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/214842.shtml
英文地址,請注明出處:http://en.pswp.cn/news/214842.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

排程系統中關于任務優先級的需求延伸與設計構思

無論是面向銷售訂單的MPS&#xff0c;還是基于多工序制約關系的APS&#xff0c;還是具體車間生產中針對單一工序的任務作業調度優化&#xff0c;都存在基于被排程對象(例如銷售訂單、生產工單、工序任務)的優先級進行優化的需求場景。當我們僅在宏觀、較高層次的角度考慮&#…

[go 面試] 緩存策略與應對數據庫壓力的良方

關注公眾號【愛發白日夢的后端】分享技術干貨、讀書筆記、開源項目、實戰經驗、高效開發工具等&#xff0c;您的關注將是我的更新動力&#xff01; 在高并發場景中&#xff0c;緩存是提高系統性能的關鍵利器。然而&#xff0c;緩存穿透、緩存擊穿、緩存雪崩等問題可能會給系統帶…

高效掃頻阻垢裝置廣譜感應水處理設備介紹工作原理使用參數和選型

? 1&#xff1a;高效掃頻阻垢裝置設備介紹 高效掃頻阻垢裝置是一種通過控制箱釋放變頻電磁信號&#xff0c;傳輸到信號放大裝置&#xff0c;管道外側的電磁線圈和電錘產生高頻機械振動&#xff0c;在管道和水中傳輸&#xff0c;通過共振機理破壞水分子之間的氫鍵&#xff0c;產…

記錄 | shell腳本開頭#!/bin/bash的作用

在 Shell 腳本中&#xff0c;#!/bin/bash 是指定腳本使用 Bash 解釋器的 shebang 語句。它出現在腳本的第一行&#xff0c;并告訴操作系統使用 Bash 解釋器來執行該腳本 #!/bin/bash....具體作用如下&#xff1a; 指定解釋器&#xff1a;#!/bin/bash 指定了使用 Bash 作為腳本…

Java - Lombok介紹、使用、工作原理、優缺點

介紹 Project Lombok is a java library that automatically plugs into your editor and build tools, spicing up your java.Never write another getter or equals method again, with one annotation your class has a fully featured builder, Automate your logging vari…

oracle修改SYS用戶(系統內置超級賬號)的方法和注意事項

Oracle數據庫中的SYS用戶是最高權限的賬號&#xff0c;擁有對整個數據庫的控制權。因此&#xff0c;在正常情況下&#xff0c;不建議修改SYS用戶。但是有些時候為了解決特定問題&#xff0c;可能需要修改SYS用戶的默認設置。 本文將介紹一些修改SYS用戶的方法和注意事項。 修…

算法Day28 二進制差異序列(格雷碼)

二進制差異序列&#xff08;格雷碼&#xff09; Description n 位二進制差異序列是一個由2^n個整數組成的序列&#xff0c;其中&#xff1a; 每個整數都在范圍[0, 2^n - 1]內&#xff08;含0和2^n - 1&#xff09; 第一個整數是0 一個整數在序列中出現不超過一次 每對相鄰整數…

linux 13-2day 日志輪轉 日志目錄 輪轉參數

目錄 日志系統rsyslog一、處理日志的進程二、常見的日志文件(系統、進程、應用程序)日志優先級 三、logrotate日志輪轉1、配置日志輪轉的路徑2、日志配置路徑四 、案例 日志系統rsyslog Linux 系統內核和許多程序會產生各種錯誤信息、告警信息和其他的提示信息&#xff0c; 這…

【go語言實踐】基礎篇 - 流程控制

if語句 go里面if不需要括號將條件表達式包含起來&#xff0c;這與python也有點類似 if 條件表達式 { } if num > 18 {// ... } else if num > 20 {// ... } else {// ... }需要注意的是go支持在if的條件表達式中直接定義一個變量&#xff0c;變量的作用域只在if范圍內…

【網絡安全】CTF入門教程(非常詳細)從零基礎入門到進階,看這一篇就夠了!

一、CTF簡介 CTF&#xff08;Capture The Flag&#xff09;中文一般譯作奪旗賽&#xff0c;在網絡安全領域中指的是網絡安全技術人員之間進行技術競技的一種比賽形式。CTF起源于1996年DEFCON全球黑客大會&#xff0c;以代替之前黑客們通過互相發起真實攻擊進行技術比拼的方式。…

計算機丟失msvcp140dll怎么恢復?快速解決dll缺失問題

在計算機使用過程中&#xff0c;我們經常會遇到一些錯誤提示&#xff0c;其中之一就是“msvcp140dll丟失”。msvcp140.dll是一個動態鏈接庫文件&#xff0c;它包含了許多C標準庫函數的實現。這些動態鏈接庫文件是程序運行所必需的&#xff0c;它們包含了許多函數和資源&#xf…

圣誕新奇驚喜:利用 AI 技術幫助圣誕老人創建手寫信件

人工智能甚至正在接管北極的任務。在即將到來的圣誕節假期之前&#xff0c;圣誕老人和他的助手們迎來了一項革命性的技術支持。一群樂于助人的精靈采用了人工智能技術&#xff0c;制作出獨一無二、看似親手書寫的信件&#xff0c;以確保遵守圣誕老人的「北極標準」。 這些信件通…

C語言實現選擇排序

完整代碼&#xff1a; #include<stdio.h>//交換函數&#xff0c;交換兩個數 void swap(int *a,int *b){int temp;temp*a;*a*b;*btemp; }//選擇排序&#xff0c;從小到大 //參數&#xff1a;arr[]表示待排序數組&#xff0c;len表示該數組長度 void select_sort(int arr[…

愛智EdgerOS之深入解析安全可靠的開放協議SDDC

一、協議簡介 在 EdgerOS 的智慧生態場景中&#xff0c;許多智能設備或傳感器的生命周期都與 SDDC 協議息息相關&#xff0c;這些設備可能是使用 libsddc 智能配網技術開發的&#xff0c;也有可能是因為主要功能上是使用其他技術如 MQTT、LoRa 等但是設備的上下線依然是使用上…

圖的遍歷(深度優先遍歷 + 廣度優先遍歷)

目錄 &#x1f33c;廣度優先遍歷 &#xff08;1&#xff09;鄰接矩陣BFS &#xff08;2&#xff09;鄰接表BFS &#xff08;3&#xff09;非連通圖BFS &#xff08;4&#xff09;復雜度分析 &#x1f33c;深度優先遍歷 &#xff08;1&#xff09;鄰接矩陣的DFS &#x…

Caching the Application Engine Server 緩存應用程序引擎服務器

Caching the Application Engine Server 緩存應用程序引擎服務器 Application Engine caches metadata just like the application server. This caching enhances performance because a program can refer to the local cache for any objects that it uses. 應用程序引擎…

科技云報道:從數據到生成式AI,是該重新思考風險的時候了

科技云報道原創。 OpenAI“宮斗”大戲即將塵埃落定。 自首席執行官Sam Altman突然被董事會宣布遭解雇、董事長兼總裁Greg Brockman辭職&#xff1b;緊接著OpenAI員工以辭職威脅董事會要求Altman回歸&#xff1b;再到OpenAI董事會更換成員、Altman回歸OpenAI。 表面上看&…

深入解析Java中的String:特點、重要方法及源碼分析

Java的String類是Java語言中最常用的類之一。 作為一位Java高級工程師&#xff0c;了解String類的特性和方法對于編寫高效和優化的Java代碼至關重要。在這篇技術博客中&#xff0c;我們將深入探討String類的特點&#xff0c;介紹其中一些重要的方法&#xff0c;并分析其源碼以獲…

java--LocalDate、LocalTime、LocalDateTime、ZoneId、Instant

1.為什么要學習JDK8新增的時間 LocalDate&#xff1a;代表本地日期(年、月、日、星期) LocalTime&#xff1a;代表本地時間(時、分、秒、納秒) LocalDateTime&#xff1a;代表本地日期、時間(年、月、日、星期、時、分、秒、納秒) 它們獲取對象的方案 2.LocalDate的常用API(…

Android的開機logo生成

生成可用的uboot和kernel的logo圖片 可以通過命令轉換BMP格式的圖片 ### 將 png 轉為顏色深度為8bit的的bmp圖片。jpeg使用jpegtopnm ### pngtopnm logo.png | ppmquant 31 | ppmtobmp -bpp 8 > logo.bmp然后就可以使用新圖替換舊圖片&#xff0c;在kernel目錄下的logo.bmp…