一. exec族函数

  1.1. 为什么需要exec函数

    a. fork子进程是为了执行新程序(fork创建了子进程后,子进程和父进程同时被OS调度执行,因此子进程可以单独的执行一个程序,这个程序宏观上将会和父进程程序同时进行)

    b. 使用exec族运行新的可执行程序(exec族函数可以直接把一个编译好的可执行程序直接加载运行)

    c. 我们有了exec族函数后,我们典型的父子进程程序是这样的:子进程需要运行的程序被单独编写、单独编译连接成一个可执行程序(叫hello),(项目是一个多进程项目)主程序为父进程,fork创建了子进程后在子进程中exec来执行hello,达到父子进程分别做不同程序同时(宏观上)运行的效果。

  1.2. 函数介绍


    a. execl和execv 这两个函数是最基本的exec,都可以用来执行一个程序,区别是传参的格式不同。execl是把参数列表(本质上是多个字符串,必须以NULL结尾)依次排列而成(l其实就是list的缩写),execv是把参数列表事先放入一个字符串数组中,再把这个字符串数组传给execv函数。

    b. execlp和execvp 这两个函数在上面2个基础上加了p,较上面2个来说,区别是:上面2个执行程序时必须指定可执行程序的全路径(如果exec没有找到path这个文件则直接报错),而加了p的传递的可以是file(也可以是path,只不过兼容了file。加了p的这两个函数会首先去找file,如果找到则执行执行,如果没找到则会去环境变量PATH所指定的目录下去找,如果找到则执行如果没找到则报错)

    c. execle和execvpe 这两个函数较基本exec来说加了e,函数的参数列表中也多了一个字符串数组envp形参,e就是environment环境变量的意思,和基本版本的exec的区别就是:执行可执行程序时会多传一个环境变量的字符串数组给待执行的程序。

  1.3. main函数的第三个    

    a. main函数的原型其实不止是int main(int argc, char **argv),而可以是int main(int argc, char **argv, char **env) 第三个参数是一个字符串数组,内容是环境变量。

    b. 如果用户在执行这个程序时没有传递第三个参数,则程序会自动从父进程继承一份环境变量(默认的,最早来源于OS中的环境变量);如果我们exec的时候使用execlp或者execvpe去给传一个envp数组,则程序中的实际环境变量是我们传递的这一份(取代了默认的从父进程继承来的那一份)

二. 进程状态和system函数

  2.1. 进程的5种状态

    a. 就绪态。这个进程当前所有运行条件就绪,只要得到了CPU时间就能直接运行。
    b. 运行态。就绪态时得到了CPU就进入运行态开始运行。
    c. 僵尸态。进程已经结束但是父进程还没来得及回收
    d. 等待态(浅度睡眠&深度睡眠),进程在等待某种条件,条件成熟后可进入就绪态。等待态下就算你给他CPU调度进程也无法执行。浅度睡眠等待时进程可以被(信号)唤醒,而深度睡眠等待时不能被唤醒只能等待的条件到了才能结束睡眠状态。
    e. 暂停态。暂停并不是进程的终止,只是被被人(信号)暂停了,还可以回复的。

  2.2. system函数

    2.2.1. 函数源码

#include
#include
#include
#include int system(const char * cmdstring)
{
pid_t pid;
int status; if(cmdstring == NULL){ return ();
} if((pid = fork())<){ status = -;
}
else if(pid == ){
execl("/bin/sh", "sh", "-c", cmdstring, (char *));
-exit(); //子进程正常执行则不会执行此语句
}
else{
while(waitpid(pid, &status, ) < ){
if(errno != EINTER){
status = -;
break;
}
}
}
return status;
}

    2.2.2. Linux下使用system函数一定要谨慎

三. 进程关系划分

  3.1. 父子进程关系

    a. 上述已介绍

  3.2. 进程组(group)

    a. process group(进程组)是一组相关联的进程,用来方便信号量的分发

  3.3. 会话(session)

    a. session(回话)是用户登录系统以后所需的context(上下文)

    b. session退出以后所有隶属于该session的进程组都会收到hup信号而挂起,这样就有了控制进程生命周期的作用

四. 守护进程

  4.1. 什么是守护进程

    4.1.1. 何谓守护进程-后台运行

    4.1.2. daemon,表示守护进程,简称为d(进程名后面带d的基本就是守护进程)

    4.1.3. 与控制台脱离(普通进程都和运行该进程的控制台相绑定,表现为如果终端被强制关闭了则这个终端中运行的所有进程都被会关闭,背后的问题还在于会话)

    4.1.4. 服务器(Server),服务器程序就是一个一直在运行的程序,可以给我们提供某种服务(譬如nfs服务器给我们提供nfs通信方式),当我们程序需要这种服务时我们可以调用服务器程序(和服务器程序通信以得到服务器程序的帮助)来进程这种服务操作。服务器程序一般都实现为守护进程。

  4.2. 关闭守护进程

    4.2.1. 由于守护进程与控制台无关,故不能通过关闭控制台关闭

    4.2.2. 向进程发送信号指令kill

      a. kill -信号编号 进程ID,向一个进程发送一个信号

      b. kill -9 xxx,将向xxx这个进程发送9号信号,也就是要结束进程

  4.3. 常见守护进程

    a. syslogd,系统日志守护进程,提供syslog功能。

    b. cron,cron进程用来实现操作系统的时间管理,linux中实现定时执行程序的功能就要用到cron。

root@ubuntu:/mnt/hgfs/windows_share/baseC/APPNet# ps -aux | grep "syslogd"
syslog 0.0 0.0 ? Ssl Mar11 : rsyslogd
root@ubuntu:/mnt/hgfs/windows_share/baseC/APPNet#

  4.4. 编写简单守护进程

    4.4.1. 任何一个进程都可以将自己实现成守护进程

    4.4.2. create_daemon函数的使用

      a. 子进程等待父进程退出

      b. 子进程使用setsid创建新的会话期,脱离控制台
      c. 调用chdir将当前工作目录设置为/
      d. umask设置为0以取消任何文件权限屏蔽
      e. 关闭所有文件描述符
      f. 将0、1、2定位到/dev/null

#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h> void create_daemon(void); int main(void)
{
create_daemon(); while ()
{
printf("I am running.\n"); sleep();
} return ;
} // 函数作用就是把调用该函数的进程变成一个守护进程
void create_daemon(void)
{
pid_t pid = ; pid = fork();
if (pid < )
{
perror("fork");
exit(-);
}
if (pid > )
{
exit(); // 父进程直接退出
}
usleep();
// 执行到这里就是子进程 // setsid将当前进程设置为一个新的会话期session,目的就是让当前进程
// 脱离控制台。
pid = setsid();
if (pid < )
{
perror("setsid");
exit(-);
} // 将当前进程工作目录设置为根目录
chdir("/"); // umask设置为0确保将来进程有最大的文件操作权限
umask(); // 关闭所有文件描述符
// 先要获取当前系统中所允许打开的最大文件描述符数目
int cnt = sysconf(_SC_OPEN_MAX);
int i = ;
for (i=; i<cnt; i++)
{
close(i);
} open("/dev/null", O_RDWR);
open("/dev/null", O_RDWR);
open("/dev/null", O_RDWR); }

  4.5. 使用syslog来记录调试信息

    4.5.1. openlog、syslog、closelog的使用

    4.5.2. 一般log信息都在操作系统的/var/log/messages这个文件中存储着,但是ubuntu中是在/var/log/syslog文件中的。

#include <stdio.h>
#include <syslog.h>
#include <sys/types.h>
#include <unistd.h> int main(void)
{
printf("my pid = %d.\n", getpid()); openlog("b.out", LOG_PID | LOG_CONS, LOG_USER); syslog(LOG_INFO, "this is my log info.%d", ); syslog(LOG_INFO, "this is another log info.");
syslog(LOG_INFO, "this is 3th log info."); closelog();
}

linux 进程2的更多相关文章

  1. Linux进程管理及while循环

    目录 进程的相关概念 进程查看及管理工具的使用 Linux系统作业控制 调整进程优先级 网络客户端工具 bash之while循环 20.1.进程类型 守护进程 daemon,在系统引导过程中启动的进程 ...

  2. 如何灵活运用Linux 进程资源监控和进程限制

    导读 每个 Linux 系统管理员都应该知道如何验证硬件.资源和主要进程的完整性和可用性.另外,基于每个用户设置资源限制也是其中一项必备技能. 在这篇文章中,我们会介绍一些能够确保系统硬件和软件正常工 ...

  3. TODO:Golang Linux进程退出说明

    TODO:Golang Linux进程退出说明 Golang使用os.Exit(code)进程退出导致当前程序退出并返回给定的状态代码.传统上,code代码为零表示成功退出,非零错误退出. sysca ...

  4. Linux进程管理子系统分析【转】

    本文转载自:http://blog.csdn.net/coding__madman/article/details/51298732 Linux进程管理: 进程与程序: 程序:存放在磁盘上的一系列代码 ...

  5. 12个Linux进程管理命令介绍(转)

    12个Linux进程管理命令介绍 [日期:2015-06-02] 来源:Linux中国  作者:Linux [字体:大 中 小]   执行中的程序在称作进程.当程序以可执行文件存放在存储中,并且运行的 ...

  6. linux 进程管理相关内容

    简介 当我们运行程序时,Linux会为程序创建一个特殊的环境,该环境包含程序运行需要的所有资源,以保证程序能够独立运行,不受其他程序的干扰.这个特殊的环境就称为进程. 每个 Linux 命令都与系统中 ...

  7. Linux - 进程查看与管理

    标签(空格分隔): Linux 进程的静态查看 查看系统所有进程 ps -ef -- 输出来好乱,看不懂..: ps aux -- a表示所有与终端相关的进程,u表示所有以用户组织的进程状态的信息,x ...

  8. Linux进程关系

    Linux进程关系   作者:Vamei 出处:http://www.cnblogs.com/vamei 欢迎转载,也请保留这段声明.谢谢! Linux的进程相互之间有一定的关系.比如说,在Linux ...

  9. Linux进程基础

    Linux进程基础   作者:Vamei 出处:http://www.cnblogs.com/vamei 欢迎转载,也请保留这段声明.谢谢! 计算机实际上可以做的事情实质上非常简单,比如计算两个数的和 ...

  10. linux进程编程:子进程创建及执行函数简介

    linux进程编程:子进程创建及执行函数简介 子进程创建及执行函数有三个: (1)fork();(2)exec();(3)system();    下面分别做详细介绍.(1)fork()    函数定 ...

随机推荐

  1. anaconda历史版本下载

    anaconda历史版本安装: anaconda所有版本链接:https://repo.continuum.io/archive/ 清华大学开源软件镜像站:https://mirrors.tuna.t ...

  2. jmeter--单个接口通,自动化不通时

    单个接口通,自动化不通时,对比两者请求 post 请求的格式,内容编码

  3. JS几种数组遍历方式总结

    JS数组遍历的几种方式 JS数组遍历,基本就是for,forin,foreach,forof,map等等一些方法,以下介绍几种本文分析用到的数组遍历方式以及进行性能分析对比 第一种:普通for循环 代 ...

  4. 关于:基于http协议大文件断点续传上传至web服务器

    关键部分 前端用file.slice()分块 前端用FileReader获取每一分块的md5值 后端用MultipartFile接受分块文件 后端用FileOutputStream拼装分块文件 话不多 ...

  5. UVa 572 Oil Deposits (Floodfill && DFS)

    题意 :输入一个m行n列的字符矩阵,统计字符“@”组成多少个八连块.如果两个字符“@”所在的格子相邻(横竖以及对角方向),就是说它们属于同一个八连块. 分析 :可以考虑种子填充深搜的方法.两重for循 ...

  6. SPOJ 913 Query on a tree II

    spoj题面 Time limit 433 ms //spoj的时限都那么奇怪 Memory limit 1572864 kB //1.5个G,疯了 Code length Limit 15000 B ...

  7. Android网络技术之WebView常用方法

    public class WebViewTest extends Activity {       private WebView wv;     private EditText et;       ...

  8. 我的docker笔记

    下面的链接全部是我在CSDN的关于docker的博文,我认为已经很是详细了,没有再次总结的必要性,特给出链接地址 docker容器技术基础 https://blog.csdn.net/zisefeiz ...

  9. HDU1237--简单计算器(栈的应用)

    Problem Description 读入一个只包含 +, -, *, / 的非负整数计算表达式,计算该表达式的值. Input 测试输入包含若干测试用例,每个测试用例占一行,每行不超过200个字符 ...

  10. [CSP-S模拟测试]:Cicada与排序(概率DP)

    题目传送门(内部题93) 输入格式 第一行一个整数$n$,代表数列的长度. 接下来一行$n$个数$a_i$,用空格分隔开. 输出格式 输出一行$n$个数,表示原数列上这个位置在执行后的期望位置,注意输 ...