Google Maps API v2上的随机NullPointerException

时间:2013-10-27 22:32:53

标签: java android google-maps android-fragments

我已经在我的Android应用上使用Google Maps API v2(版本3.2.25(761454-30)),有时它虽然有时候很有效(每次都会更频繁地使用)I&#39 ;我得到一个NullPointerException。我没有做任何与地图有太大不同的事情,并且在一些Nexus 4以及Nexus S上也发生过这种情况。虽然我在gmaps-api-issues项目上发现了issue我从那里得不到任何帮助。我在这里提出了一个问题,然后提供了一些细节,看看是否有人已经通过了这个问题。

正如我所说,我没有做任何不同的事情,就像他们在API上所建议的那样,我认为我可能做的唯一事情并非每个人都必须这样做(这可能会导致一些问题,idk )是:

  • 我在另一个Fragment,a.k.a。中使用SupportMapFragment作为嵌套片段。我也在使用ActionBar Sherlock,所以其他片段就是SherlockFragment。
  • 我在Fragment的onCreateView上实例化SupportMapFragment(而不是把它放在XML上),然后将它添加到我的布局上的FrameLayout。
  • 我将一些Y翻译应用到地图上(使用九个工作室),这样我就能获得与Foursquare应用程序相近的效果(尽管我在Foursquare上没有获得NPE)
  • 随着用户在应用上的进展,我创建并清除了一些标记(不应该是这样,因为错误发生在启动时,加载了framgent)

就是这么多,我不会把代码放在这里,因为它不像它只是一个不起作用的片段但是我已经列出了我所做的大部分内容这样做。这是堆栈跟踪(它总是几乎相同):

java.lang.NullPointerException
at java.nio.ReadWriteDirectByteBuffer.put(ReadWriteDirectByteBuffer.java:137)
at java.nio.ShortToByteBufferAdapter.put(ShortToByteBufferAdapter.java:163)
at maps.z.d.d(Unknown Source)
at maps.z.d.a(Unknown Source)
at maps.aq.a.a(Unknown Source)
at maps.aq.ao.b(Unknown Source)
at maps.aq.ao.a(Unknown Source)
at maps.v.g.a(Unknown Source)
at maps.v.g.b(Unknown Source)
at maps.p.p.l(Unknown Source)
at maps.p.p.run(Unknown Source)

1 个答案:

答案 0 :(得分:8)

好吧,我设法“修复”了这个问题。它实际上并不是一个修复程序,但至少我会避免在抛出异常时我的应用程序崩溃。我所做的是覆盖默认的未捕获异常处理程序,并禁止在未捕获的异常是这个时崩溃应用程序。崩溃是因为NullPointerException并且它总是发生在GLThread上(必须来自Google Maps的绘图代码)。因此,我检查是否所有先决条件都适用,如果是,我发送一个带有自定义操作的广播,这样我就可以在我使用Google Map的片段上收听该广播,当我得到一个时替换我的SupportMapFragment。由于我们得到了一个未被捕获的异常,因此GLThread被中断并且Google Maps Fragment的绘图被暂停,因此我们需要替换Google Maps Fragment以使地图重新开始工作。

以下是我必须添加的代码:

在我的应用程序的onCreate上:

final UncaughtExceptionHandler defaultHandler = Thread.getDefaultUncaughtExceptionHandler();
Thread.setDefaultUncaughtExceptionHandler(new UncaughtExceptionHandler() {
    @Override
    public void uncaughtException(Thread thread, Throwable ex) {
        if (ex instanceof NullPointerException && thread.getName().startsWith("GLThread")) {
            for (StackTraceElement stackTraceElement : ex.getStackTrace()) {
                if (stackTraceElement.getClassName().contains("maps")) {
                    sendBroadcast(new Intent(ACTION_MAPS_NPE));
                    return;
                }
            }
        }
        defaultHandler.uncaughtException(thread, ex);
    }
});

然后在我使用Google Maps Fragment的Fragment上,我在将SupportMapFragment添加到onCreateView上的布局之前注册了这个广播接收器(并在onDestroyView上取消注册):

private class GoogleMapsNPEReceiver extends BroadcastReceiver {

    @Override
    public void onReceive(Context context, Intent intent) {
        Fragment mapFragment = getChildFragmentManager().findFragmentById(R.id.map_fragment_container);
        if (mapFragment != null) {
            getChildFragmentManager().beginTransaction()
                    .replace(R.id.map_fragment_container, SupportMapFragment.newInstance())
                    .commit();
            mMapInitialized = false;
            // We won't have onResume to initialize our map anymore. Try to initialize it after 100ms.
            final Handler handler = new Handler();
            handler.postDelayed(new Runnable() {
                @Override
                public void run() {
                    if (getGoogleMap() == null) {
                        handler.postDelayed(this, 100);
                    }
                }
            }, 100);
        }
    }
}

希望这可以帮助任何有同样问题的人。如果有人有实际的解决方法,请告诉我。