Linux System V IPC 消息隊列

来源:http://www.cnblogs.com/xiaojiang1025/archive/2016/10/07/5935831.html
-Advertisement-
Play Games

模型: 1. 獲取key值 :ftok() 2. 創建/獲取消息隊列 :msgget() 3. 發消息到消息隊列/從消息隊列收信息 :msgsnd()/msgrcv() 3. 刪除消息隊列 :msgctl() 使用的頭文件 ftok() pathname :文件名 proj_id : 1~255的一 ...


模型:

  1. 獲取key值 :ftok()
  2. 創建/獲取消息隊列 :msgget()
  3. 發消息到消息隊列/從消息隊列收信息 :msgsnd()/msgrcv()
  4. 刪除消息隊列 :msgctl()

使用的頭文件

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>

ftok()

//獲取key值, key值是System V IPC的標識符,成功返回key,失敗返回-1設errno
//同pathname+同 proj_id==>同key_t;
key_t ftok(const char *pathname, int proj_id);

pathname :文件名
proj_id: 1~255的一個數,表示project_id

key_t key=ftok(".",100);    //“.”就是一個存在且可訪問的路徑, 100是假設的proj_id
    if(-1==key)
        perror("ftok"),exit(-1);

msgget()

//創建/獲取消息隊列,成功返回shmid,失敗返回-1
int msgget(key_t key, int msgflg);  //ATTENTION:用int msqid=msgget()比較好看

msgflg:具體的操作標誌

  • IPC_CREAT 若不存在則創建, 需要在msgflg中"|許可權信息"; 若存在則打開
  • IPC_EXCL若存在則創建失敗
  • 0 獲取已經存在的消息隊列

消息隊列的容量由msg_qbytes控制,在消息隊列被創建的過程中,這個大小被初始化為MSGMNB,這個限制可以通過msgctl()修改

int msqid=msgget(key,IPC_CREAT|IPC_EXCL|0664);
if(-1==msqid)
    perror("msgget"),exit(-1);

msgsnd()

//向指定的消息隊列發送指定的消息,如果消息隊列已經滿了,預設的行為是堵塞,直到隊列有空間容納新的消息,成功返回0,失敗返回-1設errno
int msgsnd(int msqid, const void *msgp, size_t msgsz, int msgflg);

msqid msgget()返回的消息隊列的ID
msgp消息的的首地址, 消息的參考數據類型如下

struct msgbuf {
    long mtype;       /* message type, must be > 0 */   //消息的類型
    char mtext[1];    /* message data */                //消息的內容
};
ATTENTION:The  mtext field is an array (or other structure) whose size is specified by msgsz, a nonnegative integer value. 

msgsz消息的大小, 該參數用於指定消息內容的大小, 不包括消息的類型。只能sizeof(Msgbuf.mtext),不能sizeof(Msgbuf)
msgflg發送的標誌, 預設給0即可

Msg msg1={1,"hello"};//消息的類型是1,內容是hello
int res=msgsnd(msqid,&msg2,sizeof(msg2.buf),0);
if(-1==res)
    perror("msgsnd"),exit(-1);

msgrcv()

//向指定的消息隊列取出指定的消息,成功返回實際接受到的byte數,失敗返回-1設errno
ssize_t msgrcv(int msqid, void *msgp, size_t msgsz, long msgtyp, int msgflg);

msqid: 消息隊列的ID(returned by msgget)
msgp: 存放接收到消息的緩衝區首地址
msgsz: 消息的最大大小, 不包括消息的類型==>只能sizeof(Msgbuf.mtext),不能sizeof(Msgbuf)

  • 如果消息的長度>msgsz且msgflg里有MSG_NOERROR,則消息會被截斷,被截斷的部分會丟失
  • 如果消息的長度>msgsz且msgflg里沒有MSG_NOERROR,那麼會出錯,報E2BIG。

msgtyp: 消息的類型

  • 0:讀取消息隊列中的第一個消息
  • >0:讀取消息隊列中第一個type為msgtype的消息, 除非msg_flg里有MSG_EXCEPT,此時隊列中的第一個不是msgtyp的消息會被讀取
  • <0讀取消息隊列中type<=|msgtype|的消息, 這裡再優先讀取最小的

msgflg: 發送的標誌, 預設給0即可

Msg msg1;
int res=msgrcv(msqid,&msg1,sizeof(msg1.buf),1,0);
if(-1==res)
    perror("msgrcv"),exit(-1);

msgctl()

// 消息操作,成功返回0,失敗返回-1設errno
int msgctl(int msqid, int cmd, struct msqid_ds *buf);

msqid :消息隊列的ID,由msgget()
buf 結構體指針

struct msqid_ds {
    struct ipc_perm msg_perm;       /* Ownership and permissions */
    time_t          msg_stime;      /*Time of last msgsnd(2) */
    time_t          msg_rtime;      /* Time of last msgrcv(2) */
    time_t          msg_ctime;      /* Time of last change */
    unsigned long   __msg_cbytes;   /* Current number of bytes in queue (nonstandard) */
    msgqnum_t       msg_qnum;       /* Current number of messages in queue */
    msglen_t        msg_qbytes;     /* Maximum number of bytes allowed in queue */
    pid_t           msg_lspid;      /* PID of last msgsnd(2) */
    pid_t           msg_lrpid;      /* PID of last msgrcv(2) */
};
struct ipc_perm {
    key_t           __key;      /* Key supplied to msgget(2) */
    uid_t           uid;        /* Effective UID of owner */
    gid_t           gid;        /* Effective GID of owner */
    uid_t           cuid;       /* Effective UID of creator */
    gid_t           cgid;       /* Effective GID of creator */
    unsigned short  mode;       /* Permissions */
    unsigned short  __seq;      /* Sequence number */
};

cmd

  • IPC_STAT從內核相關結構體中拷貝消息隊列相關的信息到buf指向的結構體中
  • IPC_SET把buf指向的結構體的內容寫入到內核相關的結構體中,同時更顯msg_ctimer成員,同時以下成員也會被更新:msg_qbytes, msg_perm.uid, msg_perm.gid, msg_perm.mode。調用隊列的進程的effective UID必須匹配隊列所有者或創建者的msg_perm.uid或msg_perm.cuid或者該進程擁有特權級別,
  • IPC_RMID立即銷毀消息隊列,喚醒所有正在等待讀取或寫入該消息隊列進程,調用的進程的UID必須匹配隊列所有者或創建者或者該進程擁有足夠的特權級別
  • IPC_INFO (Linux-specific)返回整個系統對與消息隊列的限制信息到buf指向的結構體中

    //_GNU_SOURCE
    //<sys/msg.h>
    struct msginfo {
        int msgpool;/*Size in kibibytes of buffer pool used to hold message data; unused within kernel*/
        int msgmap; /*Maximum number of entries in message map; unused within kernel*/
        int msgmax; /*Maximum number of bytes that can be written in a single message*/
        int msgmnb; /*Maximum number of bytes that can be written to queue; used to initialize msg_qbytes  during queue creation*/
        int msgmni; /*Maximum number of message queues*/
        int msgssz; /*Message segment size; unused within kernel*/
        int msgtql; /*Maximum number of messages on all queues in system; unused within kernel*/
        unsigned short int  msgseg; /*Maximum number of segments; unused within kernel*/
    };
    int res=msgctl(msqid,IPC_RMID,NULL);
    if(-1==res)
    perror("msgctl"),exit(-1);

例子

//Sys V IPC msg
#include<sys/types.h>
#include<sys/ipc.h>
#include<signal.h>
#include<stdio.h>
#include<stdlib.h>
typedef struct{
    long mtype;     //消息的類型
    char buf[20];       //消息的內容
}Msg;
int msqid;          //使用全局變數,這樣就可以在fa中使用msqid了
void fa(int signo){
    printf("deleting..\n");
    sleep(3);
    int res=msgctl(msqid,IPC_RMID,NULL);
    if(-1==res)
        perror("msgctl"),exit(-1);
    exit(0);
}
int main(){
    //ftok()
    key_t key=ftok(".",150);
    if(-1==key)
        perror("ftok"),exit(-1);
    printf("key%#x\n",key);
    //msgget()
    msqid=msgget(key,IPC_CREAT|IPC_EXCL|0664);
    if(-1==msqid)
        perror("msgget"),exit(-1);
    printf("msqid%d\n",msqid);
    //msgsnd()
    Msg msg1={1,"hello"};//消息的類型是1,內容是hello
    Msg msg2={2,"world"};
    int res=msgsnd(msqid,&msg2,sizeof(msg2.buf),0);
    if(-1==res)
        perror("msgsnd"),exit(-1);
    res=msgsnd(msqid,&msg1,sizeof(msg1.buf),0);
    if(-1==res)
        perror("msgsnd"),exit(-1);
    //msgctl()
    //Ctrl+C delete msq
    printf("Press CTRL+C to delete msq\n");
    if(SIG_ERR==signal(SIGINT,fa))
        perror("signal"),exit(-1);
    while(1);
    return 0;
}

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • atitit.http原理與概論attilax總結 1. 圖解HTTP 作者:[日]上野宣 著1 2. HTTP權威指南(國內首本HTTP及其相關核心Web技術權威著作)1 3. TCP/IP詳解(中文版) (共3冊), 這套叢書還有 《TCP/IP詳解 捲2:實現》,《TCP/IP詳解 捲3:TC ...
  • Atitit.軟體架構高擴展性and相容性原理與概論實踐attilax總結 1. 什麼是可擴展的應用程式?1 2. 松耦合(ioc)2 3. 介面的思考 2 4. 單一用途&模塊化,小粒度化2 5. 組合(Composition),而不是繼承(inheritance) 2 6. Ocp原則開閉原則2 ...
  • 首先。題主在試過直接ubuntu終端命令安裝chrome失敗。 把經歷過的錯誤稍微提一下: 在終端輸入 下載安裝包 sudo wget https://dl.google.com/linux/direct/google-chrome-stable_current_amd64.deb 由於在進行解壓安 ...
  • Ubuntu 下安裝QT 本文使用的環境 安裝註意事項 因為安裝需要 所以確保 g++ 已經安裝,如果未安裝。 、 、 這三個包,這點非常重要,所以我們在安裝QT之前需要安裝這三個包: 準備工作完成之後,我們就可以開始安裝了 首先我們先把那兩個安裝包移動到 目錄下 安裝QT分為四步,第一步是QT L ...
  • 一. Nginx 開機啟動1、在/etc/init.d/目錄下創建腳本vim /etc/init.d/nginx2、編寫腳本內容 (將以下複製進去相應改動安裝路徑)#!/bin/bash# nginx Startup script for the Nginx HTTP Server# it is v ...
  • 一、配置主機NFS過程中的一些問題。 1、showmount -e 時出現clnt_create:RPC:Unknown host 解決:設置/etc/hosts,將伺服器IP解析到伺服器主機名。 2、開發板nfs掛載後複製大文件時主機未響應(still trying)問題 解決:mount -t ...
  • 模型 1. 獲取key ftok() 2. 創建/獲取信號量集 semget() 3. 初始化信號量集 semctl() 4. 操作信號量集 semop() 3. 刪除信號量集 semctl() 使用的頭文件: ftok() pathname :文件名 proj_id : 1~255的一個數,表示p ...
  • 其實樹莓派自帶realvnc server 不需要額外tightvnc server 1.點擊左上角樹莓派圖標-->Preferences-->Raspberry Pi Configuration 2.Enable VNC 3.點擊"OK"後,其實VNCserver已經打開了: 4.點擊"More" ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...