如何修复更新小部件的ANR错误

时间:2018-05-23 14:14:28

标签: android widget android-anr-dialog

我有一个JobScheduler服务触发更新小部件UI。

@Override
    public boolean onStartJob(final JobParameters params) {
        //Offloading work to a new thread.
        new Thread(() -> {
            try {
                // Here we receive updated data and store to SharedPreferences 
                ...
                MyWidgetProvider.updateWidgets(getApplicationContext());
            } catch (Exception e) {
                Log.e(TAG, "onStartJob: " + e.getMessage());
            } finally {
                jobFinished(params, false);
            }
        }).start();
        return true;
    }

我的AppWidgetProvider:

protected static void updateWidget(Context context) {
    try {
        ComponentName thisWidget = new ComponentName(context, MyWidgetProvider.class);
        int[] appWidgetIds = AppWidgetManager.getInstance(context.getApplicationContext()).getAppWidgetIds(thisWidget);
        if (appWidgetIds != null) {
            if (appWidgetIds.length > 0) {
                for (int appWidgetId : appWidgetIds) {
                    try {
                        Intent intent = new Intent(context, MyWidgetProvider.class);

                        intent.putExtra(AppWidgetManager.EXTRA_APPWIDGET_ID, appWidgetId);
                        intent.setAction(UPDATE_WIDGET);
                        context.sendBroadcast(intent);
                    } catch (Exception e) {
                    }
                }
            }
        }
    } catch (Exception e) {
        Log.e(TAG, "error" + e.getMessage());
    }
}


@Override
public void onReceive(Context context, Intent intent) {
    super.onReceive(context, intent);
    try {
        Bundle extras = intent.getExtras();
        String action = intent.getAction();
        if (extras != null && !TextUtils.isEmpty(action)) {
            int appWidgetId = extras.getInt(AppWidgetManager.EXTRA_APPWIDGET_ID, AppWidgetManager.INVALID_APPWIDGET_ID);
            if (appWidgetId != AppWidgetManager.INVALID_APPWIDGET_ID) {
                if (action.equals(UPDATE_WIDGET)) {

                    RemoteViews remoteViews = getRemoteView(context);
                    // Here read from the SharedPreferences and updates the remoteViews
                    ....

                    AppWidgetManager appWidgetManager = AppWidgetManager.getInstance(context);
                    appWidgetManager.partiallyUpdateAppWidget(appWidgetId, remoteViews);
                } 
            }
        }
    } catch (Exception e) {
        Log.e(TAG, "onReceive() error: " + e.getMessage());
    }
}

我在意向广播中收到ANR错误{act = ..... UPDATE_WIDGET}

由于我无法在我的设备上重现,我必须猜测解决方案是什么。

解决方案#1 我可以传递减少UI运行时的值,而不是保存到SharedPreferences并从SharedPreferences中读取。我可以马上做出改变。

解决方案#2 我应该调用一个广播并传递所有小部件ID,而不是为每个小部件ID循环并为每个小部件ID调用广播吗?不会让onReceive()花费更长的时间来执行吗? (因为它必须更新所有小部件)。

解决方案#2是我不确定的。这种变化会有所帮助吗?或导致ANR更频繁发生?

1 个答案:

答案 0 :(得分:0)

回答我自己的问题。

解决方案#1 通过传递数据值来避免从SharedPreferences中加载数据和/或从WorkerThread中加载SharedPreferences,都消除了ANR错误。

我已经尝试了解决方案2,但不确定是否有助于消除ANR。