线程和fork

一、简介

    当线程调用fork时,就为子进程创建了整个进程地址空间的副本,父子进程通过写时复制技术来共享内存页的这一副本。

    子进程通过几成整个地址空间的副本,也从父进程那里继承了所有互斥量、读写锁和条件变量的状态。如果父进程包含多个线程,子进程在fork返回后,如果紧接着不是马上调用exec的话,就需要清理锁状态。

    在子进程内部只存在一个线程,它是由父进程中调用fork的线程的副本构成的。如果父进程中线程占用锁,子进程同样占用这些锁。问题就是子进程并不包含占用锁的线程的副本,所以子进程没办法知道它占用了哪些锁并需要释放哪些锁。

   1、在子进程从fork返回后立马调用exec函数,可以避免这个问题。这种情况下,老的地址空间被丢弃,所以锁的状态无关紧要了。但如果子进程需要继续做处理工作的话,这种方法就行不通了,所以还需要其他策略。

   2、另一种方法就是通过调用pthread_atfork函数建立fork处理程序。其原型如下:

#include 

int pthread_atfork(void (*prepare)(void), void (*parent)(void), void (*child)(void));
    这一函数的作用是为fork安装三个帮助清理锁的函数。其中:

    prepare函数由父进程在fork创建子进程之前调用,这个fork处理程序的任务是获取父进程定义的所有锁;

    parent函数在fork创建子进程后,但在fork返回之前在父进程环境中调用的,其任务是对prepare获取的所有锁进行解锁;

    child函数是在fork返回前在子进程环境中调用的,和parent函数一样,child函数也必须释放prepare处理函数中的所有的锁。

    重点来了,看似这里会出现加锁一次,解锁两次的情况。其实不然,因为fork后对锁进行操作时,子进程和父进程通过写时复制已经不对相关的地址空间进行共享了,所以,此时对于父进程,其释放原有自己在prepare中获取的锁,而子进程则释放从父进程处继承来的相关锁。两个并不冲突。

    下面是一段代码,用来重现此现象:

#include "apue.h"
#include 

pthread_mutex_t lock1 = PTHREAD_MUTEX_INITIALIZER;
pthread_mutex_t lock2 = PTHREAD_MUTEX_INITIALIZER;

void
prepare(void)
{
	printf("preparing locks...\n");
	pthread_mutex_lock(&lock1);
	pthread_mutex_lock(&lock2);
}

void
parent(void)
{
	printf("parent unlocking locks...\n");
	pthread_mutex_unlock(&lock1);
	pthread_mutex_unlock(&lock2);
}

void
child(void)
{
	printf("child unlocking locks...\n");
	pthread_mutex_unlock(&lock1);
	pthread_mutex_unlock(&lock2);
}

void *
thr_fn(void *arg)
{
	printf("thread started...\n");
	pause();
	return(0);
}

int
main(void)
{
	int			err;
	pid_t		pid;
	pthread_t	tid;

#if defined(BSD) || defined(MACOS)
	printf("pthread_atfork is unsupported\n");
#else
	if ((err = pthread_atfork(prepare, parent, child)) != 0)
		err_exit(err, "can't install fork handlers");
	err = pthread_create(&tid, NULL, thr_fn, 0);
	if (err != 0)
		err_exit(err, "can't create thread");
	sleep(2);
	printf("parent about to fork...\n");
	if ((pid = fork()) < 0)
		err_quit("fork failed");
	else if (pid == 0)	/* child */
		printf("child returned from fork\n");
	else		/* parent */
		printf("parent returned from fork\n");
#endif
	exit(0);
}
    运行结果如下(假设子进程先运行):

$ ./a.out
thread started...
parent about to fork...
preparing locks...
child unlocking locks...
child returned from fork
parent unlocking locks...
parent returned from fork
    可以看出,prepare函数在调用fork后运行,child在fork返回到子进程之前运行,parent在fork返回到父进程前运行。

你可能感兴趣的:(杂七杂八的linux)