Android上的LocationManagerAPI对于一个只需要偶尔粗略地了解用户位置的应用程序来说似乎有点痛苦.

我正在开发的应用程序本身并不是一个定位应用程序,但它确实需要获得用户的位置才能显示附近企业的列表.它不需要担心用户是否在四处走动或诸如此类的事情.

以下是我想做的事情:

  1. 向用户显示附近位置的列表.
  2. 预加载用户的位置,这样当我需要它时,它将在Activity倍可用.
  3. 我并不特别关心更新的准确性或频率.只要不远,只要抓住一个位置就足够了.也许如果我想成为一名花花公子,我会每隔几分钟左右更新一次位置,但这不是一个很大的优先事项.
  4. 适用于任何设备,只要它有GPS或网络位置Provider .

这看起来不应该那么难,但在我看来,我必须启动两个不同的位置Provider (GPS和网络),并管理每个Provider 的生命周期.不仅如此,我还必须在多个活动中复制相同的代码才能满足第二个要求.我在过go try 过使用getBestProvider()来将解决方案减少到只使用一个位置提供程序,但这似乎只给出了最好的"理论"提供程序,而不是实际上会给你带来最好结果的提供程序.

有没有更简单的方法来完成这件事?

推荐答案

我是这样做的:

  1. 首先,我判断启用了哪些提供者.有些可能在设备上被禁用,有些可能在应用程序 list 中被禁用.
  2. 如果有任何提供程序可用,我会启动位置侦听程序和超时计时器.在我的例子中是20秒,对GPS来说可能不够,所以你可以放大它.
  3. 如果我从Location Listener获得更新,我将使用提供的值.我停止监听程序和计时器.
  4. 如果我没有得到任何更新,并且计时器超时,我必须使用最后已知的值.
  5. 我从可用的提供程序中获取最近已知的值,并 Select 其中最新的值.

以下是我如何使用我的课堂:

LocationResult locationResult = new LocationResult(){
    @Override
    public void gotLocation(Location location){
        //Got the location!
    }
};
MyLocation myLocation = new MyLocation();
myLocation.getLocation(this, locationResult);

下面是我的位置课:

import java.util.Timer;
import java.util.TimerTask;
import android.content.Context;
import android.location.Location;
import android.location.LocationListener;
import android.location.LocationManager;
import android.os.Bundle;

public class MyLocation {
    Timer timer1;
    LocationManager lm;
    LocationResult locationResult;
    boolean gps_enabled=false;
    boolean network_enabled=false;

    public boolean getLocation(Context context, LocationResult result)
    {
        //I use LocationResult callback class to pass location value from MyLocation to user code.
        locationResult=result;
        if(lm==null)
            lm = (LocationManager) context.getSystemService(Context.LOCATION_SERVICE);

        //exceptions will be thrown if provider is not permitted.
        try{gps_enabled=lm.isProviderEnabled(LocationManager.GPS_PROVIDER);}catch(Exception ex){}
        try{network_enabled=lm.isProviderEnabled(LocationManager.NETWORK_PROVIDER);}catch(Exception ex){}

        //don't start listeners if no provider is enabled
        if(!gps_enabled && !network_enabled)
            return false;

        if(gps_enabled)
            lm.requestLocationUpdates(LocationManager.GPS_PROVIDER, 0, 0, locationListenerGps);
        if(network_enabled)
            lm.requestLocationUpdates(LocationManager.NETWORK_PROVIDER, 0, 0, locationListenerNetwork);
        timer1=new Timer();
        timer1.schedule(new GetLastLocation(), 20000);
        return true;
    }

    LocationListener locationListenerGps = new LocationListener() {
        public void onLocationChanged(Location location) {
            timer1.cancel();
            locationResult.gotLocation(location);
            lm.removeUpdates(this);
            lm.removeUpdates(locationListenerNetwork);
        }
        public void onProviderDisabled(String provider) {}
        public void onProviderEnabled(String provider) {}
        public void onStatusChanged(String provider, int status, Bundle extras) {}
    };

    LocationListener locationListenerNetwork = new LocationListener() {
        public void onLocationChanged(Location location) {
            timer1.cancel();
            locationResult.gotLocation(location);
            lm.removeUpdates(this);
            lm.removeUpdates(locationListenerGps);
        }
        public void onProviderDisabled(String provider) {}
        public void onProviderEnabled(String provider) {}
        public void onStatusChanged(String provider, int status, Bundle extras) {}
    };

    class GetLastLocation extends TimerTask {
        @Override
        public void run() {
             lm.removeUpdates(locationListenerGps);
             lm.removeUpdates(locationListenerNetwork);

             Location net_loc=null, gps_loc=null;
             if(gps_enabled)
                 gps_loc=lm.getLastKnownLocation(LocationManager.GPS_PROVIDER);
             if(network_enabled)
                 net_loc=lm.getLastKnownLocation(LocationManager.NETWORK_PROVIDER);

             //if there are both values use the latest one
             if(gps_loc!=null && net_loc!=null){
                 if(gps_loc.getTime()>net_loc.getTime())
                     locationResult.gotLocation(gps_loc);
                 else
                     locationResult.gotLocation(net_loc);
                 return;
             }

             if(gps_loc!=null){
                 locationResult.gotLocation(gps_loc);
                 return;
             }
             if(net_loc!=null){
                 locationResult.gotLocation(net_loc);
                 return;
             }
             locationResult.gotLocation(null);
        }
    }

    public static abstract class LocationResult{
        public abstract void gotLocation(Location location);
    }
}

有人可能还想修改我的逻辑.例如,如果您从网络Provider 处获得更新,不要停止侦听器,而是继续等待.GPS提供了更精确的数据,因此值得等待.如果计时器已过,并且您已从网络获得更新,但未从GPS获得更新,则您可以使用网络提供的值.

另一种方法是使用LocationClient http://developer.android.com/training/location/retrieve-current.html.但它要求在用户设备上安装Google Play Services apk.

Android相关问答推荐

Jetpack编写日期 Select 器

将HTML约束布局转换为Jetpack编写约束布局-链样式、偏差

如何在Reaction Native中显示Google Map iFrame?

Android 14上的慢速意图广播交付

穿戴与iPhone连接的安卓操作系统

如何从sqlite数据库中检索数据到碎片android?

Android Kotlin ImageView内置于Kotlin ImageView中.适配器未按预期更新

Modifer.Align()不适用于行/列/框中的文本.未解决的作用域实例错误

Play Google上发布的一款应用的房间数据库迁移

插入视图模型时,dagger 未命中绑定错误

从我的 Android 应用程序发送到 Gin 时失败,但从 Postman 发送到 Gin 时成功

Android 构建失败:找不到 flexbox2.0.1.aar

在 kotlin 中动态添加 GridView

React Native Android 应用程序在调试模式下运行良好,但当我们发布 apk 时,它会生成旧版本的应用程序

如何关闭可组合对话框?

自定义 Recyclerview [Android & Kotlin]

如何从日期 Select 器计算年龄?

Unity:Android 上随机接近零的 FPS 下降(提供了很多线索)

复用 RecyclerView 适配器,避免不必要的 API 调用

Google Play 控制台您的应用是否使用广告 ID?