Android GPS坐标分散

时间:2013-01-07 18:43:32

标签: android geolocation gps

我在下面的代码似乎工作正常:

    // GPS Reader
    // Criteria
    Criteria criteria = new Criteria();
    //criteria.setAccuracy(Criteria.ACCURACY_COARSE); // Used when we can't get a GPS Fix
    criteria.setAccuracy(Criteria.ACCURACY_FINE); // Used when we can get a GPS Fix

    criteria.setAltitudeRequired(false);
    criteria.setBearingRequired(false);
    criteria.setCostAllowed(true);
    criteria.setPowerRequirement(Criteria.POWER_LOW);

    LocationListener locationListener = new MyLocationListener();
    LocationManager lm = (LocationManager) getSystemService(Context.LOCATION_SERVICE);
    lm.requestLocationUpdates(lm.getBestProvider(criteria, true), 0, 0,
            locationListener);

然而,当我绘制我在谷歌地图上收集的数据时,GPS坐标在某些情况下非常分散,或者它将沿着我走的路径发现,然后突然跳到一个点一英里之外然后回来。有没有什么办法解决这一问题?我估计会进行某种准确性检查吗?

更新

基本上我的问题就像一个例子 - GPS Jitter

更新2

我认为这不是一个赏金,但我想我也可以完全理解这里发生了什么,看看我的方法是否可能过度杀戮。我仍然有同样的问题,我的坐标在它们中有抖动,尽管事实上我有3米的准确度等等。现在这可能是有用的卫星等我不知道,但基本上我是想了解如何所有这些其他应用程序,尤其是练习应用程序,能够在相同的情况下获得如此平滑的读数。

我在Quora上并且能够找到这个Does any running app have a feature of filtering GPS data to get more accurate tracking?不幸的是它没有对我的问题有太多了解,除非你可以使用卡尔曼滤波器,但肯定必须有更简单的手段因为我怀疑大多数应用程序都在实现这一点。

无论如何,如果一位开发人员希望与一些非常感谢的伪代码分享他们正在做的事情。我的意思是,如果我坚持卡尔曼,但我确信必须更容易实现算法,并希望学习这些以及如何实现它们,这是合适的。

背景信息:这是一个移动行人应用程序。

我试图从收集信息的相关SO问题 Create a smooth curve from a series of GPS coordinates Smooth gps data:这是一个好的开始,虽然我不确定我需要实现什么伪代码来正确地使最小二乘拟合才能正常工作,这样我就可以得到一个样条GPS数据,然后我可以查看某些东西像谷歌地图确认我做得对。我认为问题是如果这是我正在处理的一般X和Y数据,而不是GEO坐标,我可以在matlab中编写一些测试它,然后继续。

更新3

我收到的混乱的GPS数据的图像 https://www.dropbox.com/s/ilsf8snao2no65e/gpsdata2.png

代码 https://gist.github.com/4505688

4 个答案:

答案 0 :(得分:6)

GPS数据本身容易出错。即使您已正确设置了所有LocationManager(并且看起来像这样),您也会看到位置偶尔出现抖动。

<强>精度

有一点需要注意的是,GPS精度只能估算出接收GPS信号的计算效果。系统误差可以为您提供良好的准确度数字(<10m),但仍然表示较大的位置误差。这就是为什么即使您的准确度非常好也会看到抖动的原因。准确度测量可以很好地消除非常大的误差(> 100m),但在较低的水平上,它只是给出了计算收敛的指示。

<强>过滤

像许多事情一样,过滤最好是尽可能少。精度过滤器应该能够消除大的错误,如果最终得到一个好的阈值,您可能能够获得平滑的数据。

如果为运行应用程序的位置更改设置阈值,您可能也会获得一些好处。例如,跑步者只能在一段时间内覆盖一定距离,并且通过设置上限阈值(Usain Bolt Cutoff),您应该能够移除坏点。问题是如果你的第一个点是错误,你最终会删除所有其他点。

卡尔曼过滤器

卡尔曼滤波器是一个很好的解决方案,我已经在我正在开发的导航应用上实现了它。结果非常合理,甚至可以在GPS不良或不可用的情况下进行有限的航位推算。不幸的是我无法分享源代码,但如果您决定采用这种方式,我可以给您一些指导。最好的结果来自6 DoF滤波器,您可以在其中计算加速度和速度,并使用它来估算位置。这不是最简单的解决方案,但我们已经从中看到了良好的结果。

<强>最小二乘

卡尔曼滤波器很棒,因为它可以实时用于过滤位置。它跟踪自己的状态,您不需要存储旧位置。但另一方面,如果你想对你的路线进行后期处理,最小二乘拟合是一种最佳方式。 (卡尔曼来自LSQ公式)。我没有做很多后期处理,但我怀疑我可以在上面挖掘一些旧的教科书。然而,理论应该是相同的。

大多数GPS设备相当不错,从我看过的所有测试中我都不会经常看到您在示例中看到的抖动。然而,我实施卡尔曼滤波器的一大优势和原因是您的行进距离和速度计算更准确

答案 1 :(得分:4)

您得到的是因为您的标准将最佳提供者标识为NETWORK_PROVIDER。 这不能识别您的位置,而是为您提供其范围涵盖您设备的手机信号塔的位置。因此,当您在任何建筑物之外时,最好使用GPS_PROVIDER来获得精确的地理位置准确度。为什么散乱的坐标是因为你的设备来到另一个蜂窝塔的范围,因此位置跳跃发生。直接使用GPS_PROVIDER是您将获得的最佳解决方案。

答案 2 :(得分:2)

查看Reto Meier的this文章(包含代码)。它应该给你很多信息来思考。您至少应该设置最小距离和时间参数。

lm.requestLocationUpdates(lm.getBestProvider(criteria, true), 5000, 20,
        locationListener);

此外,您可以组合多个提供商,以获得您目前可以获得的最佳坐标。祝你好运!

答案 3 :(得分:2)

一种简单的方法是丢弃任何旧的或不准确的数据,并保持经度和纬度的平均值。

private static final long WILDLY_OUT = 15;
private static final long TOO_OLD = 30000;
private static final long NO_SAMPLES = 5;

double lastLocationTime;

double calclongitude(Location location, double oldLongitude) {
    double newLongitude = oldLongitude;

    if (location.getAccuracy() < WILDLY_OUT) {
        newLongitude = (NO_SAMPLES * oldLongitude + location
                .getLongitude()) / (NO_SAMPLES + 1);
        lastLocationTime = System.currentTimeMillis();
    }

    if (lastLocationTime > TOO_OLD) {
        newLongitude = location.getLongitude();
    }

    return newLongitude;
}

为纬度做同样的事。