pthread_mutex实现进程间同步

本文介绍了如何利用pthread_mutex_t在进程间实现同步,强调了初始化mutex时需要设置PTHREAD_PROCESS_SHARED属性,并提供了测试代码。文章提到了平台兼容性问题,建议在编译和运行时进行检查,并说明了通过mmap实现共享内存的方式。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

前面实验室的师兄们在讨论pthread这套接口能不能实现进程间同步,自己对这个不是十分地了解,但是对这个比较感兴趣,于是就在实验室师兄的指导下去实现了这个同步机制。

测试的结果就是可以用pthread_mutex_t来实现进程间的同步。
需要注意的是初始化 mutex 时需要指定 [b]PTHREAD_PROCESS_SHARED[/b] 这个属性,代码如下


pthread_mutexattr_init(&attr);
pthread_mutexattr_setpshared(&attr, PTHREAD_PROCESS_SHARED);

pthread_mutex_init(&mutex, &attr);



其中 pthread_mutexattr_setpshared 这个 API 在 open group 的标准中定义 [url]http://pubs.opengroup.org/onlinepubs/9699919799/functions/pthread_mutexattr_getpshared.html[/url]

不过这个 feature 不是所有的平台都支持,compile time 用 _POSIX_THREAD_PROCESS_SHARED 检查,runtime 用 sysconf(_SC_THREAD_PROCESS_SHARED) 检查所在平台是否支持。


另外我是通过在 mmap 调用指定 MAP_SHARED 实现在父子进程间共享内存的,并没有使用 System V 的共享内存 API。

这是示例代码:

#include <pthread.h>
#include <stdio.h>
#include <stdlib.h>

#include <sys/mman.h>
#include <fcntl.h>
#include <unistd.h>

#ifndef _POSIX_THREAD_PROCESS_SHARED
#error "This platform does not support process shared mutex"
#endif

/* Define globally accessible variables and a mutex */

#define NUMTHRDS 4

int* shared_sum = NULL;

pthread_mutex_t *sum_mutex;


int process_fn()
{
int i;
for ( i=0; i < 100000000; i++)
{
pthread_mutex_lock(sum_mutex);
*shared_sum += 1;
pthread_mutex_unlock(sum_mutex);
}
return 0;
}

int main(int argc, char** argv)
{

int i;
pthread_mutex_t *p_map;
int* temp;
int cpid;
p_map=(pthread_mutex_t*)mmap(NULL,sizeof(pthread_mutex_t)*10,PROT_READ|PROT_WRITE,
MAP_SHARED|MAP_ANONYMOUS,-1,0);

sum_mutex = p_map;
shared_sum = (int*) (p_map+3);
*shared_sum = 0;

pthread_mutexattr_t mutex_shared_attr;

/* Set pthread_mutex_attr to process shared */
pthread_mutexattr_init(&mutex_shared_attr);
pthread_mutexattr_setpshared(&mutex_shared_attr, PTHREAD_PROCESS_SHARED);

pthread_mutex_init(sum_mutex, &mutex_shared_attr);

cpid = fork();
if(cpid == 0)
{
process_fn();
return 0;
} else {
process_fn();
waitpid(cpid, NULL, 0);
}

temp = (int*) (p_map+3);
printf("in the end, the shared sum: %d\n",*temp);
return 0;
}



这是进程中有线程的示例代码。

#include <pthread.h>
#include <stdio.h>
#include <stdlib.h>

#include <sys/mman.h>
#include <fcntl.h>
#include <unistd.h>

#ifndef _POSIX_THREAD_PROCESS_SHARED
#error "This platform does not support process shared mutex"
#endif

/* Define globally accessible variables and a mutex */

#define NUMTHRDS 4

int* shared_sum = NULL;

pthread_t callThd[NUMTHRDS];
pthread_mutex_t *sum_mutex;

void *thread_fn(void *arg)
{
int i;
int id = (int)((long)arg);
for(i = 0; i < 1000000; i++)
{
pthread_mutex_lock(sum_mutex);
*shared_sum += 1;
/*printf("Thread #%d: mysum=%d\n",id, *shared_sum);*/
pthread_mutex_unlock(sum_mutex);
}
}

int process_fn()
{

pthread_attr_t attr;


int i;
/* Create threads to perform the dotproduct */
pthread_attr_init(&attr);
pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_JOINABLE);

for(i=0;i<NUMTHRDS;i++)
{
pthread_create(&callThd[i], &attr, thread_fn, (void *)((long)i));
}

/* Wait on the other threads */
for(i=0;i<NUMTHRDS;i++) {
pthread_join(callThd[i], NULL);
}
/* After joining, print out the results and cleanup */
printf("Sum = %d\n", *shared_sum);
return 0;
}

int main(int argc, char** argv)
{

int i;
pthread_mutex_t *p_map;
int cpid;
p_map=(pthread_mutex_t*)mmap(NULL,sizeof(pthread_mutex_t)*10,PROT_READ|PROT_WRITE,
MAP_SHARED|MAP_ANONYMOUS,-1,0);

sum_mutex = p_map;
shared_sum = (int*) (p_map+3);
*shared_sum = 0;

pthread_mutexattr_t mutex_shared_attr;

/* Set pthread_mutex_attr to process shared */
pthread_mutexattr_init(&mutex_shared_attr);
pthread_mutexattr_setpshared(&mutex_shared_attr, PTHREAD_PROCESS_SHARED);

pthread_mutex_init(sum_mutex, &mutex_shared_attr);

cpid = fork();
if(cpid == 0)
{
process_fn();
return 0;
} else {
process_fn();
}

printf("In the end, the shared sum: %d\n",*shared_sum);
return 0;
}
### 回答1: PTHREAD_MUTEX_INITIALIZER是一个宏定义,用于静态初始化互斥锁pthread_mutex_t。在使用互斥锁时,需要先初始化互斥锁,可以使用PTHREAD_MUTEX_INITIALIZER宏定义来初始化互斥锁,也可以使用pthread_mutex_init()函数进行动态初始化。静态初始化互斥锁的语法如下: pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER; 这样就可以定义并初始化一个静态的互斥锁mutex,初始状态为未加锁状态。 ### 回答2: PTHREAD_MUTEX_INITIALIZER是一个宏定义,用于初始化一个互斥锁变量。互斥锁用于控制多个线程对共享资源的访问,以避免竞争条件的发生。 通过使用PTHREAD_MUTEX_INITIALIZER宏定义来初始化互斥锁变量,可以确保在创建互斥锁时,锁的状态为未锁定状态。 该宏定义被用于以下的方式来初始化互斥锁变量:pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER; 通过这种方式初始化的互斥锁变量具有以下特点: 1. 该互斥锁变量的类型为pthread_mutex_t,是线程库中定义的互斥锁数据类型。 2. 互斥锁变量被初始化为未锁定状态,即不属于任何线程为其持有者。 3. 可以使用pthread_mutex_lock函数来锁定该互斥锁变量,以及使用pthread_mutex_unlock函数来释放该互斥锁变量。 需要注意的是,PTHREAD_MUTEX_INITIALIZER宏定义仅适用于静态初始化的互斥锁变量,即在编译时期就已经确定了互斥锁的创建及状态。对于动态创建的互斥锁,需要使用pthread_mutex_init函数来进行初始化。 总之,PTHREAD_MUTEX_INITIALIZER是一个方便的宏定义,用于静态初始化互斥锁变量,使得互斥锁可以正确地使用。 ### 回答3: PTHREAD_MUTEX_INITIALIZER是一个用于初始化互斥锁的宏定义常量。 互斥锁是一种进程间同步的机制,它可以用来保护共享资源,以防止多个线程同时访问造成数据不一致或竞态条件等问题。在使用互斥锁时,需要先对其进行初始化,PTHREAD_MUTEX_INITIALIZER就是用来提供一个互斥锁的初始值的。 使用PTHREAD_MUTEX_INITIALIZER可以在定义互斥锁的同时,为其赋予一个初始值。这个初始值会在创建互斥锁之前被设置到互斥锁的所有字段中,以确保互斥锁在创建后就可以立即使用,而不需要再额外的初始化操作。 使用方法很简单,只需要在定义互斥锁的时候,对互斥锁赋值为PTHREAD_MUTEX_INITIALIZER即可,例如: pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER; 值得注意的是,使用PTHREAD_MUTEX_INITIALIZER可以保证一个静态初始化的互斥锁的正确初始化,但是对于动态创建的互斥锁,还是需要使用pthread_mutex_init函数进行初始化。因此,PTHREAD_MUTEX_INITIALIZER仅适用于定义静态互斥锁的情况。 总之,PTHREAD_MUTEX_INITIALIZER是一个用于静态初始化互斥锁的宏定义常量,可以在定义互斥锁时为其赋予一个初始值,以便互斥锁可以立即使用,避免数据竞争和访问冲突的问题。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值