在Unix中有八种方式可以使终止进程:
- 从主函数中返回
- 调用exit函数
- 调用_exit或_Exit函数
- 最后一个线程从启动例程返回
- 最后一个线程调用pthread_exit函数
- 调用abort函数
- 接收到一个信号并终止
- 最后一个线程中响应取消请求
1~5被称为正常终止,6~8为异常终止。exit函数被调用时将先执行一些清理工作(可由atexit函数指定),而_exit函数和_Exit函数被调用时将直接返回至内核中(_exit函数由ISO C标准定义,_Exit函数由POSIX.1标准定义)。当main函数自然返回(主函数结束或使用return返回)时,进程将自动调用exit函数进行扫尾工作。
atexit函数可以对进行扫尾工作的函数进行登记,登记过的函数在执行exit函数调用时被逆序调用,当同一个函数被登记多次时,函数将被调用多次。atexit函数原型为:
#include <stdlib.h>
int atexit( void (*fun)(void) );
图1为C程序的启动及多种结束流程,其中“exit handler”为由atexit函数登记过的扫尾函数:
图1. C语言进程启动及结束过程流程
在Unix操作系统中,每一个进程都包含一个环境列表(environment list),环境列表以数组的方式存储,数组地址存储在全局变量environ中,使用时需在文件中包含:
extern char **environ;
环境列表中的数据以"name=value"的方式存储,其存储方式如图2所示:
图2. environment lis存储形式
由于历史因素,部分Unix系统在主函数中包含了第三个参数作为环境变量,其函数原型为
int main( int argc, char *argv[], char *envp[] );
在ISO C标准中,main函数只提供两个参数(argc与argv)。在POSIX.1标准中,由全局变量environ取代了参数envp作为环境变量。当需要读取环境列表中数据时,应该使用getenv与putenv函数,其函数原型为
char *getenv( const char *name ); int putenv( char *str );
getenv函数用于从环境列表中读取名称为name的环境变量的值(value属性),putenv函数用于将指定的"name=value"作为环境变量添加进环境变量列表中。
在C程序中,可以使用setjmp和longjmp函数实现类似于C++、JAVA中异常处理的功能。其函数原型如下
#include <setjmp.h> int setjmp( jmp_buf env ); void longjmp( jmp_buf env, int val );
使用时先由setjmp函数指定一个jmp_buf类型缓存,保存寄存器中的数据,当通过longjmp函数返回至setjmp函数调用处时,寄存器中的值被恢复为setjmp函数第一次调用时的值,而存储器中的值保持不变。由于优化编译会对变量存储位置造成影响(存储在存储器中或寄存器中),因此优化编译可能导致setjmp函数通过longjmp返回时寄存器所恢复的值发生变化。记住,setjmp函数调用一次(记录寄存器中的数据)而返回多次(第一次通过setjmp函数返回0,其他时间通过longjmp函数返回,返回值为longjmp函数第二参数);longjmp函数不会返回(返回到了setjmp函数的调用处)。
当需要对进程中资源限制做出修改时,可以使用getrlimit函数与setrlimit函数,其原型为
#include <sys/resource.h>
int getrlimit( int resource, struct rlimit *rlptr ); int setrlimit( int resource, const struct rlimit *rlptr );
结构体rlimit格式为
struct rlimit { rlim_t rlim_cur; //soft limit: current limit rlim_t rlim_max; //hard limit: maximum value for rlim_cur };
当对资源限制做出修改时,需要遵守三条规则:
- 进程中的soft limit值必须小于或等于hard limit值;
- 进程中的hard limit值最小要与soft limit相等,这种修改对普通用户来说是不可逆转的;
- 只有超级用户(superuser)可以提高进程的hard limit值