首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >如果用户在设备中更改了日期和时间,那么如何在android中获取当前日期和时间?

如果用户在设备中更改了日期和时间,那么如何在android中获取当前日期和时间?
EN

Stack Overflow用户
提问于 2015-08-08 18:00:40
回答 3查看 2.5K关注 0票数 5

如果任何用户在他/她的设备中更改了日期和时间,那么如何获取实际运行的当前日期和时间。我正在处理一个使用当前日期作为开始日期项目,因此如果用户按他想要的方式更改他/她的日期,那么它的自定义日期将是我的开始日期,而不是我想要的当前日期,如果设备日期在之前或之后。

例如。现在,当前日期是2015年8月8日,那么用户将他/她的设备日期更改为2015年8月20日,那么我如何才能获得实际日期,这是2015年8月8日?先谢谢你……

EN

回答 3

Stack Overflow用户

发布于 2020-08-29 23:54:39

到NTP服务器的连接:

代码语言:javascript
复制
import android.os.AsyncTask;

import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;


public class NTPUTCTime {
    private static final String TAG = "SntpClient";

    private static final int RECEIVE_TIME_OFFSET = 32;
    private static final int TRANSMIT_TIME_OFFSET = 40;
    private static final int NTP_PACKET_SIZE = 48;

    private static final int NTP_PORT = 123;
    private static final int NTP_MODE_CLIENT = 3;
    private static final int NTP_VERSION = 3;

    // Number of seconds between Jan 1, 1900 and Jan 1, 1970
// 70 years plus 17 leap days
    private static final long OFFSET_1900_TO_1970 = ((365L * 70L) + 17L) * 24L * 60L * 60L;

    private long mNtpTime;

    public boolean requestTime() {
        try {
            new AsyncTask<Void, Void, Boolean>() {
                @Override
                protected Boolean doInBackground(Void... voids) {
                    try {
                        DatagramSocket socket = new DatagramSocket();
                        socket.setSoTimeout(1000);
                        InetAddress address = InetAddress.getByName("pool.ntp.org");
                        byte[] buffer = new byte[NTP_PACKET_SIZE];
                        DatagramPacket request = new DatagramPacket(buffer, buffer.length, address, NTP_PORT);
                        buffer[0] = NTP_MODE_CLIENT | (NTP_VERSION << 3);

                        writeTimeStamp(buffer, TRANSMIT_TIME_OFFSET);

                        socket.send(request);

                        // read the response
                        DatagramPacket response = new DatagramPacket(buffer, buffer.length);
                        socket.receive(response);
                        socket.close();

                        mNtpTime = readTimeStamp(buffer, RECEIVE_TIME_OFFSET);
                    } catch (Exception e) {
                        //  if (Config.LOGD) Log.d(TAG, "request time failed: " + e);
                        e.printStackTrace();
                        return false;
                    }
                    return true;
                }
            }.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR).get();

        } catch (Exception e) {
            //  if (Config.LOGD) Log.d(TAG, "request time failed: " + e);
            e.printStackTrace();
            return false;
        }

        return true;
    }


    public long getNtpTime() {
        return mNtpTime;
    }


    /**
     * Reads an unsigned 32 bit big endian number from the given offset in the buffer.
     */
    private long read32(byte[] buffer, int offset) {
        byte b0 = buffer[offset];
        byte b1 = buffer[offset + 1];
        byte b2 = buffer[offset + 2];
        byte b3 = buffer[offset + 3];

        // convert signed bytes to unsigned values
        int i0 = ((b0 & 0x80) == 0x80 ? (b0 & 0x7F) + 0x80 : b0);
        int i1 = ((b1 & 0x80) == 0x80 ? (b1 & 0x7F) + 0x80 : b1);
        int i2 = ((b2 & 0x80) == 0x80 ? (b2 & 0x7F) + 0x80 : b2);
        int i3 = ((b3 & 0x80) == 0x80 ? (b3 & 0x7F) + 0x80 : b3);

        return ((long) i0 << 24) + ((long) i1 << 16) + ((long) i2 << 8) + (long) i3;
    }

    /**
     * Reads the NTP time stamp at the given offset in the buffer and returns
     * it as a system time (milliseconds since January 1, 1970).
     */
    private long readTimeStamp(byte[] buffer, int offset) {
        long seconds = read32(buffer, offset);
        long fraction = read32(buffer, offset + 4);
        return ((seconds - OFFSET_1900_TO_1970) * 1000) + ((fraction * 1000L) / 0x100000000L);
    }

    /**
     * Writes 0 as NTP starttime stamp in the buffer. --> Then NTP returns Time OFFSET since 1900
     */
    private void writeTimeStamp(byte[] buffer, int offset) {
        int ofs = offset++;

        for (int i = ofs; i < (ofs + 8); i++)
            buffer[i] = (byte) (0);
    }

}

或者你也可以使用这个库

https://github.com/instacart/truetime-android

票数 1
EN

Stack Overflow用户

发布于 2015-08-08 18:06:00

类似于How can I get the actual date and time if the device date and time are inaccurate?

使用时间服务器--这将是最好的方法。或者不管你有什么后端,公开一个API来获取应用程序启动时的服务器时间,然后你可以在整个会话期间使用Timer保持时间,如果你除了日期之外还需要hh:mm:ss。

票数 0
EN

Stack Overflow用户

发布于 2015-08-08 19:16:59

来自user2629865的答案大部分是正确的。您需要使用SNTP (简单网络时间协议)之类的协议来连接到受信任的Internet时间服务器。

如果用户还将时区更改为不正确的时区,问题就来了。

在你的情况下这是一个问题的原因是NTP和SNTP检索UTC格式的时间,这在一种方式上是好的,但显然不会得到基于设备地理位置的‘本地’时间。此时,您需要查看地理位置服务以尝试识别位置和时区,以便获得准确的本地日期/时间。

简而言之,没有简单的一步法来解决这个问题,最终归结为你为什么需要这样做。如果用户希望在他们自己的设备上修改他们的日期/时间/时区设置,那是他们的选择。

票数 0
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/31891925

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档