资源与限制
运行在Linux系统上的程序是有资源限制的。这些也许是硬件引起的限制(例如内存),也许由系统策略引起的限制(例如,允许 的CPU时间),或者是实现的限制(例如,整数的尺寸或是文件名允许的最大字符个数)。Unix规范定义了一些可以由程序确定的限制。在第7章我们会进行 更为深入的讨论。
limits.h头文件定义了许多表示操作系统限制的常量。他们包括:
限制常量 用途
NAME_MAX 文件名中的最大字符个数
CHAR_BIT 一个字符值的位数
CHAR_MAX 最大字符值
INT_MAX 最大int值
还有许多其他的可以为程序使用的限制,要得到更为详细的信息,我们可以查看我们安装的头文件。
注意:NAME_MAX是文件系统相关的。为了更为可移植的代码,我们应使用pathconf函数。我们可以查看pathconf的手册页来得到更为详细的信息。
头文件sys/resource.h为资源操作提供了定义。他们包括在一个程序允许的尺寸,执行的优先级以及文件上确定和设置限制的函数。
#include <sys/resource.h>
int getpriority(int which, id_t who);
int setpriority(int which, id_t who, int priority);
int getrlimit(int resource, struct rlimit *r_limit);
int setrlimit(int resource, const struct rlimit *r_limit);
int getrusage(int who, struct rusage *r_usage);
id_t是一个用于组与用户标识的整数类型。rusage结构定义在sys/resource.h文件中,用于确定当前程序已使用了多少CPU时间。他必须至少包含下列成员:
rusage成员 描述
struct timeval ru_utime 使用的用户时间
struct timeval ru_stime 使用的系统时间
timeval结构定义在sys/time.h文件中,包含分别代表秒与微秒的tv_sev与tv_usec域。
一个程序所消耗的CPU时间被分为用户时间(程序本身执行他自己的指令所消耗的时间)和系统时间(操作系统执行程序的行为所消耗的时间;也就是说是消耗在执行输入和输出的系统调用或是其他的系统函数上的时间)。
getrusage函数将CPU时间信息写入由参数r_usage所指向的rusage结构。who参数可是下面的常之一:
who常量 描述
RUSAGE_SELF 只返回当前程序的使用
RUSAGE_CHILDREN 同时也包含子进程的使用信息
我们会在第11章讨论子进程以及任务优先级,但是为了完整,我们会在这里讨论他们所涉及的系统资源。就目前而言,我们可以说每一个运行的程序都具有一个与其相关联的优先级,而具有高优先级的程序会分配到更多的可用CPU时间。
程序可以使用getpriority和setpriority函数来确定与修改他们的优先级。优先级函数要检测或是改变的进程可以由进程标识符,组标识符,或是用户来标识。which参数指明了要如何处理who参数。
which参数 描述
PRIO_PROCESS who是一个进程标识符
PRIO_PGRP who是一个进程组
PRIO_USER who是一个用户标识符
所以,要确定当前进程的标识符,我们可以调用下面的函数:
priority = getpriority(PRIO_PROCESS, getpid());
如果可能,setpriority函数会允许设置一个新的优先级。
默认的优先级为0。正数优先级用于没有更高的优先级任务准备运行时运行的后台任务。负数优先级会使得一个程序更为频繁的运行,占用更多的可用CPU时间。可用的优先级范围为-20到+20。这通常会让人感到迷惑,数字值越高,执行的优先级越低。
如 果成功,getpriority会返回一个正确的优先级,如果失败则会返回-1,并且会设置errno变量。因为-1本身是一个可用的优先级,所以在调用 getpriority之前应将errno设置为0,并且在函数返回检查是否仍为0。如果成功setpriority函数会返回0,否则返回-1。
系统资源的限制可以通过getrlimit与setrlimit函数进行读取与设置。这两个函数都会使用一个通用目的的结构,rlimit,来描述资源限制。他是在sys/resource.h中定义,并且包含下列成员:
rlimit成员 描述
rlim_t rlim_cur 当前的软限制
rlim_t rlim_max 硬限制
rlim_t 是一个整数类型,用来描述资源级别。通常,软限制是一个不应超过的建议限制;如果这样做会使得库函数返回错误。如果超过硬限制也许会使用系统试图向其程序 发送消息来终结其运行。例如超过CPU时间的SIGXCPU信号以及超过数据尺寸限制的SIGSEGV信号。一个程序也许会将其软限制设置为小于硬限制的 任何值,也可能减少其硬限制。只有以超级权限运行的程序才可以增加硬限制。
存在许多可以限制的系统资源。这是由rlimit函数的resource参数来指定的,他们定义在sys/resource.h中,描述如下:
resource参数 描述
RLIMIT_CORE 以字节表示的核心复制文件尺寸
RLIMIT_CPU 以秒表示的CPU时间限制
RLIMIT_DATA 以字节表示的data()段限制
RLIMIT_FSIZE 以字节表示的文件尺寸限制
RLIMIT_NOFILE 可打开的文件数目的限制
RLIMIT_STACK 以字节表示的栈尺寸的限制
RLIMIT_AS 以字节表示的地址空间的限制(栈与数据)
下面的试验部分演示了一个程序,limits.c,来模拟一个通常的程序。他同时设置与突破资源限制。
试验--资源限制
1 包含我们程序中会用到的所有函数所需要头文件:
#include <sys/types.h>
#include <sys/resource.h>
#include <sys/time.h>
#include <unistd.h>
#include <stdio.h>
#include <math.h>
2 work函数将一个字符串写入一个临时文件10000次,然后执行一些算术运行来在CPU上产生负载:
void work()
{
FILE *f;
int i;
double x = 4.5;
f = tmpfile();
for(i = 0; i < 10000; i++) {
fprintf(f,”Do some output\n”);
if(ferror(f)) {
fprintf(stderr,”Error writing to temporary file\n”);
exit(1);
}
}
for(i = 0; i < 1000000; i++)
x = log(x*x + 3.21);
}
3 main函数调用work函数,然后使用getrusage函数来计算他已经使用了多少CPU时间。他会在屏幕上显示这些信息:
int main()
{
struct rusage r_usage;
struct rlimit r_limit;
int priority;
work();
getrusage(RUSAGE_SELF, &r_usage);
printf(“CPU usage: User = %ld.%06ld, System = %ld.%06ld\n”,
r_usage.ru_utime.tv_sec, r_usage.ru_utime.tv_usec,
r_usage.ru_stime.tv_sec, r_usage.ru_stime.tv_usec);
4 接下来,他会调用getpriority函数与getrlimit函数来分别查看当前的级别以及文件尺寸限制:
priority = getpriority(PRIO_PROCESS, getpid());
printf(“Current priority = %d\n”, priority);
getrlimit(RLIMIT_FSIZE, &r_limit);
printf(“Current FSIZE limit: soft = %ld, hard = %ld\n”,
r_limit.rlim_cur, r_limit.rlim_max);
5 最后,我们使用setrlimit函数来设置一个文件尺寸限制并且再次调用work函数,这会失败,因为他在尝试创建一个太大的文件:
r_limit.rlim_cur = 2048;
r_limit.rlim_max = 4096;
printf(“Setting a 2K file size limit\n”);
setrlimit(RLIMIT_FSIZE, &r_limit);
work();
exit(0);
}
当我们运行这个程序时,我们会看到他已消耗了多少CPU时间以及程序运行的默认优先级。一旦设置了文件尺寸限制,这个程序就不可以向一个临时文件写入大于2048字节的内容了。
$ cc -o limits limits.c -lm
$ ./limits
CPU usage: User = 0.980000, System = 0.010000
Current priority = 0
Current FSIZE limit: soft = -1, hard = -1
Setting a 2K file size limit
File size limit exceeded
我们可以通过使用nice命令启动来改变程序的优先级。在这里,我们将程序的优先级改为+10,结果,系统会花费更长的时间来执行这个程序:
$ nice ./limits
CPU usage: User = 1.000000, System = 0.000000
Current priority = 10
Current FSIZE limit: soft = -1, hard = -1
Setting a 2K file size limit
File size limit exceeded
工作原理
limits 程序调用work函数来模拟通常的程序行为。他会执行一些计算并且产生一些输出,在这个例子中,大约向临时文件写入了150K的内容。他调用资源函数来查 看其优先级以及文件尺寸限制。在这个例子中,文件尺寸限制并没有设置,从而可以允许我们创建我们希望大小的文件。程序然后将其文件尺寸限制为2K,并且再 次尝试执行work函数。这一次work函数会失败,因为他不可以创建这样大的一个临时文件。
注意:也可以使用特定shell的ulimit命令来设置一个运行程序的限制。
在 这个例子中,错误信息‘Error writing to temporary file’ 也许并没有如我们所期望的那样显示出来。这是因为一些系统会超过资源限制时终止程序的执行。这是通过发送一个SIGXFSZ信号来做到的。我们将会在第 11章了解到更多的关于信号以及如何使用他们的内容。而其他的一些POSIX系统也许只是简单的使得超过限制的函数返回一个错误。
总结
在这一章,我们了解了Linux环境并且检测了程序运行的条件。我们讨论了命令行参数以及环境变量,这两者都可以用来改变程序的默认行为以及提供有用的程序选项。
我们也了解了一个程序如何使用库函数来操作日期与时间值,得到自身的信息,以及他所运行的用户和主机的信息。
Linux程序通常需要共享精确的资源,所以我们也讨论了这些程序如何确定与管理。
分享到:
相关推荐
这个社区版的Wrapper提供了基础的Java应用管理和监控功能,包括自动重启、日志记录、资源限制等。对于开发者和运维人员来说,这意味着可以更有效地管理和控制Java应用程序的生命周期。 在Linux环境下,Wrapper工具...
这是因为目标硬件平台可能没有足够的资源或者不具备运行完整编译环境的能力。`arm-vfp-linux-gnu-4.3.2`就是这样一个专为带有硬件浮点单元(VFP,Vector Floating Point)的ARM9处理器设计的交叉编译工具链,它允许...
此外,包装器还提供了日志记录、故障恢复、资源限制、权限管理等功能,提高了应用程序的健壮性和可维护性。 在部署Java应用时,首先要确保系统满足包装器的硬件和软件需求,然后配置`wrapper.conf`文件,指定Java...
在arm架构上部署MySQL 5.7.32可能面临的挑战包括性能优化、资源限制和兼容性问题。由于ARM处理器的异构计算特性,优化查询性能和内存使用至关重要。此外,由于资源有限,可能需要调整MySQL的配置参数以适应低内存...
- 避免目标平台资源限制:如果目标平台资源有限,无法直接在其上编译大型软件项目。 - 兼容性测试:在不同架构间进行兼容性测试,确保软件在 ARM Linux 上的正确运行。 6. **注意事项** - 依赖库:交叉编译时需...
《深入理解ARM-LINUX-ANDROIDEABI-4.9:构建NDK编译环境与CPU架构库》 在移动设备开发领域,Android NDK(Native Development Kit)扮演着至关重要的角色,它允许开发者使用C/C++编写底层代码,以实现高性能计算或...
在Linux环境中,FTPD服务器是网络服务的重要组成部分,为用户提供安全可靠的文件交换途径。 Linux FTPD 0.17 版本可能是该软件的一个稳定版本,"0.17"通常表示开发阶段,其中“0”表示主要版本号,而“17”表示次要...
标签“嵌入式”表明了这个话题与开发嵌入式系统紧密相关,涉及硬件限制、低功耗、实时性等特性。通过学习和使用这个交叉编译工具链,开发者能够为各种嵌入式设备,如嵌入式Linux系统、路由器、智能家居设备等,创建...
在嵌入式系统开发中,由于资源限制,通常在功能更强大的桌面或服务器系统上完成软件的编译工作,然后将编译结果部署到嵌入式设备上,这就是arm-linux-gnueabihf-gcc编译器的核心作用。 LinuxGCC,即GNU Compiler ...
这种技术常用于嵌入式系统开发,因为目标系统的资源限制可能不允许在其上运行编译器。ARM-Linux-GCC-3.3.2交叉编译器是专门用于在基于Linux的主机上为ARM架构的目标系统生成可执行文件的工具。 **二、下载与准备** ...
“ubuntu”标签确认了这个工具与Ubuntu操作系统的兼容性,意味着它在Ubuntu环境下可以顺利工作,包括安装、配置和使用。 在提供的文件列表中,“readme.txt”是一个常见的文本文件,通常包含有关如何安装、配置和...
Oracle Instant Client是Oracle公司提供的...在Linux环境中,Oracle Instant Client的使用涉及到系统管理、网络配置、编程接口等多个方面,理解并熟练掌握这些知识点对于开发和维护与Oracle数据库相关的应用至关重要。
Elasticsearch是一个开源的全文搜索引擎,它以其高效、可扩展和实时性著称。这个"最新版linux elasticsearch-7.17.1-...在Linux环境下部署时,需要注意安全配置、资源管理和集群维护,以确保系统的稳定和高效运行。
在Linux环境下,Elasticsearch提供了高效、可扩展的数据存储和检索解决方案,广泛应用于日志分析、实时监控、数据可视化等领域。7.16.2是Elasticsearch的一个稳定版本,对性能和功能进行了优化。 在Linux上安装...
1. **资源限制**:许多 ARM 设备(如嵌入式板卡)可能没有足够的资源(如内存、硬盘空间)来支持本地编译。 2. **效率**:在性能更强的主机上进行编译可以大幅提高速度,缩短开发周期。 3. **隔离开发环境**:交叉...
- **资源限制**:目标平台可能资源有限,无法直接在其上进行编译操作。 - **效率考虑**:目标平台可能速度较慢,而在高性能的主机上编译可以显著提高效率。 - **安全性**:有些目标平台不允许直接在上面进行编译,以...
标签"jdk-8u144-linux-"揭示了这个压缩包与JDK 8u144在Linux平台有关,可能是为了方便搜索和分类。 压缩包内的文件名列表如下: 1. "jdk-8u144-linux-x64.rpm":如前所述,这是一个RPM格式的JDK安装包,用户可以...
Linux JDK 17.0.3.1 是Java开发工具包在Linux平台上的一个64位版本,适用于最新的操作系统环境。这个版本包含了JVM(Java虚拟机)和一系列开发工具,如javac编译器、jar打包工具以及javadoc文档生成器等。JDK是Java...
在Linux环境中,安装OpenOffice可以提供一个与Microsoft Office兼容的免费办公解决方案,特别适合那些不希望受到专有软件许可限制的个人和组织。它支持打开和保存各种Office文档格式,包括.doc、.docx、.xls、.xlsx...
在Linux环境中部署和使用Elasticsearch 7.9.2是许多IT专业人士的重要任务。这个版本提供了许多性能改进和新特性,以增强搜索、分析和监控能力。 1. **安装步骤**: - 首先,下载`elasticsearch-7.9.2-linux-x86_64...