Android获取GPS时间

Android获取所在时区正确时间的方式有两种:

一:wifi获取时间

在联网且系统设置了自动获取时间自动获取时区 的系统设置前提下,系统会自动更新正确的时间,当然如果你用了TextClockDateClock也会自动更新。

二:通过GPS获取时间。

2.1未联网且GPS可用的情况下,可以通过获取原生定位信息来获取所在地点的时间,实现如下:

注意,下面的代码时在Fragment里面实现的

java 复制代码
/**  
* 用于获取GPS时间  
*/  
private LocationManager locationManager;  
/**  
* 取消后台定位的时间  
*/  
private final static int CANCEL_LOCATION_DELAY = 5 * 60 * 1000;  
/**  
* 避免多次取消定位和反注册  
*/  
private boolean isHadCancelLocation = false;

/**  
* 如果网络不可用,使用GPS进行刷新系统时间,系统时间更新后会自动更新TextClock和DateClock;如果有网络,理论上会自动刷新系统时间,所以不处理。  
*/  
private void initSystemTime() {  
    if(!NetworkHelper.isNetworkAvailable(getContext())){  
        getLocationBySystem();
        // 注意,这里请用原始Handler.postDelay代替,我用的自定义延时类。
        ArchTaskExecutor.getInstance().postToMainThreadDelay(cancelLocationRunnable, CANCEL_LOCATION_DELAY);  
    }  
}

// 注意,普通app需要申请定位权限,我的是系统级app,所以忽略。
@SuppressLint("MissingPermission")  
private void getLocationBySystem() {  
    XLog.i("无网络启动获取Gps定位");  
    locationManager = (LocationManager) getContext().getSystemService(Context.LOCATION_SERVICE);  
    // 注册位置监听器  
    locationManager.requestLocationUpdates(LocationManager.GPS_PROVIDER, 0, 0, locationListener);  
}  

/**  
* 通过GPS获取定位信息  
*/  
private final LocationListener locationListener = location -> {  
    if(null != location) {  
        // 就这里是关键获取代码,其实高德地图Api能自己触发系统时间的更新,不知道怎么实现的,只要打开他们的地图就可以,不管是第三方的高德地图app还是自己内嵌的高德地图api模块,理论上也是location.getTime()获取到的,不理解的是他们为什么有权限更新系统时间。
        long gpsTime = location.getTime();  
        cancelLocationUpdates();  
        XLog.i("Gps定位获取成功并注销定位监听,GPS时间 = " + gpsTime);  
        // 用过这个进行模拟测试,resetSystemTime(1712800932000L);显示的时间是2024-04-11 10:02:12,正确  
        resetSystemTime(gpsTime);  
    }  
};
  
private final Runnable cancelLocationRunnable = () -> {  
    XLog.i("获取定位超时,注销定位监听");  
    cancelLocationUpdates();  
};  
  
/**  
* 获取定位时间成功取消定位  
*/  
public void cancelLocationUpdates() {  
    XLog.i("注销定位监听" + isHadCancelLocation);  
    if (locationManager != null && !isHadCancelLocation) {  
        locationManager.removeUpdates(locationListener);  
        isHadCancelLocation = true;  
        locationManager = null;  
    }  
}  

/**
* 注意,我的是系统级app,所以忽略动态申请权限,普通应用无法设置系统时间但是可以参考取值。
* 系统级app配置 <uses-permission android:name="android.permission.SET_TIME"/> 权限即可
**/
@SuppressLint("MissingPermission")  
private void resetSystemTime(Long utcTimeMillis) {  
    try {  
        // 将UTC时间转换为Date对象  
        Date locationTime = new Date(utcTimeMillis);  
        long localTimeMillis = locationTime.getTime();  

        // 获取AlarmManager实例  
        AlarmManager alarmManager = (AlarmManager) getActivity().getSystemService(Context.ALARM_SERVICE);  
        // 设置系统时间(本地时间)  
        if (alarmManager != null) {  
            alarmManager.setTime(localTimeMillis);  
            XLog.i("重置系统时间成功" + localTimeMillis);  
        }  
     } catch (Exception e){  
        XLog.e("重置系统时间失败");  
    }  
}

2.2 如果你接入了高德地图SDK,也可以使用如下api去获取当前地点的时间:

java 复制代码
/**  
* 这里目前获取到GPS返回的正确的时间  
* @throws Exception  
*/  
@SuppressLint("MissingPermission")  
public void getLocationByAMap() throws Exception {  
    // 高德地图的api  
    mlocationClient = new AMapLocationClient(getContext());  
    //初始化定位参数  
    mLocationOption = new AMapLocationClientOption();  
    //设置定位监听  
    mlocationClient.setLocationListener(new AMapLocationListener() {  
    @Override  
    public void onLocationChanged(AMapLocation aMapLocation) {  
        if (aMapLocation != null) {  
            if (aMapLocation.getErrorCode() == 0) {  
                //定位成功回调信息,设置相关消息  
                aMapLocation.getLocationType();//获取当前定位结果来源,如网络定位结果,详见定位类型表  
                SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");  
                Toast.makeText(getContext(), "时间戳为:" + aMapLocation.getTime(),Toast.LENGTH_SHORT).show();  
                Date date = new Date(aMapLocation.getTime());  
                String time = df.format(date);//定位时间  


                SimpleDateFormat dateFormat = new SimpleDateFormat("MM月dd日");  
                SimpleDateFormat hourMinuteSecondFormat = new SimpleDateFormat("HH:mm aa");  

                mlocationClient.stopLocation();// 获取成功停止定位  
                Toast.makeText(getContext(), "时间为:" + time + ";停止了定位并开始设置系统时间" + dateFormat + ";" + hourMinuteSecondFormat,Toast.LENGTH_SHORT).show();  

                // 需要配置 <uses-permission android:name="android.permission.SET_TIME"/>
                resetSystemTime(aMapLocation.getTime());  
            } else {  
                //显示错误信息ErrCode是错误码,errInfo是错误信息,详见错误码表。  
                Log.e("AmapError", "location Error, ErrCode:"  
                + aMapLocation.getErrorCode() + ", errInfo:"  
                + aMapLocation.getErrorInfo());  
            }  
        }  
    }});  
    //设置定位模式为高精度模式,Battery_Saving为低功耗模式,Device_Sensors是仅设备模式   mLocationOption.setLocationMode(AMapLocationClientOption.AMapLocationMode.Hight_Accuracy);  
    //设置定位间隔,单位毫秒,默认为2000ms  
    mLocationOption.setInterval(1000);  
    //设置定位参数  
    mlocationClient.setLocationOption(mLocationOption);  
    // 此方法为每隔固定时间会发起一次定位请求,为了减少电量消耗或网络流量消耗,  
    // 注意设置合适的定位时间的间隔(最小间隔支持为1000ms),并且在合适时间调用stopLocation()方法来取消定位请求  
    // 在定位结束后,在合适的生命周期调用onDestroy()方法  
    // 在单次定位情况下,定位无论成功与否,都无需调用stopLocation()方法移除请求,定位sdk内部会移除  
    //启动定位  
    mlocationClient.startLocation();  
}

三:注意,预加载 GPS 并不是解决 Android 时间显示不准确的最佳方法 。虽然 GPS 可以提供精确的时间和位置信息,但在大多数情况下,并不需要通过 GPS 来确保时间的准确性。通常,Android 设备会自动从网络提供商(例如移动网络或Wi-Fi连接)获取准确的时间和时区信息。这个过程通常称为网络时间同步。因此,如果你的自定义 Launcher 在获取时间时不准确,很可能是由于设备的网络时间同步设置出现了问题,而不是时区设置的问题。

相关推荐
酷酷的阿云6 分钟前
不用ECharts!从0到1徒手撸一个Vue3柱状图
前端·javascript·vue.js
微信:137971205879 分钟前
web端手机录音
前端
齐 飞14 分钟前
MongoDB笔记01-概念与安装
前端·数据库·笔记·后端·mongodb
神仙别闹31 分钟前
基于tensorflow和flask的本地图片库web图片搜索引擎
前端·flask·tensorflow
GIS程序媛—椰子1 小时前
【Vue 全家桶】7、Vue UI组件库(更新中)
前端·vue.js
DogEgg_0011 小时前
前端八股文(一)HTML 持续更新中。。。
前端·html
ZL不懂前端2 小时前
Content Security Policy (CSP)
前端·javascript·面试
木舟10092 小时前
ffmpeg重复回听音频流,时长叠加问题
前端
王大锤43912 小时前
golang通用后台管理系统07(后台与若依前端对接)
开发语言·前端·golang
我血条子呢2 小时前
[Vue]防止路由重复跳转
前端·javascript·vue.js