7. 系统信息与系统资源

news2025/1/23 9:26:55

7. 系统信息与系统资源

  • 1. 系统信息
    • 1.1 系统标识 uname()
    • 1.2 sysinfo()
    • 1.3 gethostname()
    • 1.4 sysconf()
  • 2. 时间、日期
    • 2.1 Linux 系统中的时间
      • 2.1.1 Linux 怎么记录时间
      • 2.1.2 jiffies 的引入
    • 2.2 获取时间 time/gettimeofday
      • 2.2.1 time()
      • 2.2.2 gettimeofday()
    • 2.3 时间转换函数
      • 2.3.1 ctime()
      • 2.3.2 localtime()
      • 2.3.3 gmtime()
      • 2.3.4 mktime()
      • 2.3.5 asctime()
      • 2.3.6 strftime()
    • 2.4 设置时间 settimeofday
  • 3. 进程时间
    • 3.1 times()
    • 3.2 clock()
  • 4. 产生随机数
    • 4.1 rand()
    • 4.2 srand()
  • 5. 休眠
    • 5.1 秒级休眠 sleep()
    • 5.2 微秒级休眠 usleep()
    • 5.3 高进度休眠 nanosleep()
  • 6. 申请堆内存
    • 6.1 在堆上分配内存
    • 6.2 在堆上分配对齐内存
  • 7. proc 文件系统
    • 7.1 使用cat命令读取
    • 7.2 使用read读取

1. 系统信息

1.1 系统标识 uname()

系统调用 uname() 用于获取当前操作系统内核的名称和信息。调用时先创建一个结构体类型变量,然后传入地址即可

#include <sys/utsname.h>
int uname(struct utsname *buf);

在这里插入图片描述

1.2 sysinfo()

可用于获取一些系统统计信息

#include <sys/sysinfo.h>
int sysinfo(const sysinfo *info);

在这里插入图片描述

1.3 gethostname()

可以单独获取 Linux 系统主机名

#include <unistd.h>
int gethostname(char *name, size_t len);
// 参数:用于存放主机名的缓冲区以及缓冲区大小

1.4 sysconf()

可在运行时获取系统的一些配置信息

#include <unistd.h>
long sysconf(int name);
/* name:指定要获取哪个配置信息,可以使用 man 手册查看
 * _SC_ARG_MAX:exec族函数的参数的最大长度,这里先不用了解exec族
 * _SC_CHILD_MAX:每个用户的最大并发进程数
 * _SC_HOST_NAME_MAX:主机名的最大长度
 * _SC_LOGIN_NAME_MAX:登录名的最大长度
 * _SC_CLK_TCK:每秒时钟滴答数,也就是系统节拍率
 * _SC_OPEN_MAX:一个进程可以打开的最大文件数
 * _SC_PAGESIZE:系统页大小
 * _SC_TTY_NAME_MAX:终端设备名的最大长度
 * ...
 * /

2. 时间、日期

2.1 Linux 系统中的时间

操作系统中有两个时钟,一个是系统时钟,一个是实时时钟,也叫 RTC。系统时钟在系统启动之后由内核来维护,使用 date 命令获取到的就是系统时钟,在系统关机情况下不存在;实时时钟一般由 RTC 时钟芯片提供,该芯片有相应的电池,可以保证在系统关机的情况下依旧能够继续工作。

2.1.1 Linux 怎么记录时间

在启动之后先读取 RTC 作为系统时钟的初始值,之后一直维护系统时钟,然后就不会再对 RTC 进行读取操作。系统关机后,会将系统时钟写入到 RTC,进行同步操作。

2.1.2 jiffies 的引入

内核定义的一个全局变量,使用该变量来记录系统从启动以来的系统节拍数,所以这个变量用来记录以系统节拍时间为单位的时间长度,Linux 内核在编译配置时定义了一个节拍时间,使用节拍率(一秒钟多少个节拍数)来表示,譬如节拍率为 200Hz,就表示一秒钟 200 个节拍,节拍时间为 1s/200。节拍率越低,每一个系统节拍的时间就越短,意味着 jiffies 记录的时间精度越高。 但是高节拍率会导致系统中断的产生更加频繁,会加剧系统的负担,一般默认情况下都是采用 100Hz 作为系统节拍率。
内核其实通过 jiffies 来维护系统时钟,在系统开机时会设置一个初始值,上面所说的读取 RTC就是用来初始化 jiffies 变量的。当我们需要获取到系统当前时间点时,就可以使用 jiffies 变量去计算。

2.2 获取时间 time/gettimeofday

2.2.1 time()

获取当前时间,以秒为单位,返回值是从 1970-01-01 00:00:00 +0000(UTC) 以来的秒数

#include <time.h>
time_t time(time_t *tloc);
// 如果 tloc 不是空,则返回值存储在 tloc 指向的内存中

2.2.2 gettimeofday()

time 函数只能精确到秒,而这个函数可以精确到微秒

#include <sys/time.h>
int gettimeofday(struct timeval *tv, struct timezone *tz);
// tz 直接设置为NULL即可
// 获取到的时间存储在tv中,这个结构体的有两个变量,得到的时间就是 tv_sec+tv_usec。tv_sec 的值和time获取的值相同

2.3 时间转换函数

2.3.1 ctime()

将时间转换为字符串形式

#include <time.h>
char *ctime(const time_t *timep);
char *ctime_r(const time_t *timep, char *buf);
// 成功就返回指向得到的字符串的指针,失败就返回NULl
// 但是 ctime 是一个不可重入的函数,推荐使用 ctime_r,是一个可重入函数,buf就是用于存放字符串的缓冲区地址。可重入后面再讲,_r 就一般表示可重入,也就是如果成功就返回buf
#include <stdio.h>
#include <stdlib.h>
#include <time.h>
#include <iostream>
using namespace std;
int main()
{
	char str[100]={0};
	time_t tm;
	tm=time(NULL);
	ctimr_t(&tm,str);
	cout << str<<endl;
	return 0;
}

2.3.2 localtime()

讲时间转换为一个 struct tm 结构体所表示的时间,对应的是本地时间

#include <time.h>
struct tm *localtime(const time_t *timep);
struct tm *localtime_r(const time_t *timep, struct tm *result);

在这里插入图片描述

2.3.3 gmtime()

将 time_t 转换为 struct tm 结构体表示的时间,但是是 UTC 时间,不是本地时间

#include <time.h>
struct tm *gmtime(const time_t *timep);
struct tm *gmtime_r(const time_t *timep, struct tm *result);

2.3.4 mktime()

将 struct tm 结构体表示的时间转换为 time_t 时间

#include <time.h>
time_t mktime(struct tm *tm);

2.3.5 asctime()

和 ctime() 一样,但是可以将时间分解转换为固定格式字符串。

#include <time.h>
char *asctime(const struct tm *tm);
char *asctime_r(const struct tm *tm, char *buf);

2.3.6 strftime()

可以根据自己的需要设定格式

#include <time.h>
size_t strftime(char *s, size_t max, const char *format, const struct tm *tm);
/* s:存放生成的字符串
 * max:字符串的最大字节数
 * format:格式
 * tm:指向结构体的指针 
 * /

在这里插入图片描述
这些格式字段可以任意组合,比如:"%Y-%m-%d %H:%M:%S<%p> %B %A"可以输出"2021-01-14 16:30:25<PM> January Thursday"

2.4 设置时间 settimeofday

设置系统本地时间,而且只有超级用户才能设置时间

#include <sys/time.h>
int settimeofday(const struct timeval *tv, const struct timezone *tz);
// tz 目前已经废弃,直接设置NULL即可

3. 进程时间

进程时间指的是进程从创建后,也就是程序运行后,到目前为止这段时间内使用 CPU 资源的时间总数。内核将 CPU 时间,也就是进程时间分为两部分:

  1. 用户 CPU 时间:进程在用户空间下运行所花费的 CPU 时间,也成为虚拟时间
  2. 系统 CPU 时间:进程在内核态下运行所花费的时间。这是内核执行系统调用或代表进程执行的其他任务所花费的时间。
    一般来说,进程时间指的是这两个时间总和
    注意: 进程时间不等于程序的整个生命周期所消耗的时间,如果进程一直处于休眠状态(进程被挂起、不会得到系统调度),那么它并不会使用 CPU 资源,所以休眠这段时间不计算在进程时间中。

3.1 times()

获取当前进程时间

#include <sys/times.h>
clock_t times(struct tms *buf);
// 返回值类型实际就是long类型,成功时返回从过去任意一个时间点所经过的时钟滴答数,就是系统节拍数,将节拍数/节拍率就是秒数,返回值可能会溢出,调用失败返回-1

在这里插入图片描述

// 计算一段程序耗费的进程时间和总的时间
#include <stdio.h>
#include <stdlib.h>
#include <sys/times.h>
#include <unistd.h>
#include <iostream>
using namespace std;
int main(int argc, char *argv[])
{
	struct tms _start;
	struct tms _end;
	clock_t start;
	clock_t end;
	long tck;
	int i,j;

	// 获取系统节拍率
	tck = sysconf(_SC_CLK_TCK);

	// 开始时间
	start = times(&_start);
	
	for(i=0;i<20000;i++);
	{
		for(j=0;j<20000;j++);
		{}
	}
	sleep(1);

	end = times(&_end);
	cout << "时间总和: << (end-start)/double(tck) << endl;// 2.910000秒,从起点到终点所经过的时间,并不是进程时间
	cout << "用户时间: << (_end.tms_utime-_start.tms_utime)/double(tck) << endl;// 1.900000秒
	cout << "系统时间: << (_end.tms_stime-_start.tms_stime)/double(tck) << endl;// 0.000000秒
}

3.2 clock()

更简单的函数获取进程时间,返回值描述了进程使用的总的 CPU 时间,也就是进程时间。

#include <time.h>
clock_t clock();
// 返回值就是到目前为止程序的进程时间,并不是系统节拍数,如果想要获取秒数,需要除 CLOCKS_PER_SEC,失败返回-1。但是此函数不能获取到单独的用户CPU时间和系统CPU时间

4. 产生随机数

4.1 rand()

#include <stdlib.h>
int rand();
// 这个函数每一次运行程序获得的随机数都是相同的,需要使用srand函数设置随机数种子。rand将1作为随机数种子

4.2 srand()

#include <stdlib.h>
void srand(unsigned int seed);

一般都是将当前时间作为种子传递给seed,因为时间是一直在变化的。

// 生成一组随机数
#include <stdio.h>
#include <stdlib.h>
#include <time.h>

int main()
{
	int random_number_arr[8];
	int count;
	srand(time(NULL));
	for(count = 0; count<8;count++)
	{
		random_number_arr[count] = rand()%100;
	}
	for(count = 0; count<8;count++)
	{
		cout << random_number_arr[count] << " ";
	}
	cout << endl;
	return 0;
}

5. 休眠

5.1 秒级休眠 sleep()

#include <unistd.h>
unsigned int sleep(unsigned int seconds);
// 如果休眠时长等于seconds,就返回0;如果被信号中断,返回剩余的秒数

5.2 微秒级休眠 usleep()

#include <unistd.h>
int usleep(useconds_t usec);
// 成功返回0,失败返回-1

5.3 高进度休眠 nanosleep()

#include <time.h>
int nanosleep(const struct timespec *req, struct timespec *rem);
// req:设置休眠时长,可精确到纳秒级
// rem:可以设置NULL
// 如果成功休眠设定的时长,返回0;如果被信号中断或错误,返回-1,并将剩余时间记录在rem中,如果为NULL,表示不接收剩余时间

6. 申请堆内存

6.1 在堆上分配内存

#include <stdlib.h>
void *malloc(size_t size);	// 在使用时通常需要强制类型转换,该函数不会初始化内存空间
void *calloc(size_t nmemb, size_t size);	// 分配nmemb个单位长度为size的连续空间,并初始化为0,总的大小是nmemb*size
void *realloc(void *ptr, size_t size); // 可以在原有的空间上扩容
void free(void *ptr);		// 释放堆内存空间

当进程终止时,会自动释放申请的空间,有时,因为加入了多次的 free() 调用,可能会消耗掉大量的 CPU 时间。但是依旧建议手动释放,否则容易发生内存泄漏。

6.2 在堆上分配对齐内存

#include <stdlib.h>
int posix_memalign(void **memptr, size_t alignment, size_t size);
void *aligned_alloc(size_t alignment, size_t size);
void *valloc(size_t size);// 不建议使用

#include <malloc.h>
void *memalign(size_t alignment, size_t size); // 不建议使用
void *pvalloc(size_t size);

posix_memalign: 在堆上分配 size 个字节大小的对齐内存空间,将 *memptr 指向分配的空间。alignment 表示对其字节数,必须是 2 的幂次方,也是指针大小的整数倍。size 是分配的内存大小,如果为0,指向的空间值为NULL
aligned_alloc: 分配 size 个字节大小的内存空间,返回指向该空间的指针
valloc: 以 pagesize 页大小作为对齐的长度,可以通过 getpagesize() 获取页大小

7. proc 文件系统

是一个虚拟文件系统,它以文件系统的方式为应用层访问系统内核数据提供了接口,用户和应用程序可以通过 proc 文件系统得到系统信息和进程相关信息,对 proc 文件系统的读写作为与内核进行通信的一种手段。但是与普通文件不同的是,proc 文件系统是动态创建的,文件本身不存在于磁盘当中,只存在于内存中。该文件系统挂载在 /proc 目录下

7.1 使用cat命令读取

比如查看内核版本信息cat /proc/version

7.2 使用read读取

将你需要直到的信息读取到文件中

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1284185.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

MySQL笔记-第02章_MySQL环境搭建

视频链接&#xff1a;【MySQL数据库入门到大牛&#xff0c;mysql安装到优化&#xff0c;百科全书级&#xff0c;全网天花板】 文章目录 第02章_MySQL环境搭建1. MySQL的卸载步骤1&#xff1a;停止MySQL服务步骤2&#xff1a;软件的卸载步骤3&#xff1a;残余文件的清理步骤4&am…

【Cesium】实现卷帘对比

一、创建滑块 <style>import url(../Build/CesiumUnminified/Widgets/widgets.css);import url(./Sandcastle/templates/bucket.css);#slider {position: absolute;left: 50%;top: 0px;background-color: #d3d3d3;width: 5px;height: 100%;z-index: 9999;}#slider:hover…

leetcode 876.链表的中间结点

补充上次的环形链表没细讲的快慢指针&#xff08;这三道题现在可以连起来看&#xff09;&#xff0c;希望对你做题思路有帮助 876.链表的中间结点 题目 给你单链表的头结点 head &#xff0c;请你找出并返回链表的中间结点。 如果有两个中间结点&#xff0c;则返回第二个中间结…

机械学习概述

1.1 什么是机器学习 1.1.1 机器的学习能力 1997年5月11日&#xff0c;一台名为“深蓝”的超级电脑战胜国际象棋名家卡斯帕罗夫。20世纪末的一场人机大战终于以计算机的微弱优势取胜。 2016年3月&#xff0c;阿尔法围棋程序&#xff08;AlphaGo&#xff09;挑战世界围棋冠军李…

windows11 调整鼠标灵敏度方法

首先 我们打开电脑设置 或者在 此电脑/此计算机/我的电脑 右击选择属性 然后 有的电脑 左侧菜单中 直接就有 设备 然后在设备中直接就可以找到 鼠标 选项 调整光标速度即可 如果操作系统和我的一样 可以直接搜索鼠标 然后 选择 鼠标设置 然后 调整上面的鼠标指针速度即可

锁策略之干货分享,确定不进来看看吗?️️️

&#x1f308;&#x1f308;&#x1f308;今天给大家分享的是关于锁策略方面的基础知识。 清风的CSDN博客 &#x1f6e9;️&#x1f6e9;️&#x1f6e9;️希望我的文章能对你有所帮助&#xff0c;有不足的地方还请各位看官多多指教&#xff0c;大家一起学习交流&#xff01; …

基于stm32的LCD1602与无线蓝牙温湿度显示

这一篇博客是为了实现温湿度的显示&#xff0c;温湿度传感器将数据穿给单片机&#xff0c;单片机又把数据送给LCD1602和蓝牙&#xff0c;让温度和湿度可以再LCD1602显示屏和手机上显示&#xff0c;它的执行逻辑和C51那里基本一样&#xff0c;就是要修改程序&#xff0c;在程序上…

选择排序、插入排序、希尔排序

1.选择排序 算法描述 将数组分为两个子集&#xff0c;排序的和未排序的&#xff0c;每一轮从未排序的子集中选出最小的元素&#xff0c;放入排序子集 重复以上步骤&#xff0c;直到整个数组有序 选择排序呢&#xff0c;就是首先在循环中&#xff0c;找到数组中最小的元素。在…

基恩士软件的基本操作(六,KV脚本的使用)

目录 什么是KV脚本&#xff1f; KV脚本有什么用&#xff1f; 怎么使用KV脚本&#xff08;脚本不能与梯形图并联使用&#xff09;&#xff1f; 插入框脚本&#xff08;CtrlB&#xff09; 插入域脚本&#xff08;CtrlR&#xff09; 区别 脚本语句&#xff08;.T是字符串类…

详解原生Spring当中的事务

&#x1f609;&#x1f609; 学习交流群&#xff1a; ✅✅1&#xff1a;这是孙哥suns给大家的福利&#xff01; ✨✨2&#xff1a;我们免费分享Netty、Dubbo、k8s、Mybatis、Spring...应用和源码级别的视频资料 &#x1f96d;&#x1f96d;3&#xff1a;QQ群&#xff1a;583783…

Redis实战篇笔记(最终篇)

Redis实战篇笔记&#xff08;七&#xff09; 文章目录 Redis实战篇笔记&#xff08;七&#xff09;前言达人探店发布和查看探店笔记点赞点赞排行榜 好友关注关注和取关共同关注关注推送关注推荐的实现 总结 前言 本系列文章是Redis实战篇笔记的最后一篇&#xff0c;那么到这里…

界面组件DevExpress Reporting v23.1新版亮点 - UX功能增强

DevExpress Reporting是.NET Framework下功能完善的报表平台&#xff0c;它附带了易于使用的Visual Studio报表设计器和丰富的报表控件集&#xff0c;包括数据透视表、图表&#xff0c;因此您可以构建无与伦比、信息清晰的报表 界面组件DevExpress Reporting v23.1已于前段时间…

探讨Unity中的动画融合技术(BlendTree)

动画在游戏和虚拟现实应用中扮演着关键的角色&#xff0c;而动画融合技术则是使角色动作更加流畅和逼真的核心。在Unity引擎中&#xff0c;我们可以使用动画混合树&#xff08;Blend Trees&#xff09;来实现这一目标。本篇技术博客将深入讨论动画融合技术的实现原理、在Unity中…

C++ 指针详解

目录 一、指针概述 指针的定义 指针的大小 指针的解引用 野指针 指针未初始化 指针越界访问 指针运算 二级指针 指针与数组 二、字符指针 三、指针数组 四、数组指针 函数指针 函数指针数组 指向函数指针数组的指针 回调函数 指针与数组 一维数组 字符数组…

FreeRTOS调度器启动过程分析

目录 引出思考 vTaskStartScheduler()启动任务调度器 xPortStartScheduler()函数 FreeRTOS启动第一个任务 vPortSVCHandler()函数 总结 引出思考 首先想象一下如何启动第一个任务&#xff1f; 假设我们要启动的第一个任务是任务A&#xff0c;那么就需要将任务A的寄存器值…

腾讯云手动下发指令到设备-用于设备调试

打开腾讯云API Explorer&#xff0c;Publish Msg https://console.cloud.tencent.com/api/explorer?Productiotcloud&Version2021-04-08&ActionPublishMessagehttps://console.cloud.tencent.com/api/explorer?Productiotcloud&Version2021-04-08&ActionPub…

【模电】设置静态工作点的必要性

设置静态工作点的必要性 静态工作点为什么要设置静态工作点 静态工作点 在放大电路中&#xff0c;当有信号输入时&#xff0c;交流量与直流量共存。将输入信号为零、即直流电源单独作用时晶体管的基极电流 I B I\tiny B IB、集电极电流 I C I\tiny C IC、b - e间电压 U B E U\t…

语义分割网络FCN

语义分割是一种像素级的分类&#xff0c;输出是与输入图像大小相同的分割图&#xff0c;输出图像的每个像素对应输入图像每个像素的类别&#xff0c;每一个像素点的灰度值都是代表当前像素点属于该类的概率。 语义分割任务需要解决的是如何把定位和分类这两个问题一起解决&…

佛罗里达大学利用神经网络,解密 GPCR-G 蛋白偶联选择性

内容一览&#xff1a;G 蛋白偶联受体 (GPCRs) 是一种将细胞膜外的刺激&#xff0c;传递到细胞膜内的跨膜蛋白&#xff0c;广泛参与到人体生理活动当中。近日&#xff0c;佛罗里达大学的研究者测定了 GPCRs 和 G 蛋白的结合选择性&#xff0c;并开发了预测二者选择性的算法&…

kubernetes监控GPA安装部署

本文在于指导如何对k8s的监控GPA(Grafana&#xff0c;prometheus以及alertmanager)进行安装部署。 1. 介绍 Prometheus 在真正部署Prometheus之前&#xff0c;应了解一下Prometheus的各个组件之间的关系及作用&#xff1a; 1&#xff09;MertricServer&#xff1a;是k8s集群…