Time (简体中文)
一个操作系统通过如下内容确定时间:时间数值、时间标准、时区和夏令时调节(中国已经废止)。本文分别介绍各个部分的定义及如何设置他们。要维护准确的系统时间,请参考 网络时间协议 一文。
Contents
硬件时钟和系统时钟
系统用两个时钟保存时间:硬件时钟和系统时钟。
硬件时钟(即实时时钟 RTC 或 CMOS 时钟)仅能保存:年、月、日、时、分、秒这些时间数值,无法保存时间标准(UTC 或 localtime)和是否使用夏令时调节。
系统时钟(即软件时间) 与硬件时间分别维护,保存了:时间、时区和夏令时设置。Linux 内核保存为自 UTC 时间 1970 年1月1日经过的秒数。初始系统时钟是从硬件时间计算得来,计算时会考虑/etc/adjtime的设置。系统启动之后,系统时钟与硬件时钟独立运行,Linux 通过时钟中断计数维护系统时钟。
因为系统时间是按 32 为整数保存的,最大只能记到 2038 年,所以 32 位 Linux 系统将在 2038 年停止工作。
大部分操作系统的时间管理包括如下方面:
- 启动时根据硬件时钟设置系统时间
- 运行时通过时间同步联网校正时间
- 关机时根据系统时间设置硬件时间
读取时间
下面命令可以获得硬件时间和系统时间(硬件时钟按 localtime 显示):
$ timedatectl
设置时间
设置系统时间的本地时间:
# timedatectl set-time "yyyy-MM-dd hh:mm:ss"
例如:
# timedatectl set-time "2014-05-26 11:13:54"
设置时间为2014年,5月26日,11时13分54秒。
时间标准
时间表示有两个标准:localtime 和 UTC(Coordinated Universal Time) 。UTC 是与时区无关的全球时间标准。尽管概念上有差别,UTC 和 GMT (格林威治时间) 是一样的。localtime 标准则依赖于当前时区。
时间标准由操作系统设定,Windows 默认使用 localtime,Mac OS 默认使用 UTC 而 UNIX 系列的操作系统两者都有。使用 Linux 时,最好将硬件时钟设置为 UTC 标准,并在所有操作系统中使用。这样 Linux 系统就可以自动调整夏令时设置,而如果使用 localtime 标准那么系统时间不会根据夏令时自动调整。
通过如下命令可以检查当前设置,systemd 默认硬件时钟为协调世界时(UTC)。
$ timedatectl status | grep local
硬件时间可以用 hwclock 命令设置,将硬件时间设置为 localtime:
# timedatectl set-local-rtc true
硬件时间设置成 UTC:
# timedatectl set-local-rtc false
上述命令会自动生成/etc/adjtime,无需单独设置。
/etc/adjtime,systemd 会假定硬件时间按 UTC 设置。系统启动装入 rtc 驱动时可能会根据系统时钟设置硬件时钟。是否设置依赖于平台、内核版本和内核编译选项。如果进行了设置,此时会假定硬件时钟为 UTC 标准,/sys/class/rtc/rtcN/hctosys(N=0,1,2,..) 会设置成 1。后面 systemd 会根据/etc/adjtime重新设置。
如果设置成本地时间,处理夏令时有些麻烦。如果夏令时调整发生在关机时,下次启动时时间会出现问题(更多信息)。最新的内核直接从实时时钟芯片(RTC)读取时间,不使用 hwclock,内核把从 RTC 读取的时间当作 UTC 处理。所以如果硬件时间是地方时,系统启动一开始识别的时间是错误的,之后很快会进行矫正。这可能导致一些问题(尤其是时间倒退时)。
Windows 系统使用 UTC
如果同时安装了 Windows 操作系统(默认使用地方时),那么一般 RTC 会被设置为地方时。Windows 其实也能处理 UTC,需要修改注册表。建议让 Windows 使用 UTC,而非让 Linux 使用地方时。Windows 使用 UTC 后,请记得禁用 Windows 的时间同步功能,以防 Windows 错误设置硬件时间。如上文所说,Linux 可以使用NTP服务来在线同步硬件时钟。
使用 regedit,新建如下 DWORD 值,并将其值设为十六进制的 1。
HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Control\TimeZoneInformation\RealTimeIsUniversal
也可以用管理员权限启动命令行来完成:
reg add "HKEY_LOCAL_MACHINE\System\CurrentControlSet\Control\TimeZoneInformation" /v RealTimeIsUniversal /d 1 /t REG_DWORD /f
如果以上操作不起作用,并且你使用的是 Windows 64位系统,将 DWORD 修改为 QWORD。
如果 Windows 要求根据夏令时更新时钟,可以允许。时钟仍然是 UTC,仅是显示时间会改变。
设置时间标准后需要重新设置硬件时间和系统时间。 updated[broken link: invalid section]。
如果你有时间偏移问题,重新安装 tzdata 并再次设置你的时区:
# timedatectl set-timezone America/Los_Angeles
UTC 在Ubuntu的设置
Ubuntu and its derivatives have the hardware clock set to be interpreted as in "localtime" if Windows was detected on any disk during Ubuntu installation. This is apparently done deliberately to allow new Linux users to try out Ubuntu on their Windows computers without editing the registry.
要改变Ubuntu的此问题,你需要做到以下几点。打开文件:
/etc/default/rcS
并改变 UTC 标记为 UTC=yes.
时区
检查当前时区:
$ timedatectl status
显示可用时区:
$ timedatectl list-timezones
修改时区:
# timedatectl set-timezone <Zone>/<SubZone>
例如:
# timedatectl set-timezone Asia/Shanghai
此命令会创建一个/etc/localtime软链接,指向/usr/share/zoneinfo/中的时区文件,如果手动创建此链接请确保是相对链接而不是绝对链接,参阅archlinux(7).
See timedatectl(1), localtime(5), and archlinux(7) for more details.
时钟偏移
最能代表“真实时间”的是国际原子时钟),所有的时钟都是有误差的。电子时钟的时间是不准的,但是一般有固定的偏移。这种于基值的差称为“time skew”或“时间偏移”。用 hwclock 设置硬件时间时,会计算每天偏移的秒数。偏移值是原硬件时间与新设置硬件时间的差,并且考虑上次硬件时间设置时的偏移。新的偏移值会在设置时钟时写到文件 /etc/adjtime 。
如果硬件时钟总是过快或过慢,可能是计算了错误的偏移值。硬件时钟设置错误或者时间标准与其他操作系统不一致导致。删除文件 /etc/adjtime 可以删除偏移值,然后设置正确的硬件时钟和系统时钟,并检查时间标准是不是设置正确。
提高系统时间精度的方法有:
- NTP 可以通过网络时间协议同步 Linux 系统的时间。NTP 也会修正中断频率和每秒滴答数以减少时间偏移。并且每隔 11 分钟同步一次硬件时钟。
时钟同步
网络时间协议 (NTP) 是一个通过包交换和可变延迟网络来同步计算机系统时间的协议。下列为这个协议的实现:
- NTP 守护进程是这个协议的参考实现,推荐用于时间服务器。它也可以调节中断频率和每秒滴答次数以减少系统时钟误差,使得硬件时钟每隔11秒重新同步一次。
- sntp 是 ntp 包里附带的一个 SNTP 客户端。它取代了 ntpdate ,并被推荐用于非服务器环境。
- systemd-timesyncd 是一个简单的 SNTP 守护进程。它只实现了客户端,专用于从远程服务器查询时间,更适用于绝大部分安装的情形。
- OpenNTPD 是 OpenBSD 项目的一部分,同时实现了客户端和服务器。
- Chrony 是一个客户端和服务器,更适合漫游,是为不能始终保持在线的系统而特别设计。
- ntpclientAUR 是简单的命令行 NTP 客户端
Per-user/会话或临时设置
For some use cases it may be useful to change the time settings without touching the global system values. For example to test applications relying on the time during development or adjusting the system time zone when logging into a server remotely from another zone.
To make an application "see" a different date/time than the system one, you can use the faketime (from libfaketime) or the datefudge utilities.
If instead you want an application to "see" a different time zone than the system one, set the TZ environment variable, for example: 
$ date && export TZ="/usr/share/zoneinfo/Pacific/Fiji" && date
Sa 24. Mai 12:38:26 CEST 2014 Sa 24. Mai 22:38:26 FJT 2014
This is different than just setting the time, as for example it allows to test the behaviour of a program with positive or negative UTC offset values, or the effects of DST changes when developing on systems in a non-DST time zone.
Another use case is having different time zones set for different users of the same system: this can be accomplished by setting the TZ variable in the shell's configuration file, see Environment variables#Defining variables and Autostarting#Shells.
故障排除
时间显示的既不是UTC也不是本地时间
This might be caused by a number of reasons. For example, if your hardware clock is running on local time, but timedatectl is set to assume it is in UTC, the result would be that your timezone's offset to UTC effectively gets applied twice, resulting in wrong values for your local time and UTC.
To force your clock to the correct time, and to also write the correct UTC to your hardware clock, follow these steps:
- Setup ntpd (enabling it as a service is not necessary).
- Set your time zone[broken link: invalid section] correctly.
-  Run ntpd -qgto manually synchronize your clock with the network, ignoring large deviations between local UTC and network UTC.
-  Run hwclock --systohcto write the current software UTC time to the hardware clock.
提示和技巧
fake-hwclock
alarm-fake-hwclock designed especially for system without battery backed up RTC, it includes a systemd service which on shutdown saves the current time and on startup restores the saved time, thus avoiding strange time travel errors.
Install fake-hwclock-gitAUR, start and enable the service fake-hwclock.service.