System V消息隊列

来源:https://www.cnblogs.com/songhe364826110/archive/2019/09/18/11546021.html
-Advertisement-
Play Games

[TOC] 1. 概述 System V消息隊列使用消息隊列標識符標識,和Posix消息隊列一樣,發送消息和接收消息的線程(進程)是相互獨立、互不依賴的。 對於系統中的每個消息隊列,內核維護一個定義在sys/msg.h頭文件中的結構,其中帶註釋的是我們需要關註的成員變數。 2. 消息隊列API ms ...


目錄

1. 概述

System V消息隊列使用消息隊列標識符標識,和Posix消息隊列一樣,發送消息和接收消息的線程(進程)是相互獨立、互不依賴的。
對於系統中的每個消息隊列,內核維護一個定義在sys/msg.h頭文件中的結構,其中帶註釋的是我們需要關註的成員變數。

struct msqid_ds
{
    struct ipc_perm msg_perm;
    struct msg     *msg_first; //指向隊列中第一條消息
    struct msg     *msg_last;  //指向隊列中最後一條消息
    msglen_t       msg_cbytes; //消息隊列當前第幾個位元組
    msgqnum_t      msg_qnum;   //消息隊列當前第幾條消息
    msglen_t       msg_qbytes; //消息隊列允許的最大位元組數,僅針對消息數據,不包括與每個消息關聯的長整型消息類型
    pid_t          msg_lspid;
    pid_t          msg_lrpid;
    time_t         msg_stime;a
    time_t         msg_rtime;
    time_t         msg_ctime;
};

2. 消息隊列API

msgget

msgget用於創建一個新的消息隊列或訪問一個已存在的消息隊列,參數key和oflag的含義及使用方法和semget一樣,不再贅述。

//成功返回消息隊列標識符,失敗返回-1
int msgget(key_t key, int oflag);

msgsnd

msgsnd用於向消息隊列中添加一條消息。

//成功返回0,失敗返回-1
int msgsnd(int msqid, const void *ptr, size_t length, int flag);

參數說明:

  • msqid是msgget返回的標識符
  • ptr是一個由應用程式按如下模板定義的struct sembuf結構指針,只要保證第一個成員變數是long type即可,其後的數據部分可根據需要任意擴展
struct msgbuf
{
    long mtype;       /* message type, must be > 0 */
    char mtext[1];    /* message data */
};
  • length是待發送消息的長度,即消息類型之後的用戶自定義數據的長度,該長度可以是0
  • flag可以是0或者IPC_NOWAIT,一般設為0

msgrcv

msgrcv用於從消息隊列中取出一條消息。

//成功返回實際讀入緩衝區的數據長度,失敗返回-1
int msgrcv(int msqid, void *ptr, size_t length, long type, int flag);

參數說明:

  • msqid是msgget返回的標識符
  • ptr指向數據接收緩衝區,它應該和msgsnd的ptr具有同樣的struct sembuf結構
  • length為緩衝區大小
  • type用於指定希望從消息隊列中讀出什麼樣的消息
  • flag可以是0、IPC_NOWAIT或MSG_NOERROR,一般設為0

msgrcv第四個參數type用於指定希望從消息隊列中讀出什麼樣的消息,假設有消息隊列中有三條消息:

  • 第一條消息的類型為100,長度為1
  • 第二條消息的類型為200,長度為2
  • 第二條消息的類型為300,長度為3

那麼:

  • type = 0:返回隊列中第一個消息
  • type > 0:返回消息類型等於type的第一個消息
  • type < 0:返回消息類型 <= abs(type)的消息中類型值最小的第一個消息

msgctl

msgctl提供在一個消息隊列上的各種控制操作。

//成功返回0,失敗返回-1
int msgctl(int msqid, int cmd, struct msqid_ds *buf);

msgctl支持三個cmd命令:

  • IPC_RMID:從系統中刪除由msqid指定的消息隊列,此時第三個參數被忽略,設為NULL即可
  • IPC_STAT:通過buf參數返回由msqid指定消息隊列對應的msqid_ds結構
  • IPC_SET:通過buf參數返回由msqid指定消息隊列對應的msqid_ds結構,但僅設置以下4個成員:msg_perm.uid、msg_perm.gid、msg_perm.mode和msg_perm.qbytes

3. 簡單的程式

代碼實現

common.h

#ifndef _COMMON_H_
#define _COMMON_H_

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>

#define FTOK_FILE          "/home/delphi/ftok.file"
#define FTOK_ID            1

#define MSG_RD_PERMISSION  0444
#define MSG_WR_PERMISSION  0222
#define MSG_RW_PERMISSION  (MSG_RD_PERMISSION | MSG_WR_PERMISSION)

#define MAX_MSG_LEN        (8192 + sizeof(long))

struct msgbuf
{
    long type;
    char *data;
};

#endif

msgcreate.c

#include "common.h"

int main()
{
    key_t key = ftok(FTOK_FILE, FTOK_ID);
    int oflag = IPC_CREAT | MSG_RW_PERMISSION;
    int msgid = msgget(key, oflag);

    if (msgid >= 0)
    {
        printf("msgget create success, msgid = %d\n", msgid);
    }

    return 0;
}

msgsnd.c

#include "common.h"

int main(int argc, char **argv)
{
    int msgid;
    int msglen;
    long msgtype;
    struct msgbuf *msg;

    msgid   = msgget(ftok(FTOK_FILE, FTOK_ID), MSG_WR_PERMISSION);
    msglen  = atoi(argv[1]);
    msgtype = atol(argv[2]);

    msg = (struct msgbuf *)calloc(sizeof(long) + msglen, sizeof(char));
    msg->type = msgtype;

    msgsnd(msgid, msg, msglen, 0);

    return 0;
}

msgrcv.c

#include "common.h"

int main(int argc, char **argv)
{
    int msgid;
    int rcvlen;
    long msgtype;
    struct msgbuf *msg;

    msgid   = msgget(ftok(FTOK_FILE, FTOK_ID), MSG_RD_PERMISSION);
    msgtype = atol(argv[1]);
    msg = (struct msgbuf *)calloc(MAX_MSG_LEN, 1);

    rcvlen = msgrcv(msgid, msg, MAX_MSG_LEN, msgtype, 0);
    printf("read %d bytes, type = %ld\n", rcvlen, msg->type);

    return 0;
}

msgrmid.c

#include "common.h"

int main(int argc, char **argv)
{
    int msgid = msgget(ftok(FTOK_FILE, FTOK_ID), 0);
    msgctl(msgid, IPC_RMID, 0);

    return 0;
}

代碼測試


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

-Advertisement-
Play Games
更多相關文章
  • 前言 不多BB,直接上圖 Linux命令行的組成結構 Linux系統命令操作語法格式 | 命令 | 空格 | 參數 | 空格 | 【文件或路徑】需要處理的內容 | | | | | | | | rm | | rf | | /tmp/ | | ls | | la | | /home | | 結婚 | | ...
  • [TOC] 由Linux內核提供的基本時間是自1970 01 01 00:00:00 +0000 (UTC)這一特定時間以來經過的秒數,這種描述是以數據類型time_t表示的,我們稱其為日曆時間。 獲得日曆時間的函數有3個:time、clock_gettime和gettimeofday。 time函 ...
  • [TOC] 1. 文件類型 Linux下一切皆文件,但文件也分為7種類型,文件類型信息包含在struct stat結構的st_mode成員中,可以用下表中的巨集來檢測文件類型,這些巨集的參數都是stat.st_mode。 | 文件類型 | 說 明 | 檢測文件類型的巨集 | | | | | 普通文件 | ...
  • [TOC] 1. IO模型 Linux下可用的IO模型有5種,分別是: 阻塞式IO 非阻塞式IO IO復用 信號驅動式IO(SIGIO) 非同步IO(Posix的aio_系列函數) 其中,除了非同步IO,其餘都屬於同步IO模型。 在這5種模型中,我們目前只關註前3種,並且把IO復用放在網路編程專題中講, ...
  • [TOC] 1. 系統調用IO(無緩衝IO) 系統調用 在Linux中一切皆文件,文件操作在Linux中是十分重要的。為此, Linux內核提供了一組用戶進程與內核進行交互的介面用於對文件和設備進行訪問控制,這些介面被稱為系統調用。 系統調用對於應用程式最大的作用在於: 以統一的形式,為應用程式提供 ...
  • [TOC] 1. 管道(無名管道) pipe函數 管道由pipe函數創建,提供一個單向半雙工數據流,它沒有名字,只能在父子進程間使用。 pipe通過參數fd[2]返回兩個文件描述符:fd[0]用於讀,fd[1]用於寫,管道的典型的用法如下: 父進程創建一個管道,然後調用fork創建子進程 父進程關閉 ...
  • 定製ubuntu-server鏡像,預置各種資料庫,工具,單機項目,用戶安裝該鏡像即可直接自動化部署項目。 ...
  • [TOC] 1. 概述 System V共用記憶體在概念上類似於Posix共用記憶體,代之以調用shm_Open後調用mmap的是,先調用shmget,再調用shmat。 對於每個System V共用記憶體,內核都維護如下的信息結構,它定義在sys/shm.h頭文件中,其中帶註釋的是我們需要關註的成員。 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...