1、当父进程创建子进程的时候,默认子进程与父进程同属一个进程组,可以用kill -SIGKILL -进程组id(负数)来将整个进程组的id全部杀死。
2、组长进程不能创建会话。
3、setsid 函数
4、创建一个会话,并以自己的ID设置进程组ID,同时也是新会话ID。
5、pid_t setsid(void) ;
成功:返回调用进程的会话ID;失败:-1,设置errno
6、调用了setsid函数的进程,既是新的会长,也是新的组长
7、守护进程:
又称daemon进程。通常运行于操作系统后台,脱离控制终端。一般不与用户直接交互。周期性的等待某个时间的发生或周期性执行某一个动作,不受用户登录注销影响。通常采用以d结尾的命名方式。
守护进程创建步骤(重要):
8、创建daemon:守护进程的过程:
1)、fork子进程,让父进程终止(exit(0))
2)、子进程调用setsid()创建新会话(pid=setsid())
3)、通常根据需要,改变工作目录位置 chdir(“目录”)一般设置为根目录,防止被卸载
4)、通常根据需要,重设umask文件权限掩码,影响新文件的创建权限。022-755
5)、通常根据需要,关闭/重定向 文件描述符
6)、守护进程 业务逻辑。while()
#include <stdio.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <unistd.h>
#include <string.h>
#include <stdlib.h>
#include <time.h>
#include <fcntl.h>
#include <sys/time.h>
void work(int num)
{
time_t tm=time(NULL);
struct tm *loc=localtime(&tm);
char buf[1024];
sprintf(buf,"%d-%d-%d %d:%d:%d\n",loc->tm_year,loc->tm_mon,loc->tm_mday,loc->tm_hour,loc->tm_min,loc->tm_yearloc->tm_sec);
printf("%s\n",buf);
}
int main()
{
//1、fork子进程,让父进程终止(exit(0))
pid_t pid=fork();
if(pid>0)
exit(0);
//2、子进程调用setsid()
if(pid==0)
{
setsid();
}
//3、通常根据需要,改变工作目录位置 chdir(“目录”)一般设置为根目录,防止被卸载
chdir("/");
//4、通常根据需要,重设umask文件权限掩码,影响新文件的创建权限。022-755
umask(022);
//5、通常根据需要,关闭/重定向 文件描述符
int fd=open("/dev/null",O_RDWR);
dup2(fd,STDIN_FILENO);//0
dup2(fd,STDOUT_FILENO);//1
dup2(fd,STDERR_FILENO);//2
//6、守护进程 业务逻辑。while()
//(1)捕捉信号
struct sigaction act;
act.sa_flags=0;
act.sa_handler=work;
sigemptyset(&act.sa_mask);
sigaction(SIGALRM,&act,NULL);
//(2)创建定时器
struct itimerval val;
val.it_value.tv_sec=2;
val.it_value.tv_usec=0;
val.it_interval.tv_sec=2;
val.it_interval.tv_usec=0;
setitimer(ITIMER_REAL,&val,NULL);
while(1)
{
sleep(10);
}
return 0;
}
运行结果如下:
但是最后单纯的用CTRL+C是杀不死进程的,需要用kill -9 进程号的形式把他杀死。