1. 程式人生 > >Android 校正系統時間的三種解決方案

Android 校正系統時間的三種解決方案

前言

在開發過程中我們常常需要獲取系統時間。
Android系統的自動確認時間,是由系統通過訪問廠家的NTP伺服器的時間,然後修改後得到的。
所以當沒有網路或者在內網環境下的時候,系統無法訪問到NTP伺服器,便會造成系統時間錯誤。
所以這個時候我們就需要程式去修改系統的時間,或者獲取一個正確的時間來代替系統時間。

NTP伺服器

【Network Time Protocol(NTP)】是用來使計算機時間同步化的一種協議,它可以使計算機對其伺服器或時鐘源(如石英鐘,GPS等等)做同步化,它可以提供高精準度的時間校正(LAN上與標準間差小於1毫秒,WAN上幾十毫秒),且可介由加密確認的方式來防止惡毒的協議攻擊。時間按NTP伺服器的等級傳播。按照離外部UTC源的遠近把所有伺服器歸入不同的Stratum(層)中。

解決方案

根據不同的情況,我實現瞭如下三種解決方案:

  1. 修改系統時間。
    優點:程式啟動時執行一次即可,一勞永逸。
    缺點:只能在原生系統中使用,非原生系統無法安裝。(具體後面會解釋)。

  2. 獲取NTP伺服器時間代替系統時間。
    優點:無需Root,適用於任何手機及系統。
    缺點:需要可以訪問外部網路,內網環境下則需要一臺自己的NTP伺服器。

  3. 獲取網頁時間代替系統時間。
    優點:無需Root,適用於任何手機及系統,適用於任何網路環境。
    缺點:需要一條額外的執行緒,去維護時間準確,容易造成誤差。

程式碼及目錄

這裡寫圖片描述

1.修改系統時間

1.配置系統JDK環境變數

2.修改系統時間,一行程式碼如下:

SystemClock.setCurrentTimeMillis(long millis);

3.修改AndroidManifest.xml

在應用程式的AndroidManifest.xml中的manifest節點中加入android:sharedUserId=”android.uid.system”這個屬性。
新增該屬性後,由於許可權衝突,程式無法直接安裝,必須重新簽名。

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.demon.setsystemtime" android:sharedUserId="android.uid.system" >
</manifest>

4.編譯apk,重新簽名

Android Studio編譯生成apk,然後將apk複製到上述目錄的sign資料夾(重新簽名的資料夾)下,將apk的名字與bat指令碼中的一致。
用壓縮軟體開啟apk檔案,刪掉META-INF目錄下的CERT.SF和CERT.RSA兩個檔案。
雙擊執行bat指令碼,指令碼程式碼如下:

@echo off
java -jar signapk.jar platform.x509.pem platform.pk8 demo.apk test.apk
pause

signapk.jar: Android提供的簽名工具。
platform.x509.pem&platform.pk8: Android原始碼目錄中”build/target/product/security”,下面的兩個檔案。
demo.apk:程式編譯生成的apk。
test.apk:重新簽名後的apk。

這也有一個問題,就是這樣生成的程式只有在原始的Android系統或者是自己編譯的系統中才可以用,因為這樣的系統才可以拿到platform.pk8和platform.x509.pem兩個檔案。要是別家公司做的Android上連安裝都安裝不了。

安裝重新簽名的apk,執行即可修改系統時間。

獲取NTP伺服器時間代替系統時間

直接使用truetime-android框架即可同步NTP伺服器時間,調取框架內的方法便可以實時獲取最新時間。
GitHub:https://github.com/instacart/truetime-android
具體的使用可以參考程式碼,或者GitHub文件。

### 阿里雲提供了7個NTP伺服器
ntp1.aliyun.com
ntp2.aliyun.com
ntp3.aliyun.com
ntp4.aliyun.com
ntp5.aliyun.com
ntp6.aliyun.com
ntp7.aliyun.com


### 中國科學技術大學NTP伺服器
time.ustc.edu.cn

獲取網頁時間代替系統時間

根據下列程式碼,我們就可以獲取任何一個網址的時間(內網伺服器地址)。
拿到這個時間後,可以開啟一個執行緒,做定時任務,不斷更新該時間,以到達時間時間同步的效果。
具體實現方法不再闡述。

/**
     * 網址訪問
     * @param url  網址
     * @return urlDate 物件網址時間
     */
    public static String VisitURL(String url){
        String urlDate = null;
        try {
            URL url1 = new URL(url);
            URLConnection  conn = url1.openConnection();  //生成連線物件
            conn.connect();  //連線物件網頁
            Date date = new Date(conn.getDate());  //獲取物件網址時間
            SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");  //設定日期格式
            urlDate = df.format(date);
        } catch (Exception e) {
            e.printStackTrace();
        }
        return urlDate;
    }

效果

這裡寫圖片描述

參考