百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 优雅编程 > 正文

Linux下如何计算函数的执行时间?实例代码直接拿去用

sinye56 2024-11-09 13:34 6 浏览 0 评论

关于时间操作的文章,大家还可以可以关注我之前的一篇文章《C语言操作时间函数time.ctime,实现定时执行某个任务小例子

0、问题描述

粉丝想计算一个函数的执行时间。



一、问题分析

函数的执行时间的统计在嵌入式系统中会被频繁地用到,知识点很重要。 本文从两个方面来讨论类似的问题:

  1. 程序内计算一个函数的执行时间
  2. 计算一个程序的执行时间

二、程序内如何计算一个函数的执行时间?

1. 思路

我们在执行函数前后分别记录下时间戳,然后计算两个时间戳的差值即可。

我们需要借助函数clock_gettime来实现这个功能。 看下该函数的定义:


#include <time.h>

int clock_gettime(clockid_t clk_id, struct timespec* tp);

可以根据需要,获取不同要求的精确时间

参数:
clk_id : 
 检索和设置的clk_id指定的时钟时间。
 CLOCK_REALTIME:系统实时时间,随系统实时时间改变而改变,即从UTC1970-1-1 0:0:0开始计时,中间时刻如果系统时间被用户改成其他,则对应的时间相应改变
  CLOCK_MONOTONIC:从系统启动这一刻起开始计时,不受系统时间被用户改变的影响
  CLOCK_PROCESS_CPUTIME_ID:本进程到当前代码系统CPU花费的时间
  CLOCK_THREAD_CPUTIME_ID:本线程到当前代码系统CPU花费的时间
tp : 
 获取的时间戳会存放到该结构体变量中
 struct timespec
 {
         time_t tv_sec; /* 秒*/
         long tv_nsec; /* 纳秒*/
 };
返回值:
 成功  0
 失败 -1  ,同时errno会被赋值

因为我们希望计算执行某个函数的时间,所以我们第一个参数选择CLOCK_MONOTONIC

2. 实例1

我们先来实现一个简单的程序:

  1 #include <stdio.h>                                                               
  2 #include <stdlib.h>
  3 #include <stdint.h>
  4 #include <time.h>
  5 #include <sys/time.h>
  6 #include <sys/stat.h>
  7 #include <sys/types.h>
  8 #include <unistd.h>
  9 #include <string.h>
 10 
 11 int main()
 12 {
 13     int rc;
 14     struct timespec ts_start, ts_end;
 15     
 16     //start time before call function
 17     rc = clock_gettime(CLOCK_MONOTONIC, &ts_start);
 18     
 19     printf("you can call your function here\n");
 20     
 21     //end time before call function 
 22     rc = clock_gettime(CLOCK_MONOTONIC, &ts_end);
 23     
 24     printf("CLOCK_MONOTONIC reports %ld.%09ld seconds\n",
 25             ts_end.tv_sec - ts_start.tv_sec, ts_end.tv_nsec - ts_start.tv_nsec);
 26 }

19行 我们可以将自己要执行的函数放置在此处。

编译

gcc runtime.c -lrt

注意需要增加动态链接库lrt,函数clock_gettime()定义于该库中。

执行结果如下:

root@ubuntu:/home/peng/zhh# ./a.out 
you can call your function here
CLOCK_MONOTONIC reports 0.000013689 seconds

3. 实例2-更完善的一个例子

第一个实例比较简单,实际在应用中,尤其是在网络通信中,经常需要计算收发数据包的总共时间,以网络的速率。 现在我们增加功能如下:

  1. 检查执行函数前后的时间戳合法性,因为有时候记录的时间会比较长,会有数据溢出等问题
  2. 循环累加总共执行时间,计算总共执行时间,然后根据执行次数计算平均执行时间

a) 检查时间合法性

  1. timespec_check()
static int timespec_check(struct timespec *t)
{
 if((t->tv_nsec <0 ) || (t->tv_nsec >= 1000000000))
  return -1;

 return 0;
}
功能:
 该函数检查时间戳的成员tv_nsec,该值不能小于0,也不能大于1000000000
参数:
 t 时间戳
返回值
 成功返回 0
 非法返回-1
  1. timespec_sub()
static void timespec_sub(struct timespec *t1,  struct timespec *t2)
{
 if (timespec_check(t1) < 0) {
  fprintf(stderr, "invalid time #1: %lld.%.9ld.\n",
   (long long) t1->tv_sec,t1->tv_nsec);
  return;
 }
 if (timespec_check(t2) < 0) {
  fprintf(stderr, "invalid time #2: %lld.%.9ld.\n",
   (long long) t2->tv_sec,t2->tv_nsec);
  return;
 } 

 t1->tv_sec -= t2->tv_sec;
 t1->tv_nsec -= t2->tv_nsec;
 if (t1->tv_nsec >= 1000000000)
 {//tv_nsec 超过1000000000,秒需要加1
  t1->tv_sec++;
  t1->tv_nsec -= 1000000000;
 }
 else if (t1->tv_nsec < 0)
 {//tv_nsec 小于0,秒需要减1
  t1->tv_sec--;
  t1->tv_nsec += 1000000000;
 }
}
功能:
 该函数首先检查参数t1、t2合法性,然后用t1的时间减去t2的时间,并把结果存放到t1
参数:
 t1:对应函数执行执行结束的时间
 t2:对应函数执行之前的时间
返回值:
 无

b) 实现

  1 #include <stdio.h> 
  2 #include <stdlib.h>
  3 #include <stdint.h>
  4 #include <time.h>
  5 #include <sys/time.h>
  6 #include <sys/stat.h>
  7 #include <sys/types.h>
  8 #include <unistd.h>
  9 #include <string.h>
 10 
 11 
 12 static int timespec_check(struct timespec *t)
 13 {
 14     if((t->tv_nsec <0 ) || (t->tv_nsec >= 1000000000))
 15         return -1;
 16 
 17     return 0;
 18 }
 19 
 20 static void timespec_sub(struct timespec *t1,  struct timespec *t2)
 21 {
 22     if (timespec_check(t1) < 0) {
 23         fprintf(stderr, "invalid time #1: %lld.%.9ld.\n",
 24             (long long) t1->tv_sec,t1->tv_nsec);
 25         return;
 26     }
 27     if (timespec_check(t2) < 0) {
 28         fprintf(stderr, "invalid time #2: %lld.%.9ld.\n",
 29             (long long) t2->tv_sec,t2->tv_nsec);
 30         return;
 31     }
 32 
 33     t1->tv_sec -= t2->tv_sec;
 34     t1->tv_nsec -= t2->tv_nsec;
 35     if (t1->tv_nsec >= 1000000000)
 36     {
 37         t1->tv_sec++;
 38         t1->tv_nsec -= 1000000000;
 39     }
 40     else if (t1->tv_nsec < 0)
 41     {
 42         t1->tv_sec--;
 43         t1->tv_nsec += 1000000000;
 44     }
 45 }
 46 
 47 int main()
 48 {
 49     int rc;
 50     int count = 10;
 51     long t_time_n = 0;  //nano secend
 52     long t_time_s = 0;  //secnd
 53     struct timespec ts_start, ts_end;
 54 
 55 
 56     while (count--) {
 57 
 58         rc = clock_gettime(CLOCK_MONOTONIC, &ts_start);
 59         usleep(200);
 60 
 61         rc = clock_gettime(CLOCK_MONOTONIC, &ts_end);                                                         
 62 
 63         timespec_sub(&ts_end, &ts_start);
 64         t_time_n += ts_end.tv_nsec;
 65         t_time_s += ts_end.tv_sec;
 66 
 67         #if 0
 68         printf("CLOCK_MONOTONIC reports %ld.%09ld seconds\n", 
 69                 ts_end.tv_sec, ts_end.tv_nsec);     
 70         #endif
 71     }
 72     printf("** Total time %lds + %ld nsec\n",t_time_s,t_time_n);
 73 }

编译执行如下:

root@ubuntu:/home/peng/zhh# ./a.out 
** Total time 0s + 9636103 nsec

三、计算程序的执行时间

有时候我们还想知道执行某个程序需要多少时间,我们可以借助命令time。

1. 命令time

Linux time命令的用途,在于量测特定指令执行时所需消耗的时间及系统资源等信息。

CPU资源的统计包括实际使用时间(real time)、用户态使用时间(the process spent in user mode)、内核态使用时间(the process spent in kernel mode)。

2. 语法

time [options] COMMAND [arguments]

3. 例1

1. root@ubuntu:/home/peng/zhh# time date  
2. Tue Feb 23 03:44:27 PST 2021
3. 
4. real    0m0.001s
5. user    0m0.000s
6. sys     0m0.000s
  • 在以上实例中,执行命令"time date"(见第1行)。
  • 系统先执行命令"date",第2行为命令"date"的执行结果。
  • 第3-6行为执行命令"date"的时间统计结果,其中第4行"real"为实际时间,第5行"user"为用户CPU时间,第6行"sys"为系统CPU时间。 以上三种时间的显示格式均为MMmNN[.FFF]s。

5. 例2

我们也可以测试上一章我们编写的程序:

root@ubuntu:/home/peng/zhh# time ./a.out 
** Total time 0s + 9649603 nsec, avg_time = -9649603.000000 

real 0m0.010s
user 0m0.000s
sys     0m0.000s

下面我们将59行代码中的usleep(200)修改成sleep(1) 重新编译执行,10秒回打印如下执行结果:

root@ubuntu:/home/peng/zhh# time ./a.out 
** Total time 10s + 8178015 nsec

real 0m10.009s
user 0m0.000s
sys  0m0.000s

大家可以根据我的代码,方便地将该功能移植到自己的项目中。

想学习更多的嵌入式知识和技巧,请关注一口君:一口Linux

相关推荐

linux安装FTP

1、在nkftp目录下安装ftp,进入到nkftp里面[root@localhostbin]#cd/data/nkftp执行安装命令:[root@localhostnkftp]#rpm-i...

LINUX下搭建FTP服务器

FTP服务器介绍FTP是FileTransferProtocol(文件传输协议)的英文简称,而中文简称为“文传协议”。用于Internet上的控制文件的双向传输。同时,它也是一个应用程序(App...

Linux下如何进行FTP设置

目录:一、Redhat/CentOS安装vsftp软件二、Ubuntu/Debian安装vsftp软件一、Redhat/CentOS安装vsftp软件1.更新yum源yumupdate-y2.安...

推荐使用集串口 SSH远程登录和FTP传输三合一工具MobaXterm

来源:百问网作者:韦东山本文字数:1216,阅读时长:4分钟在以前的资料里,串口和SSH远程登使用SecureCRT,window与ubuntu数据传输使用filezilla,窗口切换来切换去,麻烦也...

如何搭建FTP服务器(Linux系统)

上次说了Windows操作系统下搭建的FTP服务器,那有朋友问我,说买的XX轻量应用服务器都是属于Linux的操作系统,我该如何为搭建FTP服务器呢?...

Linux 命令 ncftp(文件传输)——想玩转linux就请一直看下去

我是IT悟道,点击右上方“关注”,每天分享IT、科技、数码方面的干货。Linuxncftp命令...

如何用 ftp 实现一键上传

简介ftp是Internet标准文件传输协议的用户界面,它允许用户与远程网络站点之间传输文件...

Linux安装ftp

1安装vsftpd组件安装完后,有/etc/vsftpd/vsftpd.conf文件,是vsftp的配置文件。[root@bogon~]#yum-yinstallvsftpd2添加一个...

一天一点点:linux - ftp命令

linuxftp命令设置文件系统相关功能。FTP是ARPANet的标准文件传输协议,该网络就是现今Internet的前身。语法ftp[-dignv][主机名称或IP地址]参数:-d详细显示指令执...

Centos 7 搭建FTP

目录安装软件以及启动服务添加防火墙规则关闭selinuxftp配置常用常用参数详解特殊参数配置文件没有的参数也可以添加到配置中1.安装软件以及启动服务yuminstall-yvsftpdsys...

【Linux】Linux中ftp命令,没有你想的那么简单

本文介绍了Linux中FTP命令的基本用法,包括连接与登录远程服务器,以及解析了FTP协议中五个最常用的操作命令的使用和解析过程。同时,提供了一个包含常用FTP操作命令的表格,供读者参考。通过熟练掌握...

linux 命令行操作ftp

以下是linuxftp命令参数的详解。FTP>!从ftp子系统退出到外壳?FTP>?显示ftp命令说明??和help相同?格式:?[command]说明:[com...

多学习才能多赚钱之:linux如何使用ftp

linux如何使用ftp步骤1:建立FTP连接想要连接FTP服务器,在命令上中先输入ftp然后空格跟上FTP服务器的域名'domain.com'或者IP地址例如:ftpdom...

linux常用网络操作方法:ftp命令使用方法

常用网络操作方法Linux提供了一组强有力的网络命令来为用户服务,这些工具能够帮助用户登录到远程计算机上、传输文件和执行远程命令等。本节介绍下列几个常用的有关网络操作的命令:ftp传输文件tel...

Linux 5.15有望合并Memory Folios方案 内核构建速度可提升7%

甲骨文公司的长期内核开发人员MatthewWilcox已经研究了“内存对开区”概念相当长的一段时间,这可以改善Linux的内存管理,使其具有更大的效率。例如,使用内存对开的基准测试表明,内核的构建速...

取消回复欢迎 发表评论: