在 Android 上获取用户当前位置的最简单、最可靠的方法是什么?

2024-01-09

The LocationManager对于只需要偶尔粗略估计用户位置的应用程序来说,Android 上的 API 似乎使用起来有点痛苦。

我正在开发的应用程序本身并不是真正的位置应用程序,但它确实需要获取用户的位置才能显示附近企业的列表。它不需要担心用户是否正在走动或类似的事情。

这是我想做的:

  1. 向用户显示附近位置的列表。
  2. 预加载用户的位置,以便在我需要时ActivityX,它将可用。
  3. 我并不特别关心更新的准确性或频率。只要抓住一个位置就足够了,只要它不是太远。也许如果我想变得更花哨,我会每隔几分钟左右更新一次位置,但这不是一个重要的优先事项。
  4. 适用于任何具有 GPS 或网络位置提供商的设备。

似乎这不应该那么难,但在我看来,我必须启动两个不同的位置提供商(GPS 和网络)并管理每个位置的生命周期。不仅如此,我还必须在多个活动中复制相同的代码才能满足#2。我尝试过使用getBestProvider()过去,我们将解决方案简化为仅使用一个位置提供商,但这似乎只能为您提供最佳的“理论”提供商,而不是实际上会给您提供最佳结果的提供商。

有没有更简单的方法来实现这一点?


这就是我所做的:

  1. 首先,我检查启用了哪些提供商。有些可能在设备上禁用,有些可能在应用程序清单中禁用。
  2. 如果有任何提供程序可用,我会启动位置侦听器和超时计时器。在我的例子中是 20 秒,对于 GPS 来说可能不够,所以你可以放大它。
  3. 如果我从位置侦听器获取更新,我将使用提供的值。我停止听众和计时器。
  4. 如果我没有得到任何更新并且计时器已过,我必须使用最后已知的值。
  5. 我从可用的提供者那里获取最后已知的值,并选择其中最新的值。

这是我使用我的课程的方式:

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

这是 MyLocation 类:

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);
    }
}

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

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

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

在 Android 上获取用户当前位置的最简单、最可靠的方法是什么? 的相关文章

随机推荐

  • FFMPEG - 将 FFPROBE 的持续时间输出存储为变量

    我想在 enable Between t 0 0 字段中使用视频持续时间 以确保每次都以视频持续时间的 1 3 覆盖图像 根据 FFMPEG 在他们网站上的信息 我需要使用这个 ffprobe v error show entries fo
  • 自 Meteor 升级到 0.8.0 以来,当会话变量依赖关系发生变化时,不会触发模板“渲染”回调

    自从升级到 0 8 0 以来 我遇到了一个问题 渲染的模板不再被触发 除了第一次 我遵循了以下建议 https github com avital meteor ui new rendered callback blob master ne
  • GPS坐标精度

    我有一些 GPS 坐标的数据 最多有 9 位小数 例子是 39 1234567891 75 1987654321 我需要处理这些数据以获得该人所在位置的背景信息 但为了简单起见 我想将 100m 内包含的坐标视为单个位置 任何人都可以指出任
  • 代码优先:将实体映射到现有数据库表

    我在现有数据库中使用实体框架 6 代码优先 但在将实体映射到数据库表时遇到问题 通常 我会使用数据库优先的方法并生成我的实体和上下文代码 但使用设计器已成为一个巨大的痛苦 我已设置 Database SetInitializer null
  • iOS 上位置固定元素的滚动问题

    我正在构建一个移动项目 其中有许多模块 其元素定位为固定 所面临的问题仅存在于 iOS 上运行的浏览器上 确切的问题是 每当我滚动页面的主体时 例如底部工具栏 作为固定的 整个固定元素分别随滚动移动 一旦滚动完全结束 那么只有它回到原来的位
  • 根据对象在箱线图中排序

    我想在 pandas df 中有一些分组箱线图 var2 是一个对象 我想根据 var2 顺序按字母顺序显示箱线图 import seaborn as sns sns set style whitegrid ax sns boxplot x
  • 如何通过 PHP 的邮件功能发送包含阿拉伯语内容的电子邮件?

    我在使用 PHP 的邮件功能发送包含阿拉伯语内容的电子邮件时遇到了挑战 假设我有这个简单的阿拉伯字符串 我尝试了几种利用标题的方法 但电子邮件内容最终仍然是这样的 X X1X X1Y X 但是 如果我使用阿拉伯字符 电子邮件主题会被正确编码
  • 在四开图标题中将“图”文本加粗

    默认情况下 生成 html 时 quarto https quarto org 将图的标题添加到自动生成的文本 Figure NN 中 其中 NN 是图的编号 例如 以下 R 块 r label myFirstFigure fig cap
  • 无法在 CentOS 上安装 Ruby

    我只是无法在我的 CentOS 也托管 cPanel 上安装 Ruby sudo yum install ruby ruby devel ruby irb ruby rdoc ruby ri audit log user command 连
  • 有没有办法忽略 Django 中的缓存错误?

    我刚刚将我们的开发 Django 站点设置为使用 redis 作为缓存后端 一切正常 我关闭了 redis 来看看会发生什么 果然由于缓存后端行为而出现了 Django 404 连接被拒绝 或者出现各种其他错误 有什么方法可以指示 Djan
  • 用空字符串初始化所有字符串成员

    我想将对象的所有 String 成员设置为空字符串 如果它们为空 伪代码 foreach member in object if member instanceof String and member null member 实现这一目标的
  • 禁用 Flex CSS 类型选择器警告?

    我正在构建一个较大的 Flex 项目 其中包含多个模块 生成多个 SWF 的单个 Flex 项目 现在 我有一个 css 文件 正在主 SWF 标记中加载
  • 使用 JavaScript 更改输入字段的背景颜色

    我正在制作一个表单 并将输入字段设置为仅使用 JavaScript 进行读取 我想将只读属性的默认颜色更改为绿色或黄色 HTML
  • Xcode 6:如何禁用优化 (Swift)

    调试期间不会触发断点 我发现这是因为优化了代码 如何禁用优化 我浏览了每个工具栏 我能找到的最接近的选项是Product gt Perform Action gt Generate Optimisation Profile Xcode 说
  • 读取 JSF Web 应用程序静态代码中的 .properties 文件

    我想从静态块中的属性文件获取数据库连接参数 属性文件位置是WEB INF classes db properties 我会更喜欢使用getResourceAsStream 方法 我尝试了很多方法 但都返回了null private stat
  • 返回日期没有返回小时和分钟

    这是我的代码 CREATE OR REPLACE FUNCTION GET DATE P DAYS IN NUMBER RETURN DATE AS V DATE DATE BEGIN V DATE SYSDATE P DAYS DBMS
  • Odata Url 降序转换

    我需要按患者的 Visidate 降序获取数据 所以我尝试了这样的网址 192 168 1 105 33396 FalconCPDataService svc DEPhysicians format json expand DEPatien
  • NUnit:字典断言

    我想要一个一班轮 在 NUnit 中 断言两个字典是否相同 即 我想要以下代码的简洁版本 public static void DictionaryAssert
  • WPF:触发动画 - 沿 Y 轴向下移动控件

    我有一个想要向下移动的控件MouseOver 不幸的是 当我将鼠标悬停在上面时 动画不会运行 但背景颜色会改变 Toolkit Dialog是一个继承的自定义对话框控件ContentControl 另外 在最终实现中我不会使用 MouseO
  • 在 Android 上获取用户当前位置的最简单、最可靠的方法是什么?

    The LocationManager对于只需要偶尔粗略估计用户位置的应用程序来说 Android 上的 API 似乎使用起来有点痛苦 我正在开发的应用程序本身并不是真正的位置应用程序 但它确实需要获取用户的位置才能显示附近企业的列表 它不