错误提示Receiver not registered的处理

错误提示:java.lang.IllegalArgumentException: Receiver not registered。。

顾名思义就是Receiver没有被注册,这是由于取消了没有注册的Receiver导致的错误。

我们使用Receiver时必须保证注册(register)和反注册(unregister)成对出现。

遇到这种问题时,你需要检查下自己代码中注册和反注册的代码,看下是否存在多次调用unregister的时候。

很多时候都是想当然认为会那样,但是程序中可能有我们未预料的出口,比如我的程序中在onCreate注册,finish取消注册,理论进入、退出时应该成对出现,但实际报错了!分析到最后是原布局中使用了TabActivity、GroupActivity等多种结构嵌套,退出当前Activity时,finish方法被执行了多次。

所以建议Receiver注册和反注册的过程放到onResume和onPause过程中,两个理由:

(1)这两个过程肯定会成对出现,因此可以避免上述错误。

(2)一般我们使用Receiver是为了接受通知来更改UI(特殊情况除外),而对于Activity(UI)界面位于后台时,没必要去接受通知更改UI。

示例:

在 Adroid2.1 和 Adroid2.2 设备上,如果 Activity 中使用到 ViewFlipper 控件,进行横竖屏切换操作时会发生如下错误信息

java.lang.IllegalArgumentException: Receiver not registered: android.widget.ViewFlipper$1@43dee3c0

这是由于 onDetachedFromWindow() 莫名其妙地在 onAttachedToWindow() 之前被调用了。据说是一个 Bug ,不知在最新的 2.3 上是否修正。

下面提供一个很简单的解决方案,重写 ViewFlipper 的 onDetachedFromWindow() 方法

1
2
3
4
5
6
7
8
9
@Override
protected void onDetachedFromWindow () {
try {
super.onDetachedFromWindow();
}
catch (IllegalArgumentException e) {
stopFlipping();
}
}

当注册广播时,常会遇到的问题就是重复注销广播处理函数是会报错,而且会让进程崩溃。

一般来说,可以通过一个变量来保存广播处理是否被注销,每当注销时,将它标记为false。如果再次注销时遇到false就不对他进行注销处理。

但是程序比较复杂的时候会比较难控制。可以使用try catch方式捕获错误。

1
2
3
4
5
6
7
8
9
10
try {  
unregisterReceiver(receiver);
} catch (IllegalArgumentException e) {
if (e.getMessage().contains("Receiver not registered")) {
// Ignore this exception. This is exactly what is desired
} else {
// unexpected, re-throw
throw e;
}
}