这篇文章主要介绍Linux下时间处理的相关函数与操作。 比如: 系统时间设置,读取、RTC时间设置,读取、时间单位转换、延时函数、闹钟信号等等。
Linux下存在两种时间: 1. 系统时间,2. RTC时间
系统时间是每次操作系统启动之后,从RTC驱动里读取进行设置的,一般只会在系统上电启动的时候自动(启动脚本)同步一次,后续用户也可以通过特定的命令再次同步;在系统界面上看到的时间就是系统时间;系统时间每次系统关机之后就会丢失,需要每次上电从RTC驱动里获取。
系统时间设置的方法如下:需要有管理员权限
[wbyq@wbyq linux_c]$ date -s "2020-10-12 9:28:20"
date: 无法设置日期: 不允许的操作
2020年 10月 12日 星期一 09:28:20 CST
[wbyq@wbyq linux_c]$ sudo date -s "2020-10-12 9:28:20"
[sudo] password for wbyq:
2020年 10月 12日 星期一 09:28:20 CST
[wbyq@wbyq linux_c]$
RTC时间掉电不会停止运行,电源是后备电源单独供给的;可以一直运行,方便给系统提供准确的时间。
RTC时间读取与设置方法:需要有管理员权限
hwclock -r 显示RTC时间 (读取RTC时间显示)
hwclock -w 设置RTC时间 (将系统时间传递给RTC驱动,设置RTC的驱动时间)
hwclock -s 设置系统时间(将RTC时间读取出来设置给系统时间)
也可以通过代码实现:
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <sys/ioctl.h>
#include <linux/rtc.h>
/*
RTC_SET_TIME
RTC_RD_TIME
*/
struct rtc_time time;
int main(int argc,char **argv)
{
if(argc!=2)
{
printf("./app /dev/rtcX\n");
return 0;
}
//1.打开设备文件
int fd=open(argv[1],2);
if(fd<2)
{
printf("%s 设备文件打开失败.\n",argv[1]);
return 0;
}
//2.获取RTC驱动的时间
ioctl(fd,RTC_RD_TIME,&time);
printf("应用层读取的时间: %d-%d-%d %d:%d:%d\n",
time.tm_year+1900,
time.tm_mon+1,
time.tm_mday,
time.tm_hour,
time.tm_min,
time.tm_sec);
//3.设置RTC驱动的时间
time.tm_year=2021-1900;
time.tm_mon=10-1;
time.tm_mday=1;
time.tm_hour=11;
time.tm_min=10;
time.tm_sec=20;
ioctl(fd,RTC_SET_TIME,&time);
//4. 关闭驱动
close(fd);
return 0;
}
#include <time.h>
struct tm {
int tm_sec; /* seconds */
int tm_min; /* minutes */
int tm_hour; /* hours */
int tm_mday; /* day of the month */
int tm_mon; /* month */
int tm_year; /* year */
int tm_wday; /* day of the week */
int tm_yday; /* day in the year */
int tm_isdst; /* daylight saving time */
};
char *asctime(const struct tm *tm); //内部有一个全局空间存放转换的时间
char *asctime_r(const struct tm *tm, char *buf); //用户可以指定自己的空间
函数功能: 将tm时间结构体里的时间转为字符串格式返回(指针返回).
char *ctime(const time_t *timep);
char *ctime_r(const time_t *timep, char *buf);
函数功能: 将秒单位的时间转为字符串格式返回.
struct tm *gmtime(const time_t *timep);
struct tm *gmtime_r(const time_t *timep, struct tm *result);
函数功能: 将秒单位的时间转为格林威治时间返回---使用tm结构体。
struct tm *localtime(const time_t *timep);
struct tm *localtime_r(const time_t *timep, struct tm *result);
函数功能: 将秒单位的时间转为本地时间返回.---使用tm结构体
time_t mktime(struct tm *tm);
函数功能: 将tm结构体时间转为秒单位返回.
time_t time(time_t *t);
函数功能:如果形参填NULL就表示获取当期系统的秒单位时间.
size_t strftime(char *s, size_t max, const char *format,
const struct tm *tm);
函数功能: 将tm结构体的时间按照指定的格式转成字符串返回.
const char *format 格式有以下格式可以填:
%H 小时(以 00-23 来表示)
%M 分钟(以 00-59 来表示)
%S 秒(以本地的惯用法来表示)
%Y 年份(以四位数来表示)
%m 月份(以 01-12 来表示)
%d 日期(以 01-31 来表示)。
时间获取与转换示例:
#include <stdio.h>
#include <stdlib.h>
#include <time.h>
int main(int argc,char **argv)
{
/*1.获取本地的秒单位时间*/
time_t sec_time=time(NULL);
printf("当前系统的总秒数:%d\n",sec_time);
/*2. 将秒单位时间转为字符串返回*/
char time_buff[100];
ctime_r(&sec_time,time_buff);
printf("字符串格式时间(系统默认):%s\n",time_buff);
/*3. 将秒单位时间转为tm结构体返回*/
struct tm tm_time;
gmtime_r(&sec_time,&tm_time);
printf("国际时间: %d-%d-%d %d:%d:%d\n",tm_time.tm_year+1900,
tm_time.tm_mon+1,
tm_time.tm_mday,
tm_time.tm_hour,
tm_time.tm_min,
tm_time.tm_sec);
localtime_r(&sec_time,&tm_time);
printf("本地时间: %d-%d-%d %d:%d:%d\n",tm_time.tm_year+1900,
tm_time.tm_mon+1,
tm_time.tm_mday,
tm_time.tm_hour,
tm_time.tm_min,
tm_time.tm_sec);
/*4. 将tm结构体时间转为秒单位返回.*/
printf("总秒数:%d\n",mktime(&tm_time));
/*5. 将tm结构体时间格式按照指定格式转为字符串*/
strftime(time_buff,sizeof(time_buff),"%Y_%m_%d_%H_%M_%S.mp4",&tm_time);
printf("time_buff=%s\n",time_buff);
return 0;
}
#include <unistd.h>
unsigned int sleep(unsigned int seconds);
函数功能: 秒单位的延时函数.
int usleep(useconds_t usec);
函数功能: 微秒单位的延时函数.
#include <time.h>
int nanosleep(const struct timespec *req, struct timespec *rem);
函数功能: 秒+纳秒的延时函数.
struct timespec {
time_t tv_sec; /* seconds */
long tv_nsec; /* nanoseconds */
};
以上的函数都是可中断的延时函数。
比如: 延时10秒,有可能10秒钟还没有到达,它可以被其他信号终止.
示例代码:
#include <stdio.h>
#include <stdlib.h>
#include <time.h>
#include <unistd.h>
int main(int argc,char **argv)
{
printf("1234\n");
//sleep(5);
//usleep(1000*1000);
struct timespec req={5,1000}; //将要延时的时间
struct timespec rem; //保存是延时结束剩余的时间
nanosleep(&req,&rem);
printf("5678\n");
return 0;
}
函数原型介绍:
#include <unistd.h>
unsigned int alarm(unsigned int seconds);
闹钟超时之后会产生SIGALRM闹钟信号。
#include <signal.h>
typedef void (*sighandler_t)(int);
sighandler_t signal(int signum, sighandler_t handler);
函数功能: 捕获进程收到的信号.
函数参数:
int signum 要捕获的信号
sighandler_t handler 捕获信号之后调用的处理函数
示例代码:
例子代码:
#include <stdio.h>
#include <stdlib.h>
#include <time.h>
#include <unistd.h>
#include <signal.h>
void sighandler_func(int sig)
{
printf("闹钟时间到达.\n");
//定义一个闹钟
alarm(1); //重复定时
}
int main(int argc,char **argv)
{
//声明要捕获的信号
signal(SIGALRM,sighandler_func);
//定义一个闹钟
alarm(1);
while(1)
{
}
return 0;
}
运行效果:
[wbyq@wbyq linux_c]$ gcc app.c
[wbyq@wbyq linux_c]$ ./a.out
闹钟时间到达.
闹钟时间到达.
闹钟时间到达.
闹钟时间到达.
闹钟时间到达.