今天我们来讲讲进程间使用共享内存通信时为了确保数据的正确,如何进行同步?
在Linux中,进程间的共享内存通信需要通过同步机制来保证数据的正确性和一致性,常用的同步机制包括信号量
、互斥锁
、条件变量
等。
其中,使用信号量来同步进程间的共享内存访问是一种常见的方法。每个共享内存区域可以关联一个或多个信号量,以保护共享内存区域的读写操作。在访问共享内存之前,进程需要获取信号量的使用权,当完成读写操作后,再释放信号量的使用权,以便其他进程可以访问共享内存区域。
1、信号量同步
下面是一个简单的示例程序,展示了如何使用信号量来同步共享内存区域的读写操作:
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/shm.h>
#include <sys/sem.h>
#include <string.h>
#define SHM_SIZE 1024
#define SEM_KEY 0x123456
// 定义联合体,用于信号量操作
union semun {
int val;
struct semid_ds *buf;
unsigned short *array;
};
int main() {
int shmid, semid;
char *shmaddr;
struct sembuf semops[2];
union semun semarg;
// 创建共享内存区域
shmid = shmget(IPC_PRIVATE, SHM_SIZE, IPC_CREAT | 0666);
if (shmid == -1) {
perror("shmget");
exit(1);
}
// 将共享内存区域附加到进程地址空间中
shmaddr = shmat(shmid, NULL, 0);
if (shmaddr == (char *) -1) {
perror("shmat");
exit(1);
}
// 创建信号量
semid = semget(SEM_KEY, 1, IPC_CREAT | 0666);
if (semid == -1) {
perror("semget");
exit(1);
}
// 初始化信号量值为1
semarg.val = 1;
if (semctl(semid, 0, SETVAL, semarg) == -1) {
perror("semctl");
exit(1);
}
// 等待信号量
semops[0].sem_num = 0;
semops[0].sem_op = 0;
semops[0].sem_flg = 0;
if (semop(semid, semops, 1) == -1) {
perror("semop");
exit(1);
}
// 在共享内存中写入数据
strncpy(shmaddr, "Hello, world!", SHM_SIZE);
// 释放信号量
semops[0].sem_num = 0;
semops[0].sem_op = 1;
semops[0].sem_flg = 0;
if (semop(semid, semops, 1) == -1) {
perror("semop");
exit(1);
}
// 等待信号量
semops[0].sem_num = 0;
semops[0].sem_op = 0;
semops[0].sem_flg = 0;
if (semop(semid, semops, 1) == -1) {
perror("semop");
exit(1);
}
// 从共享内存中读取数据
printf("Received message: %s\n", shmaddr);