区别
开销指的是什么
进程切换包括
1. 切换页全局目录以安装一个新的地址空间
2. 切换内核态堆栈和硬件上下文
3. 刷新TLB
4. 系统调度器的代码执行
看看这些步哪一个不大...如果跨CPU的话,以局部性原理保存的cache也没啥用了,io开销更大
为什么多线程编程需要设置锁
从原理层面上讲,线程具有自己的寄存器和栈空间,因此线程读变量是在自己的栈/寄存器里去读,并不是读系统的寄存器,所以多线程编程需要注意变量的共享安全性
僵尸与孤儿
孤儿进程:父进程退出,而它的一个或多个子进程还在运行,那么那些子进程将成为孤儿进程。孤儿进程将被init进程(进程号为1)所收养,并由init进程对它们完成状态收集工作。
僵尸进程:子进程exit()之后,并非马上就消失掉,而是留下一个称为僵尸进程(Zombie)的数据结构,等待父进程处理。如果父进程并没有调用wait或waitpid获取子进程的状态信息,那么子进程的进程描述符仍然保存在系统中。这种进程称之为僵死进程。
僵尸进程危害:如果进程不调用wait / waitpid的话, 那么保留的那段信息就不会释放,其进程号就会一直被占用,但是系统所能使用的进程号是有限的,如果大量的产生僵尸进程,将因为没有可用的进程号而导致系统不能产生新的进程。
已经产生的僵尸进程,解决方法:kill掉父进程,它产生的僵尸进程就变成了孤儿进程,这些孤儿进程会被init进程接管,init进程会wait()这些孤儿进程,释放它们占用的系统进程表中的资源。
避免产生僵尸进程:
1. 子进程退出时向父进程发送SIGCHILD信号,父进程处理SIGCHILD信号。在信号处理函数中调用wait或waitpid进行处理僵尸进程。
2. fork两次,原理是将僵尸进程成为孤儿进程,从而使其的父进程变为init进程,通过init进程可以处理僵尸进程。
关于fork()
fork会拷贝当前进程的内存,并创建一个新的进程。如上图,fork函数会将整个进程的内存镜像拷贝到新的内存地址,包括代码段、数据段、堆栈以及寄存器内容。之后,我们就有了两个拥有完全一样内存的进程。fork系统调用在两个进程中都会返回,在父进程中,fork系统调用会返回子进程的pid。而在新创建的进程中,fork系统调用会返回0。