一、多进程
1、多进程创建函数fork(),
创建子进程,复制进程的运行环境。函数返回2次,对父进程返回子进程ID,对子进程返回0。子进程和父进程共用fork()之前的资源,例如打开的文件描述符。
2、wait()、waitpid()
wait()函数一般用在父进程中等待回收子进程的资源,而防止僵尸进程的产生,wait()是阻塞式,而waitpid()可以是阻塞或非阻塞,取决于参数。
3、execl()函数
包括execl execlp等函数
实验:
#include<stdio.h>
#include<sys/types.h> /* 定义数据类型,如 ssize_t,off_t 等 */
#include <fcntl.h> /* 定义 open,creat 等函数原型,创建文件权限的符号常量 S_IRUSR 等 */
#include <unistd.h> /* 定义 read,write,close,lseek 等函数原型 */
#include <errno.h> /* 与全局变量 errno 相关的定义 */
#include <sys/ioctl.h> /* 定义 ioctl 函数原型 */
int main(int argc, char *argv[]) {
pid_t pid,wait_pid = -1;
int *sub_process_status,count = 0;
pid = fork(); /* 创建进程 */
if (pid == 0) /* 对子进程返回 0 */
{
printf("Here is child, my pid = %d, parent's pid = %d\n", getpid(), getppid()); /* 打印父子进程 PID */
sleep(5);
exit(7);
} else if(pid > 0) { /*对父进程返回子进程 PID */
printf("Here is parent, my pid = %d, child's pid = %d\n", getpid(), pid);
wait_pid = wait(NULL);
printf("child exit\n");
printf("child return status value is %d\n",*sub_process_status);
printf("wait_pid is %d\n",wait_pid);
} else { /* fork 出错 */
perror("fork error\n");
}
return 0;
}
输出:
Here is child, my pid = 212, parent's pid = 211
Here is parent, my pid = 211, child's pid = 212
child exit
child return status value is -1991643855
wait_pid is 212
二、守护进程
将一个进程变为守护进程需要以下步骤
(1)子进程等待父进程退出
(2)子进程使用setsid创建新的会话期,脱离控制台
(3)调用chdir将当前工作目录设置为/
(4)umask设置为0以取消任何文件权限屏蔽
(5)关闭所有文件描述符
(6)将0、1、2定位到/dev/null
// setsid将当前进程设置为一个新的会话期session,目的就是让当前进程
// 脱离控制台。
pid = setsid();
// 将当前进程工作目录设置为根目录
chdir("/");
// umask设置为0确保将来进程有最大的文件操作权限
umask(0);
// 关闭所有文件描述符
// 先要获取当前系统中所允许打开的最大文件描述符数目
int cnt = sysconf(_SC_OPEN_MAX);
int i = 0;
for (i=0; i<cnt; i++)
{
close(i);
}
三、log调试
使用openlog() syslog() closelog();
查看log方式,log文件在 /var/log/syslog