Android C++系列:Linux进程(一)

简介: 我们知道,每个进程在内核中都有一个进程控制块(PCB)来维护进程相关的信 息,Linux内核的进程控制块是task_struct结构体。现在我们全面了解一下其中都有哪 些信息。

image.png


1. 进程概念


我们知道,每个进程在内核中都有一个进程控制块(PCB)来维护进程相关的信 息,Linux内核的进程控制块是task_struct结构体。现在我们全面了解一下其中都有哪 些信息。


  • 进程id。系统中每个进程有唯一的id,在C语言中用pid_t类型表示,其实就是一个非 负整数。
  • 进程的状态,有运行、挂起、停止、僵尸等状态。
  • 进程切换时需要保存和恢复的一些CPU寄存器。
  • 描述虚拟地址空间的信息。
  • 描述控制终端的信息。
  • 当前工作目录(Current Working Directory)。
  • umask掩码。
  • 文件描述符表,包含很多指向file结构体的指针。
  • 和信号相关的信息。
  • 用户id和组id。
  • 控制终端、Session和进程组。
  • 进程可以使用的资源上限(Resource Limit)。


目前大家并不需要理解这些信息的细节,在随后的文章中我们会知道它是保 存在PCB中的。


fork和exec是本章要介绍的两个重要的系统调用。fork的作用是根据一个现有的进程复 制出一个新进程,原来的进程称为父进程(Parent Process),新进程称为子进程(Child Process)。系统中同时运行着很多进程,这些进程都是从最初只有一个进程开始一个一个 复制出来的。在Shell下输入命令可以运行一个程序,是因为Shell进程在读取用户输入的命 令之后会调用fork复制出一个新的Shell进程,然后新的Shell进程调用exec执行新的程序。


我们知道一个程序可以多次加载到内存,成为同时运行的多个进程,例如可以同时开多 个终端窗口运行/bin/bash,另一方面,一个进程在调用exec前后也可以分别执行两个不同 的程序,例如在Shell提示符下输入命令ls,首先fork创建子进程,这时子进程仍在执行/ bin/bash程序,然后子进程调用exec执行新的程序/bin/ls 。


image.png


2. 进程环境


libc中定义的全局变量environ指向环境变量表,environ没有包含在任何头文件中,所 以在使用时要用extern声明。例如:


#include <stdio.h>
int main(void) {
  extern char **environ;
  int i;
  for(i=0; environ[i]!=NULL; i++)
    printf("%s\n", environ[i]); return 0;
}


由于父进程在调用fork创建子进程时会把自己的环境变量表也复制给子进程,所以 a.out打印的环境变量和Shell进程的环境变量是相同的。


按照惯例,环境变量字符串都是name=value这样的形式,大多数name由大写字母加下划 线组成,一般把name的部分叫做环境变量,value的部分则是环境变量的值。环境变量定义 了进程的运行环境,一些比较重要的环境变量的含义如下:


  • PATH:可执行文件的搜索路径。ls命令也是一个程序,执行它不需要提供完整的路径名/bin/ ls,然而通常我们执行当前目录下的程序a.out却需要提供完整的路径名./a.out,这 是因为PATH环境变量的值里面包含了ls命令所在的目录/bin,却不包含a.out所在的目 录。PATH环境变量的值可以包含多个目录,用:号隔开。在Shell中用echo命令可以查 看这个环境变量的值:$ echo $PATH
  • SHELL:当前Shell,它的值通常是/bin/bash。
  • TERM:当前终端类型,在图形界面终端下它的值通常是xterm,终端类型决定了一些程序的输 出显示方式,比如图形界面终端可以显示汉字,而字符终端一般不行。
  • LANG:语言和locale,决定了字符编码以及时间、货币等信息的显示格式。
  • HOME:当前用户主目录的路径,很多程序需要在主目录下保存配置文件,使得每个用户在运 行该程序时都有自己的一套配置。


用environ指针可以查看所有环境变量字符串,但是不够方便,如果给出name要在环境变量 表中查找它对应的value,可以用getenv函数。


#include <stdlib.h>
char *getenv(const char *name); 
getenv的返回值是指向value的指针,若未找到则为NULL。


修改环境变量可以用以下函数:


#include <stdlib.h>
int setenv(const char *name, const char *value, int rewrite); void unsetenv(const char *name);
putenv和setenv函数若成功则返回为0,若出错则返回非0。


setenv将环境变量name的值设置为value。如果已存在环境变量name,那么 若rewrite非0,则覆盖原来的定义; 若rewrite为0,则不覆盖原来的定义,也不返回错误。 unsetenv删除name的定义。即使name没有定义也不返回错误。


例修改环境变量


#include <stdlib.h>
#include <stdio.h>
int main(void) {
  printf("PATH=%s\n", getenv("PATH")); 
  setenv("PATH", "hello", 1); 
  printf("PATH=%s\n", getenv("PATH")); 
  return 0;
}


3.进程状态


修改进程资源限制,软限制可改,最大值不能超过硬限制,硬限制只有root用户可以修改


#include <sys/time.h>
#include <sys/resource.h>
int getrlimit(int resource, struct rlimit *rlim);
int setrlimit(int resource, const struct rlimit *rlim);


查看进程资源限制:


cat /proc/self/limits ulimit -a


4. 小结


本文介绍了进程的基本概念,进程控制块信息,fork和exec两个主要的系统调用。以及进程环境的修改及获取,进程状态的修改与获取。

目录
相关文章
|
1月前
|
算法 Linux 调度
深入理解Linux操作系统的进程管理
本文旨在探讨Linux操作系统中的进程管理机制,包括进程的创建、执行、调度和终止等环节。通过对Linux内核中相关模块的分析,揭示其高效的进程管理策略,为开发者提供优化程序性能和资源利用率的参考。
80 1
|
12天前
|
消息中间件 Linux
Linux:进程间通信(共享内存详细讲解以及小项目使用和相关指令、消息队列、信号量)
通过上述讲解和代码示例,您可以理解和实现Linux系统中的进程间通信机制,包括共享内存、消息队列和信号量。这些机制在实际开发中非常重要,能够提高系统的并发处理能力和数据通信效率。希望本文能为您的学习和开发提供实用的指导和帮助。
72 20
|
1月前
|
存储 监控 Linux
嵌入式Linux系统编程 — 5.3 times、clock函数获取进程时间
在嵌入式Linux系统编程中,`times`和 `clock`函数是获取进程时间的两个重要工具。`times`函数提供了更详细的进程和子进程时间信息,而 `clock`函数则提供了更简单的处理器时间获取方法。根据具体需求选择合适的函数,可以更有效地进行性能分析和资源管理。通过本文的介绍,希望能帮助您更好地理解和使用这两个函数,提高嵌入式系统编程的效率和效果。
105 13
|
1月前
|
SQL 运维 监控
南大通用GBase 8a MPP Cluster Linux端SQL进程监控工具
南大通用GBase 8a MPP Cluster Linux端SQL进程监控工具
|
1月前
|
运维 监控 Linux
Linux操作系统的守护进程与服务管理深度剖析####
本文作为一篇技术性文章,旨在深入探讨Linux操作系统中守护进程与服务管理的机制、工具及实践策略。不同于传统的摘要概述,本文将以“守护进程的生命周期”为核心线索,串联起Linux服务管理的各个方面,从守护进程的定义与特性出发,逐步深入到Systemd的工作原理、服务单元文件编写、服务状态管理以及故障排查技巧,为读者呈现一幅Linux服务管理的全景图。 ####
|
2月前
|
缓存 算法 Linux
Linux内核的心脏:深入理解进程调度器
本文探讨了Linux操作系统中至关重要的组成部分——进程调度器。通过分析其工作原理、调度算法以及在不同场景下的表现,揭示它是如何高效管理CPU资源,确保系统响应性和公平性的。本文旨在为读者提供一个清晰的视图,了解在多任务环境下,Linux是如何智能地分配处理器时间给各个进程的。
|
2月前
|
网络协议 Linux 虚拟化
如何在 Linux 系统中查看进程的详细信息?
如何在 Linux 系统中查看进程的详细信息?
268 1
|
6月前
|
运维 关系型数据库 MySQL
掌握taskset:优化你的Linux进程,提升系统性能
在多核处理器成为现代计算标准的今天,运维人员和性能调优人员面临着如何有效利用这些处理能力的挑战。优化进程运行的位置不仅可以提高性能,还能更好地管理和分配系统资源。 其中,taskset命令是一个强大的工具,它允许管理员将进程绑定到特定的CPU核心,减少上下文切换的开销,从而提升整体效率。
掌握taskset:优化你的Linux进程,提升系统性能
|
6月前
|
弹性计算 Linux 区块链
Linux系统CPU异常占用(minerd 、tplink等挖矿进程)
Linux系统CPU异常占用(minerd 、tplink等挖矿进程)
208 4
Linux系统CPU异常占用(minerd 、tplink等挖矿进程)
|
5月前
|
算法 Linux 调度
探索进程调度:Linux内核中的完全公平调度器
【8月更文挑战第2天】在操作系统的心脏——内核中,进程调度算法扮演着至关重要的角色。本文将深入探讨Linux内核中的完全公平调度器(Completely Fair Scheduler, CFS),一个旨在提供公平时间分配给所有进程的调度器。我们将通过代码示例,理解CFS如何管理运行队列、选择下一个运行进程以及如何对实时负载进行响应。文章将揭示CFS的设计哲学,并展示其如何在现代多任务计算环境中实现高效的资源分配。