前言
一个操作系统通过如下内容确定时间:时间数值、时间标准、时区和夏令时调节(中国已经废止)。
基础概念
时钟分类
Linux 时钟分为系统时钟(System Clock)和硬件(Real Time Clock,简称 RTC)时钟。
硬件时钟:则是主板上由电池供电的时钟,这个硬件时钟可以在BIOS 中进行设置。仅能保存:年、月、日、时、分、秒这些时间数值,无法保存时间标准(UTC 或 localtime)和是否使用夏令时调节。
系统时钟:是指当前 Linux Kernel 中的时钟,当 Linux 启动时,硬件时钟会去读取系统时钟的设置,然后系统时钟就会独立于硬件运作。Linux 中的所有命令(包括函数)都是采用的系统时钟设置。才系统时钟保存了:时间、时区和夏令时设置。Linux 内核保存为自 UTC 时间 1970 年1月1日经过的秒数。初始系统时钟是从硬件时间计算得来,计算时会考虑/etc/adjtime的设置。系统启动之后,系统时钟与硬件时钟独立运行,Linux 通过时钟中断计数维护系统时钟。
因为系统时间是按 32 为整数保存的,最大只能记到 2038 年,所以 32 位 Linux 系统将在 2038 年停止工作。
时间标准
时间表示有两个标准:localtime 和 UTC(Coordinated Universal Time) 。
UTC 是与时区无关的全球时间标准。尽管概念上有差别,UTC 和 GMT (格林威治时间) 是一样的。
localtime 标准则依赖于当前时区。
时间标准由操作系统设定,Windows 默认使用 localtime,Mac OS 默认使用 UTC 而 UNIX 系列的操作系统两者都有。使用 Linux 时,最好将硬件时钟设置为 UTC 标准,并在所有操作系统中使用。这样 Linux 系统就可以自动调整夏令时设置,而如果使用 localtime 标准那么系统时间不会根据夏令时自动调整。
通过如下命令可以检查当前设置,终端执行:
timedatectl status | grep local
硬件时间可以用 hwclock 命令设置,将硬件时间设置为 localtime,终端执行:
timedatectl set-local-rtc 1
硬件时间设置成 UTC,终端执行:
timedatectl set-local-rtc 0
上述命令会自动生成/etc/adjtime,无需单独设置。
系统启动装入 rtc 驱动时可能会根据系统时钟设置硬件时钟。是否设置依赖于平台、内核版本和内核编译选项。如果进行了设置,此时会假定硬件时钟为 UTC 标准,/sys/class/rtc/rtcN/hctosys(N=0,1,2,..) 会设置成 1。后面 Upstart 会根据/etc/adjtime重新设置。所以讲硬件时间设置为 localtime 会在启动时导致某些莫名其妙的问题,往回设置系统时间一般都是不安全的。
RTC clock
实时时钟(Real-time clock,RTC)是指可以像时钟一様输出实际时间的电子设备,一般会是集成电路,因此也称为时钟芯片。
大部分操作系统的时间管理包括如下方面:
启动时根据硬件时钟设置系统时间
运行时通过 NTP 守护进程联网校正时间
关机时根据系统时间设置硬件时间。