【Linux】守护线程(Daemon)

1. 目的:

为了周期性的执行某个任务,或者防止程序出现意外情况挂掉。

2.相关概念:

父进程(PPID),子进程,进程(PID)

  • 场景1:这个线程属于该终端的第一个程序,于是它的父线程就是这个shell,比如下图的ps指令就是这个终端的第一个程序,于是它的父进程就是7073,也就是这个shell的PID。
  • 场景2:这个线程是在程序中创建的子线程,那么这个子线程的父进程ID就是这个程序的ID。比如下图的父进程是22554,创建的子线程是22555,可以看出子线程的父进程就是22554。

进程组(PGID):

  • 组长进程标识: 其进程组ID==其进程ID

  • 组长进程可以创建一个进程组,创建该进程组中的进程,然后终止
  • 只要进程组中有一个进程存在,进程组就存在,与组长进程是否终止无关
  • 进程组可根据这个shell中的不同程序来创建,一般一个终端命令创建一个进程组。而我们的可执行程序也是一组进程组。

会话组(SID):

多个进程组构成一个会话组,比如打开一个终端,有前台进程组,后台进程组等等,然后这些进程组一起构成一个会话组。。

3.创建流程

  1. 创建子进程。父进程退出:fork()
  2. 现有的机制是:某个运行在终端里面的程序,如果关闭终端,就会杀死这个线程。。而使用了setdsid()后,也就是子进程独立出来,不受限于终端。关闭终端后,这个线程也不会被关闭。
  3. 改变工作目录chdir()函数:守护线程--脱离文件目录防止目录被删除,导致守护线程不可用。
  4. 重设文件权限掩码  umask()函数
  5. 关闭文件描述符
  6. 开始守护进程的任务
  7. 退出

目的:

  • 创建一个守护进程:每分钟在$HOME/log创建一个文件:名为程序名,时间戳。。

执行步骤:

  1. 编译生成可执行文件。。。
  2. 执行,在终端可看到打印。。不受终端的限制。。
  3. 如果想要杀死守护线程:ps aux即可看到这个进程,然后kill即可
#include <stdio.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <string.h>
#include <stdlib.h>
#include <sys/time.h>
#include <signal.h>
#include <time.h>

// ps aux即可查看正在运行的线程  ---即可kill 
// 此代码执行后,可脱离父进程(终端)的限制
#define _file_name_prefix_ "%s/log/mydaemon.%s"  // 定义文件格式

void touchfile(int num) {
    char *HomeDir = getenv("HOME");
    //printf("%s\n",HomeDir);
    const time_t demo = time(NULL);
    //printf("%s\n",ctime(&demo));

    char str_Filename[250] = {0};
    sprintf(str_Filename,_file_name_prefix_,HomeDir,ctime(&demo));
    printf("%s\n",str_Filename);
    int fd = open(str_Filename,O_RDWR|O_CREAT, 0666); //设置文件的可执行权限
    if (fd < 0) {
        perror("open error");
        exit(1);
    }
    close(fd);
}

int main() {
    //创建子进程,父进程退出,独立于终端
    pid_t pid = fork(); //父进程复制出一个和当前进程一样的子进程
    if(pid > 0) {
        printf("son\n");    
        //exit(1); // 如果exit结束执行的话,那么子进程会成为孤儿进程,并被init收养。
    } else if(pid == 0) {
        printf("dady\n"); 
    } 
    pid_t pid1 = setsid();
    if (-1 == pid1) {
        printf("this is process leader\n");  //当前线程是会话(session)的领头进程
        exit(1);  // 如果exit结束执行的话,那么子进程会成为孤儿进程,并被init收养。
    } else {
        printf("this is son %d\n",pid1); // 子进程重新获得了一个新的会话(session)id,那么此时父进程死了也不会影响到子进程。。。
                                         // 那么此时起子线程的任务就可以作为守护进程
    }
    //设置掩码
    //umask(0);
    //切换目录
    //chdir(getenv("home"));  // 切换到home目录
    //关闭文件描述符
    //执行核心逻辑
    struct itimerval my_it = {
        {6, 1},  // 每次的任务之间间隔6s+1us
        {1, 2}   // 设置定时器setitimer后1s+2us开始执行任务
        };
    setitimer(ITIMER_REAL, &my_it, NULL);  //ITIMER_REAL--系统真实时间,发出sigalrm信号
    
    struct  sigaction act;
    act.sa_flags = 0;
    sigemptyset(&act.sa_mask);  // 将参数信号集初始化并清空.--这里感觉就是把sa_flags清0
    act.sa_handler = touchfile; // 收到信号执行的函数

    sigaction(SIGALRM,&act,NULL);
    while(1) {
        
        sleep(1);

    }
}

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
主要功能: 1.读取配置文件程序 2.启动进程 3.监控进程,查看进程是否退出或者崩溃 4.若进程退出或者崩溃,重启程序。 5.支持sleep功能 6.进程若连续崩溃NUM_MAX次就进行相应的睡眠周期struct proc_struct proc: struct proc_ struct [mp: if(array) return 0 ∥切换到目录rse chdirldiri ifdp= opendir(dir}=NuLL}开日录/proc,矢败返回0,成功把描述指针返回给d return o 〃将φpro文件夹的描述符指针传递给reεddir,读取文件夹内容,循环赋值给结构体di while ((dirp= readdir(dp))= NULLY char data 301 ∥取文件名称赋值给数组daa(其中包含有进程的名称(pid sprintf(data, "s", dirp->d_name); ∥是否是由字符09组成的字符串,即得到所有进程的pid f((IsDigit(data)) prac =(struct proc_struct )4 malloc(sizeof(struct proc_struct) tmp proc: prac->pid =a: oi(dirp->d_name): It(proc tind( proc. array)) free( tmp); closedir(dp cturn proc_find 两个参数分别是两个进程描述的结构体指针 李比较两个进程pd是否相等 李*相等返回1,不相等返回0 幸率球事容球家草事家事球峰率享事球摩率球享享溶事*事卷寒球套事塞容寒/ int proc find( struct prcc_struct* src, struct proc- struct* dest) char buffer[40%6]. ps cmd[20] It fd. I sprintf(buffer, "ed/star", sre->pid); fd = open(butter, O_RDONLY) if(fd==-1) rerurn 0 memset(buffer, wO, sizeof(buffer)) len= read(fd, bufter, sizeof(bufter )-1) close(ld) if(l return 0: p= butter: p= strrchr(p, C) narq=strrchr(p, )) n=q-p-1 if (len >= sizeof, srt->name)) len= sizeof(src->name)-1 p+ l, len src->namelen]=0; =日 turn(strcmp( src->name, dest dest->name)==0)? 1: 0- 条善参数aay:让程结构体指针;参数sie进程列表数组aray的大小ie:配置文件路径 从配置文件得到指定的程序列表,将对应进程的信息填充到aray数组中 羋执行成功返回进程个数,执行失败返回0 int get_ proc( struct proc_struct array, int size, char file intnRet=o if(! array I‖(si 0)l‖fhle myprinttf"invalid parameterin retun o char line[4096]; FILE fp= fopen(file, T"); if(fp) printf("open file cs fail\n", file) return U memset(line, 0, 4095); while(fgets(lire, 4095, tp)&& nRet size) memcpy(void s)[(&arraylnRet )->cmdline), (void")line, strlen(line)-2 ) tmp= strrchr(line, / ) Lmp += I: memcpy((&array inRet))->name, tmp, strlen(tmp)- 2) nRet++ ); return(nReL); 康棒串串浓凉率旅浓串底率卖毒志着旅浓浓准溶房表 装 startProc *卷参数proc:要启动的进的结构体描述指针 启动程序 执行成功返回1,子进程退出 宗塞家康家家家家家家家家宋家家聚家苯家球察塞家塞家家容家塞家家家家室家家察家家家聚家聚寒撑家装家掌建察家家室事 int startProc (struct proc_ struct* proc, struct proc _struct*must_run_ proc int mIst_run_size static inti=d if( proc)return 0 if(strlen(proc->cmdline I<=0) return 0; int pid= forko: 〃进程内部返回值为0,返回给父进程的为自己的pid inta〓 if(pid pid= fork( ifpd≡0 execl(char")proc->cmdline,(char")prDc->name,NULL); ∥exit: It(o): sleep42片 waiL(NULL) sleep( I: if(i== must run size -1) if(check proc(&must run proc[i])==0) startProc( &mtust_run_proeli], must_run_prce, must_run_size); el if(i== must run size-11 i= else 1++ start Proc( &must_run_proclil, must_run_ proc, mustrun_ _size); !**幸幸串率幸米幸*家*幸毕零*幸幸半字幸字华米*幸半孝率非幸零幸学幸幸车 3a*8*daemon init 幸*启动配置文件当中的需要守护的程序 执行成功返回1,中途出错,返回-1 长界零家墨军零家零率家三哮零座零率零零容岸军零罕型率零零零零牢察座察零零零零季球军零容零 int moniter_ run(struct proc_struct"must_run_proc, int proc_ size) nti=0: for(i=0; i< must_run_size: i ++) ∥监控程序是否正在运行 if(check_ proc(&(must un_ proc[il))<=o) ∥厘新片动程序 startProc(&' must run procli]), must run proc, proc size return I: 幸*事率事率率**率**字幸学摩*率*幸幸学幸半*率幸字****幸中*幸学幸 春*着*信号处理函数 exit_proc 翥安全结束监控的程序 4来没有返回值 告参毒萨响幸帝称昨嗜幸古称索点响卷南都南请南幸难布际本啪昨青市南动南香请非市赤南本 void exit_ proc(int ar InL I struct proc struct proc for(i=0; i< must run_ Size: i++) proc=&(must_run_proc[i]): kill(proc->pid, SIGTERM); exit flag=I exit(o): void exit_proc(int pid 要main L.获取程序列表2启动进程3.监控进程,若程序岀或崩溃,重新启动程序4.收到退 出信号,安全结束监控程序 成功返回1,失败返回0 零牢容容家容字家容容察*禁容容字哮零常字容容容家察容牢容零容容容容容牢字家客字容牢容零容*字容客字容容字容家容容字岩 static void run moniter( void data) 读取程序列表 must_ run _size get proc(must_run_proc, sIZE, data if(rmust run Sizc <=1) return o struct sigaction act, oldact act,sa handler= exit_proc act. sa flags =SA ONESHOT sigaction(SIGTERM, &act, NULL) sigaction(SIGINT, &act, NULL) sigaction(sIGHUP, &act, NULL); 检测并启动未启动的程序 moniter_ run(must run proc, must run slze) eturn null int creat and run moniter(char * file) 开线程: pthread_t moniter_ thread if(pthread_create(&moniter_thread, NULL, run_moniter, file)==0) printf("thread create Ok, check thread start \n") return printf( thread check create Errin"): return -I 零零零零享享事职增零半非寥零享半容零摩率率零享剩率容半半享零半率零半率零率辱寒零享 要 IsDigit 参茶爹数a字符串的地址 *判断字符串是否有纯数字字符组成 春客是纯数字返回1,不是纯数字返回0 喜非要串思率串串串串家串润串串串串串串毒毒喜串串最率毒串串踪串率串串非球毒串妆串串毒串串影零串串毒事串 static int IsDigit[char aD) Int size ∥得到当前字符串的长度 size= strlen(a: ∥若字符串长度为0,则直接返回0:即宇符串为空则返回0: if(size ==0) return 0; ∥循环遍历整个字符串 forli=0; i< Size; i++) ∥如果字符小于字符0,或者大于字符9,则返回0 if(ai]<ol ai>9) retum ∥走到这一步说明字符串由字符09组成,返回1 return l; 主进程源文件:man,c main.c #include"process, h Include <stdio. h> include <stdlib h> 甲 include< unistd. I> 甲 nclude< signal> 却 nclude <sys/ ypes,h include <sys/stat. h> 甲 include< tenth> int main(void) creat_and_run_moniter("proclistini") while(l) sleep(D) turn 以上内容是程序全部源文件
Linux 下,守护进程(daemon)是一种在后台运行的进程,它通常在系统启动时自动启动,并在系统关闭时自动关闭。守护进程是一种长期运行的进程,它不会和任何终端关联,因此无法接收到用户的输入。 在 Linux 中,守护进程通常通过以下步骤实现: 1. 创建子进程,并让子进程成为新的会话组长和进程组长,与控制终端脱离关系。 2. 关闭所有文件描述符,避免占用系统资源。 3. 在后台运行,并定期执行一些操作(如检查日志、处理任务等)。 下面是一个简单的守护进程的代码示例: ```c #include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <sys/types.h> #include <sys/stat.h> #include <fcntl.h> int main() { pid_t pid, sid; int fd; // 创建子进程 pid = fork(); if (pid < 0) { perror("fork error"); exit(EXIT_FAILURE); } if (pid > 0) { // 父进程退出 exit(EXIT_SUCCESS); } // 子进程成为新的会话组长和进程组长 sid = setsid(); if (sid < 0) { perror("setsid error"); exit(EXIT_FAILURE); } // 关闭所有文件描述符 fd = open("/dev/null", O_RDWR, 0); if (fd < 0) { perror("open error"); exit(EXIT_FAILURE); } dup2(fd, STDIN_FILENO); dup2(fd, STDOUT_FILENO); dup2(fd, STDERR_FILENO); if (fd > STDERR_FILENO) { close(fd); } // 在后台运行 while (1) { // 定期执行一些操作 sleep(60); } exit(EXIT_SUCCESS); } ``` 在这个代码示例中,我们首先创建了一个子进程,并让子进程成为新的会话组长和进程组长,然后关闭了所有文件描述符,最后在后台运行,并定期执行一些操作。需要注意的是,在这个示例中,我们将标准输入、输出和错误输出都重定向到了 /dev/null,这样可以避免输出到控制台中,从而确保守护进程在后台运行。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值