【Linux系統】systemV共享內存

system V共享內存

在Linux系統中,共享內存是一種高效的進程間通信(IPC)機制,它允許兩個或者多個進程共享同一塊物理內存區域,這些進程可以將這塊區域映射到自己的虛擬地址空間中。

共享內存區是最快的IPC形式。一旦這樣的內存映射到共享它的進程的地址空間,這些進程間數據傳遞不再涉及到內核,換句話說進程不再通過執行進入內核的系統調用來傳遞彼此的數據。

1.共享內存示意圖

在這里插入圖片描述

2.共享內存數據結構

struct shmid_ds {struct ipc_perm       shm_perm;      /* operation perms */int                   shm_segsz;     /* size of segment (bytes) */__kernel_time_t       shm_atime;     /* last attach time */__kernel_time_t       shm_dtime;     /* last detach time */__kernel_time_t       shm_ctime;     /* last change time */__kernel_ipc_pid_t    shm_cpid;      /* pid of creator */__kernel_ipc_pid_t    shm_lpid;      /* pid of last operator */unsigned short        shm_nattch;    /* no. of current attaches */unsigned short        shm_nattch;    /* no. of current attaches */unsigned short        shm_unused;    /* compatibility */void                  shm_unused2;   /* ditto - used by DIPC */void                  shm_unused3;   /* unused */
};

3.共享內存函數

shmget函數

  1. 功能: 用來創建共享內存,在共享內存中起著關鍵的初始作用,負責在系統中分配一塊共享內存區域或者獲取已有的共享內存的表示符
  2. 原型
#include <sys/ipc.h>
#include <sys/shm.h>int shmget(key_t key, size_t size, int shmflg);
  1. 參數
    • key: 共享內存段名字(要用戶傳入),key是一個鍵值,用于唯一標識一塊共享內存段。(這個鍵值可以用ftok函數生成),若key值為IPC_PRIVATE,則創建一個新的私有共享內存段,這個段只能通過子進程繼承的方式被其他進程訪問,通常用于父子進程之間的通信
    • size: 共享內存大小
    • shmflg: 由九個權限標志構成,還包括權限位(用法和創建文件時使用的mode模式標志是一樣的)
      • 取值為IPC_CREAT: 共享內存不存在,創建并返回;共享內存已存在,獲取并返回。
      • 取值為IPC_CREAT | IPC_EXCL: 共享內存不存在,創建并返回;共享內存已存在,出錯返回一個非負整數,即該共享內存段的標識碼;失敗返回-1(只要成功,所創建的共享內存一定是新的)
  2. 返回值: 成功返回一個非負整數,即該共享內存段的標識碼;失敗返回-1

ftok函數

ftok 函數是在Linux系統里用于生成System V IPC(Inter-Process Communication,進程間通信)鍵值的函數。這個函數能夠把一個路徑名與一個項目ID轉換為一個系統V IPC鍵值,該鍵值可用于標識共享內存段、消息隊列和信號量集等。

  1. 函數原型
#include <sys/types.h>
#include <sys/ipc.h>key_t ftok(const char *pathname, int proj_id);
  1. 參數說明
  • pathname:一個存在且可訪問的文件或目錄的路徑名。該文件或目錄必須真實存在,因為 ftok 會使用它的索引節點號(inode number)。
  • proj_id:一個項目ID,它是一個1 - 255之間的非零整數。這個ID會和 pathname 的索引節點號結合起來生成IPC鍵值。
  1. 返回值
  • 若成功,返回一個有效的IPC鍵值。
  • 若失敗,返回 -1,并設置 errno 以指示錯誤類型。
  1. 示例代碼
    以下是一個使用 ftok 函數的簡單示例:
#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>int main() {const char *pathname = "/tmp";int proj_id = 'A';key_t key = ftok(pathname, proj_id);if (key == -1) {perror("ftok");return 1;}printf("Generated key: %d\n", (int)key);return 0;
}

在這個示例中,使用 /tmp 作為 pathname'A' 作為 proj_id 來生成一個IPC鍵值。若生成成功,就會把該鍵值打印出來;若失敗,則會輸出錯誤信息。

  1. 注意事項
  • 要保證 pathname 對應的文件或目錄在使用期間不會被刪除或替換,不然生成的鍵值可能會改變。
  • 不同的 pathnameproj_id 組合應該能生成不同的鍵值,但也不能完全排除生成相同鍵值的可能性。
  • proj_id 的取值范圍是1 - 255,若超出這個范圍,可能會導致未定義行為。

shmat函數

  1. 功能: 將共享內存段連接到進程地址空間
  2. 原型
void *shmat(int shmid, const void *shmaddr, int shmflg);
  1. 參數
    • shmid: 共享內存標識
    • shmaddr: 指定共享內存連接到進程空間的起始地址,通常設置為NULL,讓系統自動選擇一個合適地址自行鏈接
    • shmflg: 用于控制共享內存與進程地址空間之間的聯系方式和訪問權限等
      • SHM_RDONLY:表示一只讀方式連接共享內存段,若不指定該標志,則默認是讀寫方式連接
      • SHM_REMAP:若設置該標志,當shmaddr部位NULL時,系統會將指定地址處已經存在的映射替換新的共享內存映射
  2. 返回值: 成功返回一個指針,指向共享內存所連接的進程地址空間的起始位置;失敗返回-1

說明:

  1. shmaddrNULL,核心自動選擇一個地址
  2. shmaddr不為NULL且shmflg無SHM_RND標記,則以shmaddr為連接地址。
  3. shmaddr不為NULL且shmflg設置了SHM_RND標記,則連接的地址會自動向下調整為SHMLBA(指定共享內存段在進程地址空間中的最低地址邊界)的整數倍。公式: shmaddr - (shmaddr % SHMLBA)
  4. shmflg=SHM_RDONLY,表示連接操作用來只讀共享內存

shmdt函數

  1. 功能: 將共享內存段與當前進程脫離,但不刪除共享內存段
  2. 原型
int shmdt(const void *shmaddr);
  1. 參數
    • shmaddr: 由shmat所返回的指針
  2. 返回值: 成功返回0;失敗返回-1
  3. 注意: 將共享內存段與當前進程脫離不等于刪除共享內存段

shmctl函數

  1. 功能: 用于控制共享內存
  2. 原型
int shmctl(int shmid, int cmd, struct shmid_ds *buf);
  1. 參數
    • shmid: 由shmget返回的共享內存標識碼
    • cmd: 將要采取的動作 (有三個可取值)
    • buf: 指向一個保存著共享內存的模式狀態和訪問權限的數據結構
  2. 返回值: 成功返回0;失敗返回-1
命令說明
IPC_STATshmid_ds結構中的數據設置為共享內存的當前關聯值
IPC_SET在進程有足夠權限的前提下,把共享內存的當前關聯值設置為shmid_ds數據結構中給出的值
IPC_RMID刪除共享內存段

4.共享內存的生命周期

在這里插入圖片描述

測試代碼結構

示例1

server.c

#include <iostream>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <unistd.h>const char* PATH = ".";
const int PROJ_ID = 'a';
const int SHM_SIZE = 1024;int main() {// 生成唯一鍵值key_t key = ftok(PATH, PROJ_ID);if (key == -1) {perror("ftok");return 1;}// 獲取共享內存int shmid = shmget(key, SHM_SIZE, 0666);if (shmid == -1) {perror("shmget");return 1;}// 將共享內存附加到進程地址空間char* shm_addr = static_cast<char*>(shmat(shmid, nullptr, 0));if (shm_addr == reinterpret_cast<char*>(-1)) {perror("shmat");return 1;}// 從共享內存讀取數據std::cout << "Reader: Data read from shared memory: " << shm_addr << std::endl;// 將共享內存從進程地址空間分離if (shmdt(shm_addr) == -1) {perror("shmdt");return 1;}// 刪除共享內存if (shmctl(shmid, IPC_RMID, nullptr) == -1) {perror("shmctl");return 1;}return 0;
}

client.c

#include <iostream>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <unistd.h>
#include <cstring>const char* PATH = ".";
const int PROJ_ID = 'a';
const int SHM_SIZE = 1024;int main() {// 生成唯一鍵值key_t key = ftok(PATH, PROJ_ID);if (key == -1) {perror("ftok");return 1;}// 創建共享內存int shmid = shmget(key, SHM_SIZE, IPC_CREAT | 0666);if (shmid == -1) {perror("shmget");return 1;}// 將共享內存附加到進程地址空間char* shm_addr = static_cast<char*>(shmat(shmid, nullptr, 0));if (shm_addr == reinterpret_cast<char*>(-1)) {perror("shmat");return 1;}// 向共享內存寫入數據const char* message = "Hello, shared memory!";strcpy(shm_addr, message);std::cout << "Writer: Data written to shared memory." << std::endl;// 將共享內存從進程地址空間分離if (shmdt(shm_addr) == -1) {perror("shmdt");return 1;}return 0;
}

注意: 共享內存沒有進行同步與互斥! 共享內存缺乏訪問控制! 會帶來并發問題。

實例2.借助管道實現訪問控制版的共享內存

Comm.hpp

#pragma once
#include <fcntl.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <unistd.h>
#include <cassert>
#include <cstdio>
#include <ctime>
#include <cstring>
#include <iostream>using namespace std;#define Debug 0
#define Notice 1
#define Warning 2
#define Error 3
const std::string msg[] = {"Debug","Notice","Warning","Error"
};
std::ostream &Log(std::string message, int level) {std::cout << " | " << (unsigned)time(nullptr) << " | " << msg[level] << " | " << message;return std::cout;
}
#define PATH_NAME "/home/hyb"
#define PROJ_ID 0x66
#define SHM_SIZE 4096 // 共享內存的大小,最好是頁(PAGE: 4096)的整數倍
#define FIFO_NAME "./fifo"
class Init {
public:Init() {int n = mkfifo(FIFO_NAME, 0666);assert(n == 0);(void)n;Log("create fifo success", Notice) << "\n";}~Init() {unlink(FIFO_NAME);Log("remove fifo success", Notice) << "\n";}
};
#define READ O_RDONLY
#define WRITE O_WRONLY
int OpenFIFO(std::string pathname, int flags) {int fd = open(pathname.c_str(), flags);assert(fd >= 0);return fd;
}
void CloseFifo(int fd) {close(fd);
}
void Wait(int fd) {Log("等待中...", Notice) << "\n";uint32_t temp = 0;ssize_t s = read(fd, &temp, sizeof(uint32_t));assert(s == sizeof(uint32_t));(void)s;
}
void Signal(int fd) {uint32_t temp = 1;ssize_t s = write(fd, &temp, sizeof(uint32_t));assert(s == sizeof(uint32_t));(void)s;Log("喚醒中...", Notice) << "\n";
}
string TransToHex(key_t k) {char buffer[32];snprintf(buffer, sizeof buffer, "0x%x", k);return buffer;
}

ShmServer.cc

#include "Comm.hpp"
Init init;
int main() {// 1. 創建公共的Key值key_t k = ftok(PATH_NAME, PROJ_ID);assert(k != -1);Log("create key done", Debug) << " server key : " << TransToHex(k) << endl;// 2. 創建共享內存 -- 建議要創建一個全新的共享內存 -- 通信的發起者int shmid = shmget(k, SHM_SIZE, IPC_CREAT | IPC_EXCL | 0666);if (shmid == -1) {perror("shmget");exit(1);}Log("create shm done", Debug) << " shmid : " << shmid << endl;// 3. 將指定的共享內存,掛接到自己的地址空間char* shmaddr = (char*)shmat(shmid, nullptr, 0);Log("attach shm done", Debug) << " shmid : " << shmid << endl;// 4. 訪問控制int fd = OpenFIFO(FIFO_NAME, O_RDONLY);while (true) {// 阻塞Wait(fd);// 臨界區printf("%s\n", shmaddr);if (strcmp(shmaddr, "quit") == 0)break;}CloseFifo(fd);// 5. 將指定的共享內存,從自己的地址空間中去關聯int n = shmdt(shmaddr);assert(n != -1);(void)n;Log("detach shm done", Debug) << " shmid : " << shmid << endl;// 6. 刪除共享內存,IPC_RMID, 即便是有進程和當下的shm掛接,依舊刪除共享內存n = shmctl(shmid, IPC_RMID, nullptr);assert(n != -1);(void)n;Log("delete shm done", Debug) << " shmid : " << shmid << endl;return 0;
}

ShmClient.cc

#include "Comm.hpp"
int main() {// 1. 創建公共的Key值key_t k = ftok(PATH_NAME, PROJ_ID);if (k < 0) {Log("create key failed", Error) << " client key : " << TransToHex(k) << endl;exit(1);}Log("create key done", Debug) << " client key : " << TransToHex(k) << endl;// 2. 獲取共享內存int shmid = shmget(k, SHM_SIZE, 0);if (shmid < 0) {Log("create key failed", Error) << " client key : " << TransToHex(k) << endl;exit(2);}Log("create shm success", Error) << " client key : " << TransToHex(k) << endl;// 3. 掛接共享內存char* shmaddr = (char*)shmat(shmid, nullptr, 0);if (shmaddr == (char*)-1) {Log("attach shm failed", Error) << " client key : " << TransToHex(k) << endl;exit(3);}Log("attach shm success", Error) << " client key : " << TransToHex(k) << endl;// 4. 寫int fd = OpenFIFO(FIFO_NAME, O_WRONLY);while (true) {ssize_t s = read(0, shmaddr, SHM_SIZE - 1);if (s > 0) {shmaddr[s - 1] = 0;Signal(fd);if (strcmp(shmaddr, "quit") == 0)break;}}CloseFifo(fd);// 5. 去關聯int n = shmdt(shmaddr);assert(n != -1);Log("detach shm success", Error) << " client key : " << TransToHex(k) << endl;return 0;
}

5.共享內存系統級管理指令

1. ipcs

該指令用于顯示系統中的進程間通信(IPC)資源信息,其中就包含共享內存段。

  • 顯示所有共享內存段
ipcs -m
  • 顯示指定用戶的共享內存段
ipcs -m -u username

2. ipcrm

此指令用于刪除系統中的IPC資源,能夠刪除指定的共享內存段。

  • 通過ID刪除共享內存段
ipcrm -m shmid

其中shmid是共享內存段的ID,可以通過ipcs -m命令獲取。

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

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

相關文章

(七)RestAPI 毛子(Http 緩存/樂觀鎖/Polly/Rate limiting/異步大文件上傳)

文章目錄 項目地址一、Http Cache1.1 服務注冊1.2 Validation with ETag1. 添加ETagMiddleware中間件2. 創建內存ETag存儲器3. 服務注冊4. 測試二、使用ETag實現樂觀鎖2.1 添加樂觀鎖方法2.2 修改Controller2.3 測試樂觀鎖三、Rate Limiting3.1 添加速率控制服務1. 在Program里…

2025.4.26_STM32_SPI

1.SPI簡介 2.硬件電路 所有SPI設備的SCK(時鐘)、MOSI(主機輸出從機輸入)、MISO(主機輸入從機輸出)分別連在一起。SCK線只能被主機控制&#xff0c;和I2C相同。 主機另外引出多條SS控制線&#xff0c;分別接到各從機的SS引腳 (SS不用的時候為高電平&#xff0c;當主機需要選中某…

JAVA:單例模式

單例模式是設計模式之一 設計模式&#xff0c;就像古代打仗&#xff0c;我們都聽過孫子兵法&#xff0c;把計謀概括下來弄成一種模式&#xff0c;形成一種套路。 軟件開發中也有很多場景&#xff0c;多數類似的問題場景&#xff0c;解決方案就形成固定的模式&#xff0c;單例…

腦機接口:重塑人類未來的神經增強革命

引言 人類對大腦的探索從未停止&#xff0c;而腦機接口&#xff08;Brain-Computer Interface, BCI&#xff09;的崛起&#xff0c;正在將科幻電影中的“意念操控”變為現實。 這項技術通過解碼腦電信號&#xff0c;實現人腦與外部設備的直接交互&#xff0c;不僅為醫療康復帶來…

從SOA到微服務:架構演進之路與實踐示例

一、架構演進背景 在軟件開發領域&#xff0c;架構風格隨著業務需求和技術發展不斷演進。從早期的單體架構&#xff0c;到面向服務架構(SOA)&#xff0c;再到如今的微服務架構&#xff0c;每一次變革都是為了解決當時面臨的核心問題。 二、SOA架構解析 2.1 SOA核心概念 SOA&…

可靈AI 2.0上線:重新定義AI創作?好萊塢級特效觸手可及

2025年4月15日&#xff0c;快手正式發布可靈AI 2.0&#xff0c;這款被譽為“讓好萊塢特效師顫抖”的AI工具&#xff0c;以物理引擎級動態生成和電影語言自由操控兩大核心技術&#xff0c;徹底顛覆了內容創作的想象邊界。上線24小時內&#xff0c;全球用戶已用它生成超過100萬條…

Mysql存儲引擎、鎖機制

Mysql存儲引擎 InnoDB?&#xff08;MySQL 5.5 及以后版本中的默認存儲引擎&#xff09; ??事務支持??&#xff1a;支持 ??ACID 事務??&#xff0c;適合需要高可靠性的場景&#xff08;如支付、訂單&#xff09;。 ??鎖機制??&#xff1a;默認使用 ??行級鎖??…

飛蛾撲火算法優化+Transformer四模型回歸打包(內含MFO-Transformer-LSTM及單獨模型)

飛蛾撲火算法優化Transformer四模型回歸打包&#xff08;內含MFO-Transformer-LSTM及單獨模型&#xff09; 目錄 飛蛾撲火算法優化Transformer四模型回歸打包&#xff08;內含MFO-Transformer-LSTM及單獨模型&#xff09;預測效果基本介紹程序設計參考資料 預測效果 基本介紹 …

音視頻開發---視頻編碼基礎

一、視頻編碼的必要性 1. 存儲與傳輸成本高 未經編碼壓縮的原始視頻的數據量極大,例如:一般電影的亮度信號采樣頻率為13.5MHz;色度信號的頻帶通常為亮度信號的一半或更少,為6.75MHz或3.375MHz。以4:2:2的采樣頻率為例,Y信號采用13.5MHz,色度信號U和V采用6.75MHz采樣,…

手動安裝OpenSSL1.1.1

根據報錯信息 Could not build the ssl module! Python requires a OpenSSL 1.1.1 or newer&#xff0c;說明當前系統中的 OpenSSL 版本低于 1.1.1&#xff0c;或者 Python 編譯時未找到符合要求的 OpenSSL 庫。以下是分步解決方案&#xff1a; 1. 檢查當前 OpenSSL 版本 首先…

[原創](現代Delphi 12指南):[macOS 64bit App開發]: 跨平臺開發同樣支持retain()引用計數器處理.

[作者] 常用網名: 豬頭三 出生日期: 1981.XX.XX 企鵝交流: 643439947 個人網站: 80x86匯編小站 編程生涯: 2001年~至今[共24年] 職業生涯: 22年 開發語言: C/C++、80x86ASM、Object Pascal、Objective-C、C#、R、Python、PHP、Perl、 開發工具: Visual Studio、Delphi、XCode、…

從 “制造” 到 “品牌”:官網建設助力中國企業突圍東南亞

在全球產業鏈重構與區域經濟一體化的浪潮下&#xff0c;中國企業出海已從“戰略選項”升級為“生存剛需”。東南亞市場因其地理鄰近性、人口紅利及政策開放性&#xff0c;成為許多中企出海的“首站試驗田”。然而&#xff0c;如何在這一文化多元、競爭激烈的市場中建立品牌認知…

iPhone鬧鐘無法識別調休致用戶遲到,蘋果客服稱會記錄反饋

iPhone鬧鐘無法識別調休致用戶遲到&#xff0c;蘋果客服稱會記錄反饋 基于 6 個來源 因“五一”勞動節調休&#xff0c;4月27日&#xff08;周日&#xff09;本應上班&#xff0c;不少iPhone用戶卻因鬧鐘未響遲到&#xff0c;“調休”“當蘋果鬧鐘遇到調休”話題登上熱搜。蘋…

Ubuntu 磁盤空間占用清理(寶塔)

目錄 前言1. 基本知識2. 實戰 前言 &#x1f91f; 找工作&#xff0c;來萬碼優才&#xff1a;&#x1f449; #小程序://萬碼優才/r6rqmzDaXpYkJZF 爬蟲神器&#xff0c;無代碼爬取&#xff0c;就來&#xff1a;bright.cn 本身自搭建了一個寶塔&#xff0c;突然一下子多了好些空…

杰理-安卓通過map獲取時間的時候,部分手機切換sbc和aac時候單耳無聲音

杰理-安卓通過map獲取時間的時候&#xff0c;部分手機切換sbc和aac時候單耳無聲音 #if USER_SUPPORT_PROFILE_MAPif(tws_api_get_role()0){ //主機才獲取&#xff0c;否則切換sbc 和 aac 的時候影響單耳無聲音user_send_cmd_prepare(USER_CTRL_MAP_READ_TIME,0,NULL);} #endif…

Android 進階開發:深入掌握 ProgressBar 的使用與高級技巧

一、前言 在 Android 開發中,ProgressBar 是一個非常常見且功能強大的控件,用于向用戶反饋操作的進度。雖然它的基礎用法簡單,但對于進階開發者來說,如何通過自定義動畫、插值器、樣式和邏輯控制來提升用戶體驗,是一個值得深入研究的方向。 本文將帶你從 基本使用入手,…

AGILE:開啟LLM Agent強化學習的創新框架

在大語言模型&#xff08;LLMs&#xff09;蓬勃發展的今天&#xff0c;基于LLMs構建的智能體成為研究熱點。但如何將各組件整合優化仍是難題。本文提出的AGILE框架給出了創新解法&#xff0c;它不僅統一多組件&#xff0c;還讓智能體性能超越GPT-4。想知道它是如何做到的嗎&…

java使用websocket推送消息到頁面

文章目錄 一、項目背景二、使用方式1.vue2javaspringpom.xmlRealtimeMonitor.vueMonitorTaskExe.javaWSTopicEnum.javaWServerHelper.java 2.vue3javaspringbootpom.xmlTopologyView.vueAlarmDataInquiryController.javaPushService.javaPushWebSocketHandler.javaWebSocketCon…

小市值策略復現(A股選股框架回測系統)

相關config配置 https://quantkt.com/forumDetail?id201043 很早就知道了小市值模型&#xff0c;正好量化選股回測框架出來了&#xff0c;把最裸的小市值復現下&#xff0c;順便驗證下框架邏輯。 科普: 小市值策略基于 “小市值效應”&#xff0c;即從歷史數據來看&#xf…

解決 Flutter 在 iOS 真機上構建失敗的問題

在開發 Flutter 應用時&#xff0c;有時會在嘗試將應用部署到 iOS 真機時遇到構建失敗的問題。錯誤信息通常類似于以下內容&#xff1a; Could not build the precompiled application for the device. Uncategorized (Xcode): Timed out waiting for all destinations matchi…