树莓派开始,玩转Linux
上QQ阅读APP看本书,新人免费读10天
设备和账号都新为新人

第9章 时间的故事

对于电子设备来说,时间都是基础性的功能,很容易被人忽视。20世纪的“千年虫”问题,就是时间方面设计缺陷造成的。对于网络连接的多设备来说,保持时间同步是一个新的问题。对于树莓派的众多应用场景来说,时间的准确性都至关重要。树莓派提供了NTP服务,通过网络来校正时间。即使在断网的情况下,也可以通过物理计时来校正时间。而树莓派使用的Linux系统,也提供了date命令这样便利的时间工具。

9.1 NTP服务

树莓派中内置了NTP服务,所以连上网之后就可以自动调整时间。NTP是网络时间协议(Network Time Protocol)的简称,主要用于网络时间的同步。NTP协议早在20世纪80年代就已经诞生了,至今仍是互联网的基础性协议之一。NTP通信分为服务器和客户端两方。客户端发出的数据包包含发出时客户端的时间。服务器收到数据包并回复。在回复的数据包中,附加了服务器收到和发出数据包的时间。客户端收到回复后就可以获得网络延迟时间,以及自己和服务器的时间差。客户端据此调整自己的时钟,就可以与服务器时间保持同步。

你可以通过下面的命令来查询当前使用的NTP服务器:

    $sudo ntpq -pn

命令返回:

        remote         refid     st t when poll reach  delay  offset  jitter
    ==============================================================================
    203.135.184.123 .GPS.          1 u  322  64  20  365.136  -7.571  15.792
    223.112.179.133 .INIT.        16 u   -1024   0   0.000   0.000  0.000
    *202.112.29.82  202.118.1.46    2 u  122  64  276  53.148   0.766  0.868

行首加*号的是当前服务器。此外,还列出了网络延迟时间(Delay)、与服务器时间差(Offset)等关键的NTP时间数据。单位是毫秒(Millisecond)。

如果NTP服务出现问题,就会造成树莓派时间错误,可以强制要求NTP对表:

    $sudo service ntp stop
    $sudo ntpd -gq
    $sudo service ntp start

上面的第一句和第三句分别用于停止和启动NTP服务。

即使不使用NTP,也可以手动调整系统时间:

    $sudo date -s "1 Jan 2017 00:00:00"

即把系统时间调整为2017年1月1日00:00:00。

然后用date命令来显示系统当前时间:

    $date

可以看到,时间已经调整成功。

9.2 时区设置

因为地球自西向东转动,所以全球不同经度地点的日出、日落及正午的时间不同。人们又习惯用同样的12点来代表正午,这意味着不同经度的人要用不一样的表。可是,如果每时每刻都要根据经度调表,就会非常麻烦。因此,地球以15º的经度来划分时区,一个时区内用统一的时间,向东跨过一个时区,就需要把表调快1小时。当然,时区不是严格按照15°划分的。比如,一些地跨多个时区的国家有可能统一用一个时区,例如中国。

对于不同地区的用户来说,往往需要把树莓派调整成当地的时区。你可以用raspi-config进入树莓派的设置页面,在“4 Localisation Options”→“I2 Change Timezone”页面中修改时区。

当然,也可以用下面的命令手动修改:

    $sudo cp /usr/share/zoneinfo/Asia/Shanghai /etc/localtime

目录/user/share/zoneinfo中有多个以各大洲名字命名的文件夹,里面的文件以该州的主要城市命名。把对应城市的文件复制到/etc/localtime,就可以把系统的时区设成该城市所用的时区。这里把时区修改为“Shanghai”,也就是上海。修改之后,用date命令查看时间,可以看到时区简写变成CST,也就是“上海时间”的缩写:

    Tue  3 Jan 20:42:24 CST 2017

用date命令查看UTC时间:

    $date -u

显示的时间正好相差8个小时:

    Tue  3 Jan 12:42:24 UTC 2017

9.3 实时时钟

大多数电脑的主板上包含了一个实时时钟(RTC, Real Time Clock)。实时时钟是一个有电源的表,能在电脑断电时继续计时。因此,电脑断电后一天再开机,你会发现电脑的时钟也往前走了一天。树莓派并不包含一个实时时钟,因此,如果树莓派断电一天再开机,在NTP服务校正时间之前,树莓派的时间还停留在一天前。为了解决这一问题,可以给树莓派附加一个实时时钟,比如PiFace专门为树莓派设计的实时时钟。

这个实时时钟被设计成一个使用纽扣电池的电路板。把PiFace电路板的孔对准树莓派的GPIO针脚插入,就可以使用了。插入位置如图9-1所示。插入正确的情况下,电池正好在树莓派CPU的上方。网上也有人诟病这一设计,认为电池的发热会影响树莓派CPU的散热。不过笔者在使用中并没有遇到太大问题。

图9-1 RTC安装位置

为了使用这款实时时钟,还需要进行一些设置。首先,这块电路板是通过I2C接口与树莓派通信的,所以要在raspi-config的页面中打开I2C接口。然后,安装所需的工具包:

    $sudo apt-get install i2c-tools
    $sudo apt-get install python-smbus

接下来,赋予用户pi使用I2C接口的权限:

    $sudo usermod -aG i2c pi

打开文件/etc/modules,这里面列出了系统可以加载的模块。检查是否有如下两行,如果没有请添加:

    i2c-dev
i2c-bcm2708

下面一段程序修改自官网程序,用来让树莓派在开机时自动加载实时时钟。把下面程序保存为rtc.bash,并运行:

    #! /bin/bash
    #=======================================================================
    # NAME: set_revision_var
    # DESCRIPTION: Stores the revision number of this Raspberry Pi into
    #            $RPI_REVISION
    #==============

    set_revision_var() {
        revision=$(grep "Revision" /proc/cpuinfo | sed -e "s/Revision\t: //")
        RPI2_REVISION=$((16#a01041))
        RPI3_REVISION=$((16#a02082))
        if [ "$((16#$revision))" -ge "$RPI3_REVISION" ]; then
          RPI_REVISION="3"
        elif [ "$((16#$revision))" -ge "$RPI2_REVISION" ]; then
          RPI_REVISION="2"
        else
          RPI_REVISION="1"
        fi
    }

    #=======================================================================
    # NAME: start_on_boot
    # DESCRIPTION: Load the I2C modules and send magic number to RTC, on boot.
    #=======================================================================
    start_on_boot() {
        echo "[info]Create a new pifacertc init script to load time from PiFace
RTC."
        echo "[info]Adding /etc/init.d/pifacertc ."
        if [[ $RPI_REVISION == "3" ]]; then
            i=1  # i2c-1
        elif [[ $RPI_REVISION == "2" ]]; then
            i=1  # i2c-1
        else
            i=0  # i2c-0
        fi

        cat > /etc/init.d/pifacertc  << EOF
    #! /bin/sh
    ### BEGIN INIT INFO
    # Provides:        pifacertc
    # Required-Start:   udev mountkernfs \$remote_fs raspi-config
    # Required-Stop:
    # Default-Start:    S
    # Default-Stop:
    # Short-Description: Add the PiFace RTC
    # Description:      Add the PiFace RTC
    ### END INIT INFO

    . /lib/lsb/init-functions

    case "\$1" in
      start)
        log_success_msg "Probe the i2c-dev"
        modprobe i2c-dev
        # Calibrate the clock (default: 0x47). See datasheet for MCP7940N
        log_success_msg "Calibrate the clock"
        i2cset -y $i 0x6f 0x08 0x47
        log_success_msg "Probe the mcp7941x driver"
        modprobe i2c:mcp7941x
        log_success_msg "Add the mcp7941x device in the sys filesystem"
        # https://www.kernel.org/doc/Documentation/i2c/instantiating-devices
        echo mcp7941x 0x6f > /sys/class/i2c-dev/i2c-$i/device/new_device
        log_success_msg "Synchronise the system clock and hardware RTC"
        hwclock --hctosys
        ;;
      stop)
        ;;
      restart)
        ;;
      force-reload)
        ;;
      *)
        echo "Usage: \$0 start" >&2
        exit 3
        ;;
    esac
    EOF
        chmod +x /etc/init.d/pifacertc

        echo "[info]Install the pifacertc init script"
        update-rc.d pifacertc  defaults
    }

    set_revision_var &&
    start_on_boot

完成后重启电脑。此时树莓派应该已经自动通过I2C接口加载了实时时钟。可以通过下面的命令来检查实时时钟是否就位:

    $sudo i2cdetect -y 1

如果实时时钟就位,那么60开头的行会有一个“UU”的标准位。通过下面的命令,可以读出实时时钟的时间:

    $sudo hwclock -r

通过下面的命令可以把当前系统时间写入实时时钟:

    $sudo hwclock --systohc

有了实时时钟,就可以在无网环境下保持时间的连续性了。PiFace的产品卖得有些贵,淘宝上有一些便宜的实时时钟可以选购。

9.4 date的用法

date是UNIX系统下常用的时间命令工具,能提供非常丰富的时间功能,比如以特定格式显示时间:

    $date +"%Y year %m month %d day"

+号后面的字符串代表了时间显示格式,%开头的标识符会用时间信息填充。%Y代表年,%m代表月份,%d代表日期,所以上面的命令会返回:

    2017 year 01 month 01 day

用于控制date输出格式的标识符还有很多:

●%a,显示一周中星期几的缩写,比如Thu。

●%A,显示一周中星期几,比如Thursday。

●%b,显示月份的缩写,比如Feb。

●%B,显示月份,比如February。

●%d,显示一个月的哪一天,比如09。

●%D,显示日期,月/日/年,比如02/07/17。

●%F,显示日期,年-月-日,比如2017-02-07。

●%H,显示24小时制的小时,比如23。

●%I,显示12小时制的小时,比如11。

●%j,显示一年中的天数,比如138。

●%m,显示月份,比如02。

●%M,显示分钟,比如14。

●%S,显示秒,比如47。

●%N,显示纳秒,比如264587606。

●%T,显示24小时制的时间,时:分:秒,比如23:44:17。

●%u,显示一周中的哪一天,周一是1。

●%U,显示一年中的周数,而周日是一周的开始,比如47。

●%Y,显示完整的年份,比如2013。

●%Z,显示时区的缩写。

除了显示当前时间,date还可以用来显示用户输入的时间:

    $date --date="2017/01/03 12:00:00"

配合上面介绍的格式控制,这个功能可以实现日期格式的转换。这个功能还可以用于时间推算。比如下面的命令就可以用于推算2016年11月12日之前1个月的时间:

    $date --date="2016/11/12-1 month"

除了“-1 month”,还可以是“+1 second”“-2 day”等多种时间差,能满足各种各样的时间推算需求。