9

我编写了一个 Android 应用程序,它在我尝试过的所有手机上都可以正常运行,除了三星 S3。发生的情况是,当我的应用程序运行时,其他后台进程(即其他应用程序)将开始死亡。例如,使用我的应用一段时间后,动态壁纸会死机,用户退出我的应用后手机会出现黑色背景。最后,在使用了一段时间后,我的应用程序也被操作系统杀死了。令人沮丧的是,我在日志中看不到任何错误消息,可以提示我到底是什么问题。

这仅在三星 S3 上发生(而不是在 S2 上,例如)。

我认为这与内存有关,因为我的应用程序非常占用内存(它正在从网络加载大量图像),但我在日志中看不到任何“无内存”错误。

我还怀疑我使用的 HTTP 框架可能有问题,所以我从 Apache HttpComponents 切换到 java.net.HttpURLConnection,但没有帮助。

任何关于为什么会发生这种情况的想法或关于如何调试它的提示将不胜感激。

以下是日志的摘录,其中显示了一些后台进程正在死亡(例如,动态壁纸):

01-07 01:57:37.245: I/InputDispatcher(2296): Delivering touch to current input target: action: 0x1
01-07 01:57:37.505: D/dalvikvm(29490): GC_FOR_ALLOC freed 1535K, 17% free 28032K/33735K, paused 71ms, total 71ms
01-07 01:57:37.580: D/dalvikvm(6718): WAIT_FOR_CONCURRENT_GC blocked 0ms
01-07 01:57:37.620: D/dalvikvm(6718): GC_EXPLICIT freed 91K, 9% free 17663K/19399K, paused 3ms+5ms, total 40ms
01-07 01:57:38.190: I/InputReader(2296): Touch event's action is 0x0 (deviceType=0) [pCnt=1, s=0.428 ]
01-07 01:57:38.190: I/InputDispatcher(2296): Delivering touch to current input target: action: 0x0
01-07 01:57:38.190: I/InputDispatcher(2296): Delivering touch to current input target: action: 0x0
01-07 01:57:38.190: I/InputDispatcher(2296): Delivering touch to current input target: action: 0x0
01-07 01:57:38.190: I/InputDispatcher(2296): Delivering touch to current input target: action: 0x0
01-07 01:57:38.190: I/InputDispatcher(2296): Delivering touch to current input target: action: 0x0
01-07 01:57:38.335: I/InputReader(2296): Touch event's action is 0x1 (deviceType=0) [pCnt=1, s=]
01-07 01:57:38.335: I/InputDispatcher(2296): Delivering touch to current input target: action: 0x1
01-07 01:57:38.335: I/InputDispatcher(2296): Delivering touch to current input target: action: 0x1
01-07 01:57:38.335: I/InputDispatcher(2296): Delivering touch to current input target: action: 0x1
01-07 01:57:38.750: D/dalvikvm(29490): GC_FOR_ALLOC freed 1688K, 18% free 27981K/33735K, paused 114ms, total 114ms
01-07 01:57:39.695: W/PowerManagerService(2296): Timer 0x3->0x3|0x0
01-07 01:57:39.700: D/PowerManagerService(2296): setTimeoutLocked::SmartSleep : after19500
01-07 01:57:39.930: I/InputReader(2296): Touch event's action is 0x0 (deviceType=0) [pCnt=1, s=0.429 ]
01-07 01:57:39.930: I/InputDispatcher(2296): Delivering touch to current input target: action: 0x0
01-07 01:57:39.930: I/InputDispatcher(2296): Delivering touch to current input target: action: 0x0
01-07 01:57:39.930: I/InputDispatcher(2296): Delivering touch to current input target: action: 0x0
01-07 01:57:39.930: I/InputDispatcher(2296): Delivering touch to current input target: action: 0x0
01-07 01:57:39.930: I/InputDispatcher(2296): Delivering touch to current input target: action: 0x0
01-07 01:57:40.020: D/DeviceInfo(2296): SysScope Service has unexpectedly disconnected!
01-07 01:57:40.065: I/InputReader(2296): Touch event's action is 0x1 (deviceType=0) [pCnt=1, s=]
01-07 01:57:40.065: I/InputDispatcher(2296): Delivering touch to current input target: action: 0x1
01-07 01:57:40.065: I/InputDispatcher(2296): Delivering touch to current input target: action: 0x1
01-07 01:57:40.065: I/InputDispatcher(2296): Delivering touch to current input target: action: 0x1
**01-07 01:57:40.080: I/ActivityManager(2296): Process com.android.smspush (pid 28031) (adj 1) has died.**
01-07 01:57:40.090: W/WallpaperService(2296): Wallpaper service gone: ComponentInfo{com.sec.ccl.csp.app.secretwallpaper.themetwo/com.sec.ccl.csp.app.secretwallpaper.themetwo.SecretWallpaperService}
01-07 01:57:40.125: D/KeyguardViewMediator(2296): setHidden false
**01-07 01:57:40.135: I/ActivityManager(2296): Process com.android.server.device.enterprise:remote (pid 28016) (adj 1) has died.**
**01-07 01:57:40.145: I/ActivityManager(2296): Process com.sec.ccl.csp.app.secretwallpaper.themetwo (pid 29715) (adj 1) has died.**
01-07 01:57:40.285: D/dalvikvm(29490): GC_CONCURRENT freed 1616K, 17% free 28289K/33735K, paused 27ms+21ms, total 178ms
01-07 01:57:40.285: D/dalvikvm(29490): WAIT_FOR_CONCURRENT_GC blocked 112ms
01-07 01:57:40.445: D/dalvikvm(2296): GC_CONCURRENT freed 1869K, 59% free 24186K/57991K, paused 26ms+12ms, total 197ms
01-07 01:57:40.660: D/dalvikvm(29490): GC_CONCURRENT freed 1587K, 16% free 28622K/33735K, paused 3ms+7ms, total 51ms
01-07 01:57:40.685: D/KeyguardViewMediator(2296): setHidden false
4

2 回答 2

9

后台进程可能停止的主要原因有 3 个,我将从最可能的情况开始,从最可能的情况开始:

你不经意地问它停止

检查您的代码是否存在在尴尬情况下可能会请求服务停止的错误。这可能在服务中或在您的应用程序的其他活动中。

未捕获的异常

服务遇到未捕获的异常。在这种情况下,您应该能够看到日志中的信息,但是日志不会持续很长时间,您可能会丢失它。

为了解决这种可能性,您可以设置DefaultUncaughtExceptionHandler如下:

UncaughtExceptionHandler currentHandler = Thread.getDefaultUncaughtExceptionHandler();
if (!(currentHandler instanceof DefaultExceptionHandler)) {
    // Register default exceptions handler
    Thread.setDefaultUncaughtExceptionHandler(
            new DefaultExceptionHandler(currentHandler));
}

并定义您的DefaultExceptionHandler类以写入临时文件,如下所示:

public class DefaultExceptionHandler implements UncaughtExceptionHandler {
    private UncaughtExceptionHandler defaultExceptionHandler;

    public DefaultExceptionHandler(UncaughtExceptionHandler pDefaultExceptionHandler)
    {
        defaultExceptionHandler = pDefaultExceptionHandler;
    }

    public void uncaughtException(Thread t, Throwable e) {
        final Writer result = new StringWriter();
        final PrintWriter printWriter = new PrintWriter(result);
        e.printStackTrace(printWriter);
        try {
            String filename = "your_temp_filename";
            BufferedWriter bw = new BufferedWriter(new FileWriter(filename));
            bw.write(result.toString());
        bw.flush();
        bw.close();
        } catch (Exception e2) {
            e2.printStackTrace();
        }
        defaultExceptionHandler.uncaughtException(t, e);        
    }
}

Android 操作系统要求服务停止

根据您的描述,这是最可能的原因,也是您无法控制的原因。

当 Android 需要额外资源或认为您的服务运行时间过长时,无法阻止 Android 杀死您的应用程序。服务在被杀死之前运行的时间可能会因设备而异,因为它取决于分配的资源(即图像内存需求随屏幕大小而变化)和可用资源。

在这种情况下,您可以做的唯一合理的事情是foreground service使用以下方式启动您的服务:

    startForeground(int id, Notification notification);

它通知操作系统您的服务具有用户可见性,并将被放入不太可能终止服务的列表中。我在一些设备上使用它的经验是,即使您同时使用几个其他应用程序,该服务也可以不间断地运行。

问候。

于 2013-01-09T22:59:28.983 回答
7

我已经追查到了这个问题。它实际上是由 Typeface.createFromAsset() 方法引起的内存泄漏,如此所述。泄漏的内存是本机内存(不在 Java 堆上),所以这就是没有 OutOfMemory 异常的原因。

奇怪的是,这个问题应该从 Honeycomb (3.0) 开始解决,但就我而言,它发生在运行 JellyBean (4.1) 的 S3 上。(我尝试了几个 S3,它们都有这个问题)。如果有人可以访问运行 JellyBean 的不同手机进行测试,那将会很有趣,这样我们就可以看到它是否在那里重现(我发布的链接显示了一种重现它的简单方法)。

于 2013-01-15T18:07:20.660 回答