进程能创建多少线程与进程逻辑地址空间(32位地址长度的话,就是4GB)(更具体地说应该是有进程地址空间中内存映射区的大小有关,线程栈是通过mmap系统调用在内存映射区创建的。)和线程栈(1MB)的大小有关。
进程创建的过程:
一些操作系统是这样创建进程的:在新的地址空间创建进程,读入可执行文件,最后开始执行。
Unix和Linux采用另一种方式:
它把创建进程分成两步,fork()和exec()
fork()拷贝当前进程一个子进程(init进程是进程树中的根进程),父进程和子进程的区别仅仅在于PID(进程ID)和一些子进程没必要继承的父进程资源。
exec()函数将可执行文件读进地址空间并开始运行。
传统的fork()系统调用直接把所有的资源复制给新创建的进程,但是这种方法过于简单而且效率低下,因为它拷贝的数据可能并不能被共享,更糟糕的情况是:如果新进程打算立即执行一个新的映像,那么所有的拷贝将前功尽弃.
Linux的fork()采用写时拷贝的方式(copy-on-write),内核并不复制整个进程地址空间,而是让父进程和子进程共享一个拷贝。资源的复制只有在需要写的时候才复制,在此之前,只是以只读的方式共享。所以fork()的实际开销就是复制父进程的页表以及给子进程创建一个唯一的进程描述符。