9

我的应用程序的一位用户报告说,即使他确实打开了它,应用程序也会告诉网络位置已关闭。他给我发了几个屏幕截图,它们让我思考;

LocationManager.isProviderEnabled(LocationManager.NETWORK_PROVIDER)

无法正常工作。他的手机运行的是 Android 4.1.2,首先我认为是导致此问题的原因。但事实并非如此。他也给我发了一张该设置的屏幕截图。

然后我用谷歌搜索并找到了这个。这个问题似乎对我有帮助,但不幸的是,答案对这种情况没有帮助,提问者没有进一步追问。

我的应用程序与位置有关,并且一直在使用 LocationManager.isProviderEnabled 来了解 GPS 和网络的位置是打开还是关闭。直到最近,我才被告知我的应用程序没有正确了解这些设置。他是第一个报告此问题的用户。通过查看 Secure.LOCATION_PROVIDERS_ALLOWED,我了解到还有另一种方法可以了解 GPS 和网络的位置设置。为了看看这个方法在他的手机上是如何工作的,我写了一个简单的应用程序并让他运行。这个应用程序执行简单的任务并在屏幕上显示文本。

LocationManager locationManager = (LocationManager)getSystemService(Context.LOCATION_SERVICE);
if(locationManager.isProviderEnabled(LocationManager.GPS_PROVIDER))
{
    string = "GPS=on\n";
}
else
{
    string = "GPS=off\n";
}
if(locationManager.isProviderEnabled(LocationManager.NETWORK_PROVIDER))
{
    string += "Network=on\n";
}
else
{
    string += "Network=off\n";
}
String status = android.provider.Settings.Secure.getString(getContentResolver(), Secure.LOCATION_PROVIDERS_ALLOWED);
if(status.contains("gps"))
{
    string += "GPS=on\n";
}
else
{
    string += "GPS=off\n";
}
if(status.contains("network"))
{
    string += "Network=on\n";
}
else
{
    string += "Network=off\n";
}

他再次发回屏幕截图。它看起来;

GPS=on
Network=off
GPS=on
Network=on

这个结果并没有让我高兴。这可能有一些可能性。

  • 正如其他人之前所质疑的那样,这个问题已经出现在一些手机上。
  • 谷歌用 4.1.2 打破了这一点。isProviderEnabled 不适用于此版本。
  • 虽然没有记录,但从 4.1.2 开始,isProviderEnabled 将无法像以前那样工作。
  • 不,谷歌改变了一切。这是此特定手机的错误。

现在我的问题是;

  1. LocationManager.isProviderEnabled 对 Android 4.1.2 及更高版本仍然有效吗?
  2. 看到 Secure.LOCATION_PROVIDERS_ALLOWED 是否有一些缺点/坑洞(当我放弃使用 LocationManager.isProviderEnabled 时?

提前致谢。

编辑1:

在这里,您可以从 Google Play 下载测试应用程序以尝试或请他人尝试。

编辑6:

自从回答了这个问题后,我删除了测试应用程序。

编辑2:

我发布了我的应用程序,该应用程序通过查看 Secure.LOCATION_PROVIDERS_ALLOWED 来检查网络提供商是否可用,并在有限的手机上出现异常。这些是 ACRA 的报告。

一些运行 OS 4.1.1 的手机。

java.lang.IllegalArgumentException: requested provider network doesn't exisit
    at android.os.Parcel.readException(Parcel.java:1434)
    at android.os.Parcel.readException(Parcel.java:1384)
    at android.location.ILocationManager$Stub$Proxy.requestLocationUpdates(ILocationManager.java:675)
    at android.location.LocationManager._requestLocationUpdates(LocationManager.java:686)
    at android.location.LocationManager.requestLocationUpdates(LocationManager.java:508)

一些运行 OS 4.1.2 的手机。

java.lang.IllegalArgumentException: provider=network
    at android.os.Parcel.readException(Parcel.java:1439)
    at android.os.Parcel.readException(Parcel.java:1389)
    at android.location.ILocationManager$Stub$Proxy.requestLocationUpdates(ILocationManager.java:659)
    at android.location.LocationManager._requestLocationUpdates(LocationManager.java:690)
    at android.location.LocationManager.requestLocationUpdates(LocationManager.java:512)

在我更改了一种方法来检查网络提供商的位置是否可用之前,我从未见过这些异常。所以我认为 LocationManager.isProviderEnabled 是安全的,看到 Secure.LOCATION_PROVIDERS_ALLOWED 是有风险的。但这会让我回到原来的问题。为什么当 Secure.LOCATION_PROVIDERS_ALLOWED 告诉存在时 LocationManager.isProviderEnabled(LocationManager.NETWORK_PROVIDER) 返回 false(实际上并没有)。Android 操作系统是否设计不佳?或者我刚刚看到仅与特定(但至少有 2 个)手机相关的问题?

编辑3:

我更新了测试应用程序,通过 requestLocationUpdates() 访问显示 GPS/网络位置提供程序似乎真的可用或不可用。

我透露了 2 部手机的名称。

1) SBM200SH, OS4.1.2, Softbank mobile, Sharp Corporation
2) HTX21 (INFOBAR A02), OS4.1.1, KDDI, HTC

编辑4:

我找到了第三部手机。

3)SBM203SH,OS4.1.2,软银手机,夏普公司

编辑5:

夏普公司正在为移动开发者提供讨论空间。我通过提出这个 SO 的问题来发布主题。我希望夏普公司有人为此采取行动。我会保持更新。

4

4 回答 4

6

夏普公司提供的开发人员支持非常好,他们在不到 48 小时内回答了我的问题。

这就是我从他们那里得到的。

LocationManager.isProviderEnabled(LocationManager.NETWORK_PROVIDER) 返回 true 必须满足 2 个条件。

  1. 一些内部状态已准备好进行网络定位。
  2. 网络位置在设置屏幕上启用。

第二个很明显。但第一个不是。他们告诉如何模拟第一个是否定的。您可以通过下面显示的步骤确认问题并运行我的测试应用程序(请参阅我的问题以获取下载链接)。

  • 打开手机的设置。
  • 点击应用程序。
  • 点击全部选项卡。
  • 找到“网络位置”,点击它。
  • 点击“禁用”。
  • 重启你的手机。
  • 运行测试应用程序。

出于某种原因,我无法理解用户的手机未能执行与上面显示的第一个条件相关的操作并出现问题。

结论:

LocationManager.isProviderEnabled(LocationManager.NETWORK_PROVIDER) 是可靠的。请注意,Secure.LOCATION_PROVIDERS_ALLOWED 不太可靠。

于 2013-05-31T16:07:12.303 回答
4

LOCATION_MODE检查用户位置设置的现代方法是通过Settings.Secure

例如,如果您只是想知道用户是否禁用了它们,您可以执行以下操作:

   public static boolean isLocationEnabled(Context context) {
       return getLocationMode(context) != Settings.Secure.LOCATION_MODE_OFF;
   }

   private static int getLocationMode(Context context) {
       return Settings.Secure.getInt(context.getContentResolver(), Settings.Secure.LOCATION_MODE, Settings.Secure.LOCATION_MODE_OFF);
   }

如果启用了位置,这将返回 true。如果您需要更精细的粒度,请参阅文档了解详细信息。

此方法比旧的 NETWORK_PROVIDER 和 GPS_PROVIDER 方法更适合使用 Google 服务位置 API。注意:需要 KitKat / API19

于 2014-12-19T17:06:59.603 回答
2

不是直接回答您的问题,而是查看 Google 上周推出的新 Location API。这真的很容易实施,它会在不浪费电池的情况下检查最佳位置。 http://developer.android.com/google/play-services/location.html 这是 Google I/O 上关于这个新 API 以及如何使用它的会话。 http://www.youtube.com/watch?v=Bte_GHuxUGc

这样您就不必担心检查 GPS 是否打开以及类似的事情

于 2013-05-27T13:56:42.773 回答
0

位置管理器在某些手机上不可靠。您可能会注意到,如果您突然启动 google 地图,您的应用程序可以正常工作。那是因为谷歌地图踢了 LocationManager。这也意味着有程序化的方式可以活活踢那个家伙。所以我用

HomeScreen.getLocationManager().requestLocationUpdates(
    LocationManager.NETWORK_PROVIDER, 0, 0, new LocationListener() {
        @Override
        public void onStatusChanged(String provider, int status, Bundle extras) {
        }
        @Override
        public void onProviderEnabled(String provider) {
        }
        @Override
        public void onProviderDisabled(String provider) {
        }
        @Override
        public void onLocationChanged(final Location location) {
        }
    });

在上面的代码之后,我从 LocationManager 调用了我需要的东西,它有点工作。如果没有尝试新的 API 的LocationClient. 这应该会更好,电池,准确性和可靠性。

于 2013-05-29T14:06:25.110 回答