当前位置: 首页 > 知识库问答 >
问题:

如何在Android上使用Mapsforge修复本机SIGABRT

水焱
2023-03-14

我使用的是MapsForge0.6.1,我可以用以下简化的示例来再现这个问题,它基本上是本教程的副本:https://github.com/Mapsforge/Mapsforge/blob/master/docs/gett-start-android-app.md

我做了一些小改动以配合新版本。

package org.test.mapsforgeexample;

import java.io.File;

import org.mapsforge.core.model.LatLong;
import org.mapsforge.map.android.graphics.AndroidGraphicFactory;
import org.mapsforge.map.android.util.AndroidUtil;
import org.mapsforge.map.android.view.MapView;
import org.mapsforge.map.layer.cache.TileCache;
import org.mapsforge.map.layer.renderer.TileRendererLayer;
import org.mapsforge.map.datastore.MapDataStore;
import org.mapsforge.map.reader.MapFile;
import org.mapsforge.map.rendertheme.InternalRenderTheme;

import android.app.Activity;
import android.os.Bundle;
import android.os.Environment;

public class MainActivity extends Activity {

    private static final String MAP_FILE = "berlin.map";

    private MapView mapView;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        AndroidGraphicFactory.createInstance(this.getApplication());

        this.mapView = new MapView(this);
        setContentView(this.mapView);

        this.mapView.setClickable(true);
        this.mapView.getMapScaleBar().setVisible(true);
        this.mapView.setBuiltInZoomControls(true);
        this.mapView.getMapZoomControls().setZoomLevelMin((byte) 10);
        this.mapView.getMapZoomControls().setZoomLevelMax((byte) 20);

        TileCache tileCache = AndroidUtil.createTileCache(this, "mapcache",
                mapView.getModel().displayModel.getTileSize(), 1f,
                this.mapView.getModel().frameBufferModel.getOverdrawFactor());

        MapDataStore mapDataStore = new MapFile(new File(Environment.getExternalStorageDirectory(), MAP_FILE));
        TileRendererLayer tileRendererLayer = new TileRendererLayer(tileCache, mapDataStore,
                this.mapView.getModel().mapViewPosition, AndroidGraphicFactory.INSTANCE);
        tileRendererLayer.setXmlRenderTheme(InternalRenderTheme.OSMARENDER);

        this.mapView.getLayerManager().getLayers().add(tileRendererLayer);

        this.mapView.setCenter(new LatLong(52.517037, 13.38886));
        this.mapView.setZoomLevel((byte) 12);
    }

    @Override
    protected void onDestroy() {
        this.mapView.destroyAll();
        AndroidGraphicFactory.clearResourceMemoryCache();
        super.onDestroy();
    }
}
12-16 11:43:34.055 1496-1496/? A/DEBUG: *** *** *** *** *** *** *** *** *** *** *** *** *** *** *** ***
12-16 11:43:34.056 1496-1496/? A/DEBUG: Build fingerprint: 'google/bullhead/bullhead:7.1.1/NMF26F/3425388:user/release-keys'
12-16 11:43:34.056 1496-1496/? A/DEBUG: Revision: 'rev_1.0'
12-16 11:43:34.056 1496-1496/? A/DEBUG: ABI: 'arm64'
12-16 11:43:34.056 1496-1496/? A/DEBUG: pid: 1425, tid: 1449, name: pool-4-thread-1  >>> org.test.mapsforgeexample <<<
12-16 11:43:34.056 1496-1496/? A/DEBUG: signal 6 (SIGABRT), code -6 (SI_TKILL), fault addr --------
12-16 11:43:34.058 1496-1496/? A/DEBUG: Abort message: 'Error, cannot access an invalid/free'd bitmap here!'
12-16 11:43:34.058 1496-1496/? A/DEBUG:     x0   0000000000000000  x1   00000000000005a9  x2   0000000000000006  x3   0000000000000008
12-16 11:43:34.058 1496-1496/? A/DEBUG:     x4   0000000000000000  x5   0000000000000000  x6   00000078e82a2000  x7   0000000000000000
12-16 11:43:34.058 1496-1496/? A/DEBUG:     x8   0000000000000083  x9   ffffffffffffffdf  x10  0000000000000000  x11  0000000000000001
12-16 11:43:34.058 1496-1496/? A/DEBUG:     x12  0000000000000018  x13  0000000000000000  x14  0000000000000000  x15  003627f486f8cd75
12-16 11:43:34.058 1496-1496/? A/DEBUG:     x16  00000078e72d5ee0  x17  00000078e727faac  x18  0000000007e0f81f  x19  00000078c97ff4f8
12-16 11:43:34.058 1496-1496/? A/DEBUG:     x20  0000000000000006  x21  00000078c97ff450  x22  000000000000000b  x23  00000000138bedd0
12-16 11:43:34.058 1496-1496/? A/DEBUG:     x24  00000000138c1300  x25  00000078d79047e0  x26  00000078c8e4c720  x27  0000000013b99b98
12-16 11:43:34.058 1496-1496/? A/DEBUG:     x28  00000000138c12f0  x29  00000078c97fe320  x30  00000078e727ced8
12-16 11:43:34.058 1496-1496/? A/DEBUG:     sp   00000078c97fe300  pc   00000078e727fab4  pstate 0000000060000000
12-16 11:43:34.549 1496-1496/? A/DEBUG: backtrace:
12-16 11:43:34.549 1496-1496/? A/DEBUG:     #00 pc 000000000006bab4  /system/lib64/libc.so (tgkill+8)
12-16 11:43:34.549 1496-1496/? A/DEBUG:     #01 pc 0000000000068ed4  /system/lib64/libc.so (pthread_kill+64)
12-16 11:43:34.549 1496-1496/? A/DEBUG:     #02 pc 0000000000023f58  /system/lib64/libc.so (raise+24)
12-16 11:43:34.549 1496-1496/? A/DEBUG:     #03 pc 000000000001c810  /system/lib64/libc.so (abort+52)
12-16 11:43:34.549 1496-1496/? A/DEBUG:     #04 pc 0000000000010c24  /system/lib64/libcutils.so (__android_log_assert+224)
12-16 11:43:34.549 1496-1496/? A/DEBUG:     #05 pc 00000000001073e4  /system/lib64/libandroid_runtime.so (_ZNK7android6Bitmap11assertValidEv+40)
12-16 11:43:34.549 1496-1496/? A/DEBUG:     #06 pc 00000000001074c8  /system/lib64/libandroid_runtime.so (_ZN7android6Bitmap11getSkBitmapEP8SkBitmap+20)
12-16 11:43:34.550 1496-1496/? A/DEBUG:     #07 pc 000000000010359c  /system/lib64/libandroid_runtime.so
12-16 11:43:34.550 1496-1496/? A/DEBUG:     #08 pc 0000000001a9a3a0  /system/framework/arm64/boot-framework.oat (offset 0x1686000) (android.graphics.Canvas.nativeDrawBitmapMatrix+172)
12-16 11:43:34.550 1496-1496/? A/DEBUG:     #09 pc 0000000001a9e9b8  /system/framework/arm64/boot-framework.oat (offset 0x1686000) (android.graphics.Canvas.drawBitmap+148)
12-16 11:43:34.550 1496-1496/? A/DEBUG:     #10 pc 000000000000135c  /data/data/org.test.mapsforgeexample/cache/slice-mapsforge-map-android-0.6.0_24fb6ab0ab8de752356b68dd9111194aa0b25568-classes.dex (offset 0x1c000)

问题是,当您旋转屏幕时,会调用onDestroy(),后面是onCreate()。我可以通过在地图活动和其他活动之间切换,用具有多个活动的更复杂的应用程序来再现这一点。有时,不能立即重用AndroidGraphicFactory.ClearResourceMemoryCache()(onDestroy())中释放的资源来重绘onCreate()中的映射。

一个可能的解决方案是从onDestroy()中删除调用AndroidGraphicFactory.ClearResourceMemoryCache()。我发现,这在MapsForge0.6.0中没有任何意义,因为destroyAll()中也调用了它,删除该调用也会导致大量内存泄漏。

在当前的0.6.1版本中,使用to following方法,错误确实不会发生,因为从未调用AndroidGraphicFactory.ClearResourceMemoryCache()

@Override
protected void onDestroy() {
    this.mapView.destroyAll();
    super.onDestroy();
}

虽然我的app看起来工作很好,但我还是不服气,因为我觉得还是存在资源泄露的问题。这是真的,还是解决方案还好?是否有更好的方法,也许是代码中的不同点来打开/关闭AndroidGraphicFactory?任何提示都将受到高度赞赏。

我正在用Nexus5X和Android7.1.1进行测试,但我可以在多个设备和操作系统版本上重复这个问题。

共有1个答案

漆雕升
2023-03-14

请尝试一下,如果它可以解决您的问题:-

 @Override
      protected void onDestroy() {
        super.onDestroy();
        if (isFinishing()) {
          // do stuff
        } else { 
          //It's an orientation change.
        }
      }
 类似资料:
  • 我需要在我的Android Studio应用程序中使用Mapsforge(离线地图)。我导入了以下库: null 我已经把受抚养人包括在Build中了Gradle: 这是代码: 在XML中: 我认为在Eclipse中导入jar文件是必要的,但在Android Studio中可能需要一些其他的东西。

  • mapsforge-core-0.5.1.jar mapsforge-map-0.5.1.jar mapsforge-map-android-0.5.1.jar mapsforge-map-reader-0.5.1.jar

  • 首选项:我的android首选项是 API级别18 Android 4.3 这就是我如何读取url

  • 我用ruby 2.0在我的工作站上运行Ubuntu13.04。0,通过RVM安装。 $aptitude show libxml2 $aptitude show libxml2 dev 开发能力

  • 这是我在这个论坛上找到的提供会话工厂的类,但我一直得到JndiException:解析JNDI名称[]时出错 javax。命名。NoInitialContextException:需要在环境或系统属性中指定类名,或者作为applet参数,或者在应用程序资源文件中指定类名:java。命名。工厂以javax开头。命名。spi。NamingManager。javax上的getInitialContext

  • 我们的应用程序不包含任何c/c代码,而我们使用的第三方sdk包含。所以 现在,我们发现了一个巨大的本地崩溃? 我们找不到有用的信息来获取要承担责任的sdk。崩溃指向,比如“libc.so”、“libart.so”和“/system/framework/arm/boot framework.oat” 那么,我能做些什么来解决崩溃?升级sdk?但我不知道升级哪个 我无法获得我们应用程序的任何信息...