调试小部件导致ANR问题

5

我正在尝试调试一个AppWidget,但遇到了一个问题:D 如果不设置断点,则小部件可以正常工作,而Log.v命令也可以完美执行。然后我在方法顶部放置了一个断点:

    public void onReceive(Context context, Intent intent) {
    Log.v(TAG, "onReceive 1"); // BP on this line
    super.onReceive(context, intent);
    String action = intent.getAction();

    // Checks on action and computations ...

    Log.v(TAG, "onReceive 2");
    updateWidget(context);
    Log.v(TAG, "onReceive 3");
}

断点停止了执行,但是进程随后崩溃了。问题在于断点(我猜是 xD)导致了 ANR,然后 ActivityManager 杀死了进程。以下是日志:

01-07 14:32:38.886: ERROR/ActivityManager(72): ANR in com.salvo.wifiwidget
01-07 14:32:38.886: INFO/Process(72): Sending signal. PID: 475 SIG: 9
......
......
01-07 14:32:38.906: INFO/ActivityManager(72): Process com.salvo.wifiwidget (pid   475) has died.

这会导致调试停止。 所以问题是:有没有一种方法可以在不发生ANR的情况下调试小部件?? 谢谢提前回答。
1个回答

2
“你是对的。根据this,如果Widget的BroadcastReceiver在处理其onReceive方法时需要超过10秒钟,系统将会杀掉它。”
“我能想到两个解决方案:”
“1. 将逻辑与展示分开,在Activity中调试逻辑,然后再将其整合到小部件中。”
“2. 使用‘旧好’的日志记录。”
“或者你可以尝试以下方法。我不确定它是否有效,但也许值得一试。”
“使用自己的Activity作为小部件RemoteViews的宿主。我曾经做过类似的事情:”
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"
    >
    <LinearLayout
            android:id="@+id/widget_view"
            android:layout_width="fill_parent"
            android:layout_height="200dp"/>
    <Button
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="Inflate widget"
        android:onClick="onInflateClick"/>
    <Button
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="Update widget"
            android:onClick="onUpdateClick"/>
</LinearLayout>

在这个布局中,id为widget_view的LinearLayout将扮演小部件宿主的角色。Activity代码本身看起来像这样:
package com.example;

import android.app.Activity;
import android.os.Bundle;
import android.view.View;
import android.widget.LinearLayout;
import android.widget.RemoteViews;

public class MyActivity extends Activity {
    private LinearLayout widgetHolder;
    private View widgetView;

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);

        widgetHolder = (LinearLayout) findViewById(R.id.widget_view);
    }

    public void onInflateClick(View v) {
        RemoteViews views = new RemoteViews(getPackageName(), R.layout.widget);
        views.setImageViewResource(R.id.image, R.drawable.img1);
        views.setTextViewText(R.id.text, "Widget created");
        widgetView = views.apply(this, null);
        widgetHolder.addView(widgetView);
    }

    public void onUpdateClick(View v) {
        onUpdateWidget(0);
    }

    public void onUpdateWidget(int widgetId) {
        RemoteViews views = new RemoteViews(getPackageName(), R.layout.widget);
        views.setImageViewResource(R.id.image, R.drawable.img2);
        views.setTextViewText(R.id.text, "Widget updated");

        // Tell the AppWidgetManager to perform an update on the current app widget
        updateWidget(widgetId, views);
    }

    private void updateWidget(int widgetId, RemoteViews views) {
        views.reapply(this, widgetView);
        // appWidgetManager.updateAppWidget(appWidgetId, views);
    }
}

将您的小部件更新逻辑放入updateWidget方法中(或仅使用虚假参数调用小部件的BroadcastReceiver的正确onUpdate方法),然后您可以在活动上轻松调试它。
再次强调,我从未在真正的小部件上尝试过它,我只是想出了这个想法并尝试编写了代码。使用它要自担风险 :)
如果您想尝试,我已经将我的快速而粗糙的项目放到github上。它是一个Idea项目,但应该很容易将其导入Eclipse。

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接