共享内存区的使用

还是以操作系统第三章大作业中的一项作为导入。

大作业2要求.png

共享内存就是允许多个不相关的进程共享访问的一个逻辑内存,是进程间数据共享的一种很有效的方式。所有进程都可以对这片区域进行写入、读取,类似于全局变量吧。

使用共享内存需要一些函数接口来实现,如shmget函数创建共享内存,shmat函数将共享内存连接到进程,shmdt函数将共享内存从进程中分离,shmctl函数可用来删除共享内存段。

有关函数的具体用法可参考Linux进程间通信——使用共享内存
)

共享内存并没有提供同步机制,故我们需要使用信号量等方式来控制进程间的互斥。本例没有使用信号量,而是使用了written作为标志起到互斥量相同的作用。

下面直接丢代码了,这个问题理解起来没什么难度,就是熟悉对共享内存的操作。模型也借用了上面博客的例子,进行了部分修改。详细内容见代码及注释。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <assert.h>

#define TEXT_SIZE 1024
void shmread();
void shmwrite();

struct shared_struct
{
int written;//0表示可写,非0表示可读,类似于读写锁
char text[TEXT_SIZE];//记录读写的文本
};

void shmread()
{
int running = 1;//程序运行的标志,1为运行
void *shm = NULL;//共享内存的原始首地址
struct shared_struct *shared;//指向shm
int shmid;//共享内存标识符
key_t key;//key值

//创建key值
key = ftok("../", 1234);
assert(key!=-1);
//创建共享内存
shmid = shmget(key, sizeof(struct shared_struct), 0666|IPC_CREAT);
assert(shmid!=-1); //值为假,则打印出错信息并退出
//将共享内存连接到当前进程的地址空间
shm = shmat(shmid, 0, 0);
assert(shm!=(void*)-1);

//printf("\nMemory attached at %X\n", (int)shm);
//设置共享内存
shared = (struct shared_struct*)shm;
shared->written = 0;
while(running)//读取共享内存的数据
{
//
if(shared->written != 0)
{
printf("You just wrote: %s", shared->text);
sleep(1);
//读完数据后设置written使共享内存段可写
shared->written = 0;
//输入end退出程序
if(strncmp(shared->text, "end", 3) == 0)
running = 0;
}
else//有其他进程在写数据则不能读取数据
sleep(1);
}
//将共享内存从当前进程分离
assert(shmdt(shm)!=-1);
//删除共享内存
assert(shmctl(shmid, IPC_RMID, 0) != -1);

exit(EXIT_SUCCESS);
}

void shmwrite()
{
int running = 1;
void *shm = NULL;
struct shared_struct *shared;
int shmid;
key_t key;

key = ftok("../", 1234);
assert(key!=-1);

char buffer[BUFSIZ + 1];//缓冲区用于保存输入的文本,BUFSIZE=8192在stdlib.h中定义的
//创建
shmid = shmget(key, sizeof(struct shared_struct), 0666|IPC_CREAT);
assert(shmid!=-1);
//连接
shm = shmat(shmid, (void*)0, 0);
assert(shm!=(void*)-1);

//printf("Memory attached at %X\n", (int)shm);
//设置
shared = (struct shared_struct*)shm;
while(running)//写数据
{
//等待数据被读取后再写数据
while(shared->written == 1)
{
sleep(1);
printf("Waiting...\n");
}
//写数据
printf("Enter some text: ");
fgets(buffer, BUFSIZ, stdin);
strncpy(shared->text, buffer, TEXT_SIZE);

//写数据后设置written使共享内存段可读
shared->written = 1;
//输入end退出程序
if(strncmp(buffer, "end", 3) == 0)
running = 0;
}
//分离
assert(shmdt(shm)!=-1);
//sleep(2);
exit(EXIT_SUCCESS);
}

int main()
{
pid_t pid;
pid=fork();
if(pid==0) shmwrite(); //子进程使用共享区并读数据
else shmread(); //父进程创建并使用共享区并写数据
exit(0);
}

运行示例如图。
运行示例.png

其他参考链接也列出
linux 共享内存(实现进程之间通信)
共享内存的实现详解

文章作者: Zepeng Wang
文章链接: http://yoursite.com/2019/04/10/共享内存区的使用/
版权声明: 本博客所有文章除特别声明外,均采用 CC BY-NC-SA 4.0 许可协议。转载请注明来自 王小鹏's Blog