Linux编程之《守护进程》
Intro
-----
守护进程,也就是通常说的Daemon进程,是Linux中的后台服务进程。它是一个生存期较长的进程,通常独立于控制终端并且周期性地执行某种任务或等待处理某些发生的事件。守护进程常常在系统引导装入时启动,在系统关闭时终止。Linux系统有很多守护进程,大多数服务都是通过守护进程实现的,同时,守护进程还能完成许多系统任务,例如,作业规划进程crond、打印进程lqd等(这里的结尾字母d就是Daemon的意思)。
由于在Linux中,每一个系统与用户进行交流的界面称为终端,每一个从此终端开始运行的进程都会依附于这个终端,这个终端就称为这些进程的控制终端,当控制终端被关闭时,相应的进程都会自动关闭。但是守护进程却能够突破这种限制,它从被执行开始运转,直到整个系统关闭时才退出。如果想让某个进程不因为用户或终端或其他地变化而受到影响,那么就必须把这个进程变成一个守护进程,下面将完整代码贴上。
/************************************************
* 该例程讲解Linux守护进程的编程方法
************************************************/
#include <unistd.h>
#include <signal.h>
#include <sys/param.h> // NOFILE
#include <sys/stat.h> // umask
#include <stdio.h>
#include <stdlib.h>
#include <time.h>
#include <assert.h> bool initDaemon()
{
// 屏蔽一些有关控制终端操作的信号
// 防止守护进程没有正常运转起来时,因控制终端受到干扰退出或挂起
assert(signal(SIGINT, SIG_IGN) != SIG_ERR); // 终端中断
assert(signal(SIGHUP, SIG_IGN) != SIG_ERR); // 连接挂断
assert(signal(SIGQUIT, SIG_IGN) != SIG_ERR);// 终端退出
assert(signal(SIGPIPE, SIG_IGN) != SIG_ERR);// 向无读进程的管道写数据
assert(signal(SIGTTOU, SIG_IGN) != SIG_ERR);// 后台程序尝试写操作
assert(signal(SIGTTIN, SIG_IGN) != SIG_ERR);// 后台程序尝试读操作
assert(signal(SIGTERM, SIG_IGN) != SIG_ERR);// 终止 // [1] 创建一个子进程,父进程退出
int pid = fork();
if (pid)
{
// 父进程退出
exit();
}
else if (pid < )
{
return false;
} // 子进程继续运行 // [2] 在子进程中创建新的会话,setsid有三个作用
// a.让进程摆脱原会话的控制
// b.让进程摆脱原进程组的控制
// c.让进程摆脱原控制终端的控制
int ret = setsid();
if (ret < )
{
return false;
} // [3] 禁止进程重新打开控制终端
// 进程已经成为无终端的会话组长,但它可以重新申请打开一个控制终端
// 可以通过使进程不再成为会话组长来禁止进程重新打开控制终端
pid = fork();
if (pid)
{
// 结束第一个子进程
exit();
}
else if (pid < )
{
return false;
} // 第二个子进程继续运行 // [4] 关闭打开的文件描述符
// 进程从创建它的父进程那里继承了打开的文件描述符,如果不关闭,将会浪费系统资源,
// 造成进程所在的文件系统无法卸下以及引起无法预料的错误
for (int i = ; i < NOFILE; ++i)
{
close(i);
} // [5] 改变当前工作目录
// 进程活动时,其工作目录所在的文件系统不能卸下,一般将工作目录改变到根目录
ret = chdir("/");
if (ret < )
{
return false;
} // [6] 重新设置文件创建掩码
// 进程从创建它的父进程那里继承了文件创建掩码,它可能修改守护进程所创建的文件的存取位
// 所以将文件创建掩码清除
umask(); return true;
} int main()
{
// 初始化守护进程
bool ret = initDaemon();
if (!ret)
{
printf("Init daemon failed\n");
return ;
} FILE* file = NULL;
time_t t = ; // 每隔10秒向test.log报告运行状态
while (true)
{
sleep();
file = fopen("./var/test.log", "a+");
if (file != NULL)
{
t = time(NULL);
fprintf(file, "I am here at %s\n", asctime(localtime(&t)));
fclose(file);
}
} return ;
}
该例子的github地址:https://github.com/chxuan/samples/blob/master/Daemon/Daemon.cpp
Linux编程之《守护进程》的更多相关文章
- Linux编程之《看门狗进程》
Intro 当我们编写服务器代码时,为了让自己的服务器在意外崩溃时能够及时的重启,软件看门狗就显示出它的作用了,该看门狗进程是通过fork一个子进程(业务进程),父进程一旦捕获到了子进程的结束信号就重 ...
- Linux编程 7 (实时监测进程 top, 结束进程kill,killall)
一. 实时监测进程 top 在一篇里讲到ps命令在收集进程信息时非常有用,但它只能显示某个特定时间点的信息.想要观察那些频繁换进换出的内存进程趋势,用top命令是合适的.使用top命令如下图所示: 在 ...
- 10.3、android输入系统_必备Linux编程知识_任意进程双向通信(scoketpair+binder)
3. 任意进程间通信(socketpair_binder) 进程每执行一次open打开文件,都会在内核中有一个file结构体表示它: 对每一个进程在内核中都会有一个task_struct表示进程,这个 ...
- Linux 下如何使用看门狗
Linux内核有集成WD的选项.将其使能后,系统里就会有watchdog的设备驱动:/dev/watchdog.这样,在应用程序里只需打开这个设备使用即可:#include <fcntl.h ...
- Linux 软件看门狗 watchdog 喂狗
Linux 自带了一个 watchdog 的实现,用于监视系统的运行,包括一个内核 watchdog module 和一个用户空间的 watchdog程序.内核 watchdog 模块通过 /dev/ ...
- 【目录】linux 编程
随笔分类 - linux 编程 Linux编程 24 shell编程(结构化 if [ condition ] 数值比较,字符串比较) 摘要: 一.概述 接着上篇讲的结构化命令,最后讲到了test命令 ...
- linux设备驱动归纳总结(十一):写个简单的看门狗驱动【转】
本文转载自:http://blog.chinaunix.net/uid-25014876-id-112879.html linux设备驱动归纳总结(十一):写个简单的看门狗驱动 xxxxxxxxxxx ...
- linux内核中断之看门狗
一:内核中断 linux内核中的看门狗中断跟之前的裸板的中断差不多,在编写驱动之前,需要线把内核自带的watch dog模块裁剪掉,要不然会出现错误:在Device Drivers /Watchdog ...
- Linux C语言编程学习笔记 (1)进程控制入门
想进行Linux系统开发已经很久了,一直没有付诸实践.今日终于开始学习Linux下的C语言编程,研究一天,终于大概弄明白了Linux系统进程管理的一些基本概念和编程方法,总结下来以方便大家学习和自己实 ...
随机推荐
- 关于webpack最好的文档
这几天研究webpack打包工具,在网上搜了无数的资料,鱼龙混杂.看了几十份资料,依然没有一个可以完整的描述的. 折腾了那么久,还是放弃治疗了.回到官网,一字一句的阅读,一个小时就彻底明白了. 学习新 ...
- 关于如果修改 ie 浏览器 文本模式
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/html4/stric ...
- Matlab文件操作
1. Matlab文件操作主要有三个步骤:首先打开文件,然后对文件进行读写操作,最后要关闭文件. 2. fid=fopen(文件名,打开方式) 'r' 只读,文件必须存在(缺省的打开方式) 'w' ...
- 解开发者之痛:中国移动MySQL数据库优化最佳实践(转)
开源数据库MySQL比较容易碰到性能瓶颈,为此经常需要对MySQL数据库进行优化,而MySQL数据库优化需要运维DBA与相关开发共同参与,其中MySQL参数及服务器配置优化主要由运维DBA完成,开发则 ...
- 15个顶级Java多线程面试题及答案
1)现在有T1.T2.T3三个线程,你怎样保证T2在T1执行完后执行,T3在T2执行完后执行? 这个线程问题通常会在第一轮或电话面试阶段被问到,目的是检测你对”join”方法是否熟悉.这个多线程问题比 ...
- I - Control - HDU 4289 (最大流)
题意:有N个城市,现在城市S出现了一伙歹徒,他们想运送一些炸弹到D城市,不过警方已经得到了线报知道他们的事情,不过警察不知道他们所在的具体位置,所以只能采取封锁城市的办法来阻断暴徒,不过封锁城市是需要 ...
- POJ 3468 A Simple Problem with Integers (线段树成段更新)
题目链接:http://poj.org/problem?id=3468 题意就是给你一组数据,成段累加,成段查询. 很久之前做的,复习了一下成段更新,就是在单点更新基础上多了一个懒惰标记变量.upda ...
- C#中垃圾回收与内存管理机制
今天抽空来讨论一下.Net的垃圾回收与内存管理机制,也算是完成上个<WCF分布式开发必备知识>系列后的一次休息吧.以前被别人面试的时候问过我GC工作原理的问题,我现在面试新人的时候偶尔也会 ...
- Spring Data JPA教程, 第六部分: Sorting(未翻译)
The fifth part of my Spring Data JPA tutorialdescribed how you can create advanced queries with Spri ...
- 贪心-poj-2437-Muddy roads
题目链接: http://poj.org/problem?id=2437 题目意思: 给n个区间,每次可以用长度为L的棒A去覆盖,求将所有区间覆盖至少需要几次.给的区间不覆盖. 解题思路: 简单贪心. ...