互相独立进程间共享内存互斥访问的解决办法

前提:两个进程互相独立,访问同一片共享内存


存在问题:

1、如何避免两个进程同时访问共享内存(一旦两个进程同时访问一个临界区,后果是程序崩溃)

2、如果使用互斥锁,如何让两个进程拿到同一个互斥锁


解决办法:

针对问题1,可以使用信号,信号量,互斥锁来进行同步,但是信号和信号量需要两个进程都实现一套自己的逻辑(访问临界区前,先检查冲突标志,如果没有冲突则访问,并向其它的所有进程依次发送信号,告诉它们我要开始访问了;如果有冲突则阻塞等待。同时进程收到信号后要设置冲突标志,标识现在有其它进程在访问),比较麻烦,两个进程的话还比较简单,多个进程的话在访问共享内存前要通知所有的其它进程。所以最终决定使用互斥锁来完成同步访问。那么就出现了第二个问题,如何让两个进程拿到同一个互斥锁。

针对问题2,可以使用下面的解决方案:互斥量保存在共享内存中,在初始化该锁的时候,设置为进程间共享,这样两个进程连接到共享内存后,都可以获得这个互斥锁,因为已经设置了进程间共享,所以对锁的访问的冲突问题,系统已经解决了。


代码如下,以经过测试:

//processA.c文件
#include <stdlib.h>
#include <stdio.h>
#include <sys/shm.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <errno.h>
#include <pthread.h>

#define BUF_SIZE 4096

int main()
{
    void *shm_addr = NULL;
    char buffer[BUF_SIZE];
	pthread_mutex_t * sharedLock;
	pthread_mutexattr_t ma;

    int shmid;
    // 使用约定的键值创建共享内存
    shmid = shmget((key_t) 1234,  BUF_SIZE, 0666 | IPC_CREAT);
    printf("shmid : %u\n", shmid);
    if (shmid < 0)
    {
        perror("shmget error!");
        exit(1);
    }

    // 将共享内存附加到本进程
    shm_addr = shmat(shmid, NULL, 0);
    if (shm_addr == (void *) -1)
    {
        perror("shmat error!");
        exit(1);
    }

	sharedLock = (pthread_mutex_t *)shm_addr;

	pthread_mutexattr_init(&ma);
	pthread_mutexattr_setpshared(&ma, PTHREAD_PROCESS_SHARED);
	pthread_mutexattr_setrobust(&ma, PTHREAD_MUTEX_ROBUST);
	pthread_mutex_init(sharedLock,&ma);

    // 写入数据
    while(1){
    	pthread_mutex_lock(sharedLock);
    	bzero(buffer, BUF_SIZE);
    	sprintf(buffer, "Hello, My PID is %u\n", (unsigned int) getpid());
    	printf("send data: %s\n", buffer);
    	memcpy(((pthread_mutex_t *)shm_addr)+1, buffer, strlen(buffer));
		pthread_mutex_unlock(sharedLock);
    }

    sleep(5);

    // 分离
    if (shmdt(shm_addr) == -1)
    {
        printf("shmdt error!\n");
        exit(1);
    }
}

//processB.c文件
#include <stdio.h>
#include <stdlib.h>
#include <sys/shm.h>
#include <sys/ipc.h>
#include <sys/types.h>
#include <unistd.h>
#include <string.h>
#include <errno.h>
#include <pthread.h>

#define BUF_SIZE 4096

int main()
{
    void *shm_addr = NULL;
	pthread_mutex_t * sharedLock;
	char tmp[BUF_SIZE];

    int shmid;
    // 使用约定的键值打开共享内存
    shmid = shmget((key_t) 1234, BUF_SIZE, IPC_CREAT);
    printf("shmid : %u\n", shmid);
    if (shmid == -1)
    {
        perror("shmget error!");
        exit(1);
    }

    // 将共享内存附加到本进程
    shm_addr = shmat(shmid, NULL, 0);
    if (shm_addr == (void *) -1)
    {
        perror("shmat error!");
        exit(1);
    }

	sharedLock = (pthread_mutex_t *)shm_addr;

    // 读取数据
    while(1){
		pthread_mutex_lock(sharedLock);
    	bzero(tmp, BUF_SIZE);
    	memcpy(tmp, ((pthread_mutex_t *)shm_addr)+1, 50);
    	printf("read from shared memory: %s\n", tmp);
		pthread_mutex_unlock(sharedLock);
    }

    sleep(5);

    // 分离
    if (shmdt(shm_addr) == -1)
    {
        printf("shmdt error!\n");
        exit(1);
    }

    // 删除共享内存
    if (shmctl(shmid, IPC_RMID, 0) == -1)
    {
        printf("shmctl error!\n");
        exit(1);
    }
}




经过测试,两个进程确实实现了互斥访问,且没有出现程序崩溃。


  • 14
    点赞
  • 64
    收藏
    觉得还不错? 一键收藏
  • 9
    评论
共享内存互斥量都是进程通信的常用方式。下面是一个简单的示例程序,展示了两个非父子进程如何使用共享内存互斥量进行通信。 ```c #include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <sys/ipc.h> #include <sys/shm.h> #include <sys/types.h> #include <sys/sem.h> #define SHM_SIZE 1024 int main() { int shmid, semid, pid; char *shmaddr; struct sembuf sem_b; // 创建共享内存 shmid = shmget(IPC_PRIVATE, SHM_SIZE, IPC_CREAT|0666); if(shmid == -1) { perror("shmget"); exit(EXIT_FAILURE); } // 创建互斥量 semid = semget(IPC_PRIVATE, 1, IPC_CREAT|0666); if(semid == -1) { perror("semget"); exit(EXIT_FAILURE); } // 初始化互斥量 semctl(semid, 0, SETVAL, 1); // 创建子进程 pid = fork(); if(pid == -1) { perror("fork"); exit(EXIT_FAILURE); } else if(pid == 0) // 子进程 { // 连接共享内存 shmaddr = (char *)shmat(shmid, NULL, 0); if(shmaddr == (void *)-1) { perror("shmat"); exit(EXIT_FAILURE); } // 等待互斥量 sem_b.sem_num = 0; sem_b.sem_op = -1; sem_b.sem_flg = SEM_UNDO; semop(semid, &sem_b, 1); // 向共享内存写入数据 sprintf(shmaddr, "Hello, world!"); // 释放互斥量 sem_b.sem_op = 1; semop(semid, &sem_b, 1); // 断开共享内存连接 shmdt(shmaddr); exit(EXIT_SUCCESS); } else // 父进程 { // 连接共享内存 shmaddr = (char *)shmat(shmid, NULL, 0); if(shmaddr == (void *)-1) { perror("shmat"); exit(EXIT_FAILURE); } // 等待互斥量 sem_b.sem_num = 0; sem_b.sem_op = -1; sem_b.sem_flg = SEM_UNDO; semop(semid, &sem_b, 1); // 从共享内存读取数据 printf("Received message: %s\n", shmaddr); // 释放互斥量 sem_b.sem_op = 1; semop(semid, &sem_b, 1); // 断开共享内存连接 shmdt(shmaddr); // 删除共享内存互斥量 shmctl(shmid, IPC_RMID, NULL); semctl(semid, 0, IPC_RMID); exit(EXIT_SUCCESS); } return 0; } ``` 在这个示例程序中,父进程和子进程都连接到了同一个共享内存区域,使用互斥量来实现对共享内存互斥访问。首先创建共享内存互斥量,然后创建子进程。子进程先连接共享内存,然后等待互斥量,之后向共享内存写入数据,最后释放互斥量。父进程先连接共享内存,然后等待互斥量,之后从共享内存读取数据,最后释放互斥量。父进程和子进程断开共享内存连接后,再删除共享内存互斥量。
评论 9
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值