Linux中进程描述符
内核把进程放在任务队列(task list)的双向循环链表中,其中链表的每一项都是类型为task_struct,成为进程描述符的结构,该结构定义在<include/linux/sched.h>文件中。
进程状态
运行(TASK_RUNNING)
可中断(TASK_INTERRUPTIBLE)
不可中断(TASK_UNINTERRUPTIBLE)
僵死(TASK_ZOMBIE)
停止(TASK_STOPPED)
Linux 2.6内核中实现了一个O(1)的调度算法,也就是说每一次调度所需要的时间与该CPU内的总进程数无关。
Linux中的线程
从内核角度来说,它并没有线程这个概念。Linux把线程都当作进程来实现,仅仅将其视为使用某些共享资源的进程。每个线程都用有惟一来属于自己的task_struct,所以在内核中,它看起来就像一个普通的进程(只是该进程和其他一些进程共享某些资源,如地址空间等)。
exec函数族
在Linux中使用exec函数族主要有两种情况:
当进程认为自己不能再为系统和用户做出任何贡献时,就可以调用任何exec函数族让自己重生。
如果一个进程想执行另外一个程序,那么它就可以调用fork函数新建一个进程,然后调用一个exec,这样看起来就好像通过执行应用程序而产生了一个新进程。(这种情况非常普遍)。
exec函数族函数原型:
在使用exec函数族时,一定要加上错误判断语句,因为exec很容易执行失败,事实上,这6个函数中真正的系统调用只有execve,其他5个都是库函数,它们最终都会调用execve这个系统调用。
exit和_exit
最大区别就是exit()函数在调用exit系统之前要检查文件的打开情况,将文件缓冲区中的内容写回文件。
函数原型:
wait和waitpid
wait函数是用于使父进程(也就是调用wait的进程)阻塞,直到一个子进程结束或该进程接到了一个指定的信号为止。如果该父进程没有子进程或者其他的子进程已经结束,则wait就会立即返回。
waitpid的作用和wait一样,但它并不一定要等待第一个终止的子进程,它还有若干选项,如可提供一个非阻塞版本的wait功能,也能支持作业控制。实际上wait函数只是waitpid函数的一个特例,在Linux内部实现wait函数时直接调用的就是waitpid函数。
避免僵死进程实例
当一个进程已经终止,但是其父进程尚未对其进行善后处理(获得终止子进程的有关信息,释放它占用的资源)的进程被称为僵死进程。
如果一个进程使用fork函数创建了一个子进程,但不要它等待子进程终止,也不希望子进程处于僵死状态直到父进程终止,实现这一要求的方法就是两次调用fork函数。