首先,C语言这个语言是没有线程这一概念的。用C语言实现多线程只是用了内核或者操作系统所提供的的接口(内核级线程) C++ C11以后引入多线程,在语言层面有多线程,所以无论是windows操作系统或Linux里面都是可以实现的 先来编写一个多线程中执行fork的进程
include<stdio.h>
#include<unistd.h>
#include<string.h>
#include<pthread.h>
#include<stdlib.h>
void* fun(void*arg)
{
for(int i=0;i<5;i++)
{
printf("fun pid==%d\n",getpid());
sleep(3);
}
}
int main()
{
pthread_t id;
pthread_create(&id,NULL,fun,NULL);
fork();
for(int i=0;i<5;i++)
{
printf("main pid==%d\n",getpid());
sleep(3);
}
pthread_join(id,NULL);
exit(0);
}
以上代码是三个为一组出现的,由此可见,fork复制的整个进程,但如果有多线程,他只会启用一条执行路径,就是fork所在的路径
另外fork也会对锁的状态复制一下,并且复制过后的锁和原来的锁不是同一把锁
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#incl pthread_mutex_lock(&mutex);
ude<sys/wait.h>
#include<pthread.h>
pthread_mutex_t mutex;
void*fun(void*arg)
{
pthread_mutex_lock(&mutex);
printf("fun lock\n");
sleep(5);
pthread_mutex_unlock(&mutex);
printf("fun unlock\n");
}
int main()
{
pthread_mutex_init(&mutex,NULL);
pthread_t id;
pthread_create(&id,NULL,fun,NULL);
sleep(1);
pid_t pid=fork();
if(pid==-1)
{
exit(0);
}
if(pid==0)
{
printf("child will lock\n");
pthread_mutex_lock(&mutex);
printf("child lock\n");
pthread_mutex_unlock(&mutex);
exit(0);
}
wait(NULL);
printf("main end\n");
exit(0);
}
运行结果:直接卡住 分析原因:fork复制的时候复制整个进程,将此事在5s的锁的状态也复制了一下,但是复制之后他就成了另一把锁,即使这个进程再解锁,他也无法解锁,就卡在了子线程上锁的状态,父进程无法wait,就卡住了。
总结: 1.fork会将锁的状态也复制一下,并且这两把锁不是同一把锁
2.子进程中锁的初始值就是父进程中fork那一刻锁的状态
pthread_atfork()在fork之前会加锁,在fork后会解锁,所以会成为解锁状态
|