MessageListActivity has leaked IntentReceiver
1. 在MessagelistActivity中出现has leaked IntentReceiver的异常。异常日志如下。
07-15 08:09:53.211: E/ActivityThread(15491): Activity com.txrj.sms.activity.MessageListActivity has leaked IntentReceiver com.txrj.sms.activity.MessageListActivity$6@41e189d0 that was originally registered here. Are you missing a call to unregisterReceiver()?
07-15 08:09:53.211: E/ActivityThread(15491): android.app.IntentReceiverLeaked: Activity com.txrj.sms.activity.MessageListActivity has leaked IntentReceiver com.txrj.sms.activity.MessageListActivity$6@41e189d0 that was originally registered here. Are you missing a call to unregisterReceiver()?
07-15 08:09:53.211: E/ActivityThread(15491): at android.app.LoadedApk$ReceiverDispatcher.<init>(LoadedApk.java:763)
07-15 08:09:53.211: E/ActivityThread(15491): at android.app.LoadedApk.getReceiverDispatcher(LoadedApk.java:567)
07-15 08:09:53.211: E/ActivityThread(15491): at android.app.ContextImpl.registerReceiverInternal(ContextImpl.java:1167)
07-15 08:09:53.211: E/ActivityThread(15491): at android.app.ContextImpl.registerReceiver(ContextImpl.java:1154)
07-15 08:09:53.211: E/ActivityThread(15491): at android.app.ContextImpl.registerReceiver(ContextImpl.java:1148)
07-15 08:09:53.211: E/ActivityThread(15491): at android.content.ContextWrapper.registerReceiver(ContextWrapper.java:348)
07-15 08:09:53.211: E/ActivityThread(15491): at com.txrj.sms.activity.MessageListActivity.getDeliveryIntent(MessageListActivity.java:247)
07-15 08:09:53.211: E/ActivityThread(15491): at com.txrj.sms.activity.MessageListActivity.sendMessage(MessageListActivity.java:196)
07-15 08:09:53.211: E/ActivityThread(15491): at com.txrj.sms.activity.MessageListActivity.access$14(MessageListActivity.java:190)
07-15 08:09:53.211: E/ActivityThread(15491): at com.txrj.sms.activity.MessageListActivity$2.onClick(MessageListActivity.java:120)
07-15 08:09:53.211: E/ActivityThread(15491): at android.view.View.performClick(View.java:3567)
07-15 08:09:53.211: E/ActivityThread(15491): at android.view.View$PerformClick.run(View.java:14224)
07-15 08:09:53.211: E/ActivityThread(15491): at android.os.Handler.handleCallback(Handler.java:605)
07-15 08:09:53.211: E/ActivityThread(15491): at android.os.Handler.dispatchMessage(Handler.java:92)
07-15 08:09:53.211: E/ActivityThread(15491): at android.os.Looper.loop(Looper.java:137)
07-15 08:09:53.211: E/ActivityThread(15491): at android.app.ActivityThread.main(ActivityThread.java:4517)
07-15 08:09:53.211: E/ActivityThread(15491): at java.lang.reflect.Method.invokeNative(Native Method)
07-15 08:09:53.211: E/ActivityThread(15491): at java.lang.reflect.Method.invoke(Method.java:511)
07-15 08:09:53.211: E/ActivityThread(15491): at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:993)
07-15 08:09:53.211: E/ActivityThread(15491): at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:760)
07-15 08:09:53.211: E/ActivityThread(15491): at dalvik.system.NativeStart.main(Native Method)
07-15 08:09:53.221: E/ActivityThread(15491): Activity com.txrj.sms.activity.MessageListActivity has leaked IntentReceiver com.txrj.sms.activity.MessageListActivity$5@41df9b80 that was originally registered here. Are you missing a call to unregisterReceiver()?
07-15 08:09:53.221: E/ActivityThread(15491): android.app.IntentReceiverLeaked: Activity com.txrj.sms.activity.MessageListActivity has leaked IntentReceiver com.txrj.sms.activity.MessageListActivity$5@41df9b80 that was originally registered here. Are you missing a call to unregisterReceiver()?
07-15 08:09:53.221: E/ActivityThread(15491): at android.app.LoadedApk$ReceiverDispatcher.<init>(LoadedApk.java:763)
07-15 08:09:53.221: E/ActivityThread(15491): at android.app.LoadedApk.getReceiverDispatcher(LoadedApk.java:567)
07-15 08:09:53.221: E/ActivityThread(15491): at android.app.ContextImpl.registerReceiverInternal(ContextImpl.java:1167)
07-15 08:09:53.221: E/ActivityThread(15491): at android.app.ContextImpl.registerReceiver(ContextImpl.java:1154)
07-15 08:09:53.221: E/ActivityThread(15491): at android.app.ContextImpl.registerReceiver(ContextImpl.java:1148)
07-15 08:09:53.221: E/ActivityThread(15491): at android.content.ContextWrapper.registerReceiver(ContextWrapper.java:348)
07-15 08:09:53.221: E/ActivityThread(15491): at com.txrj.sms.activity.MessageListActivity.getSentIntent(MessageListActivity.java:232)
07-15 08:09:53.221: E/ActivityThread(15491): at com.txrj.sms.activity.MessageListActivity.sendMessage(MessageListActivity.java:195)
07-15 08:09:53.221: E/ActivityThread(15491): at com.txrj.sms.activity.MessageListActivity.access$14(MessageListActivity.java:190)
07-15 08:09:53.221: E/ActivityThread(15491): at com.txrj.sms.activity.MessageListActivity$2.onClick(MessageListActivity.java:120)
07-15 08:09:53.221: E/ActivityThread(15491): at android.view.View.performClick(View.java:3567)
07-15 08:09:53.221: E/ActivityThread(15491): at android.view.View$PerformClick.run(View.java:14224)
07-15 08:09:53.221: E/ActivityThread(15491): at android.os.Handler.handleCallback(Handler.java:605)
07-15 08:09:53.221: E/ActivityThread(15491): at android.os.Handler.dispatchMessage(Handler.java:92)
07-15 08:09:53.221: E/ActivityThread(15491): at android.os.Looper.loop(Looper.java:137)
07-15 08:09:53.221: E/ActivityThread(15491): at android.app.ActivityThread.main(ActivityThread.java:4517)
07-15 08:09:53.221: E/ActivityThread(15491): at java.lang.reflect.Method.invokeNative(Native Method)
07-15 08:09:53.221: E/ActivityThread(15491): at java.lang.reflect.Method.invoke(Method.java:511)
07-15 08:09:53.221: E/ActivityThread(15491): at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:993)
07-15 08:09:53.221: E/ActivityThread(15491): at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:760)
07-15 08:09:53.221: E/ActivityThread(15491): at dalvik.system.NativeStart.main(Native Method)
2. 出现异常原因分析。在发送信息时,创建了两个PendingIntent,它们是调用方法getBroadcast来发送广播。在此之前调用registerBroadcast注册广播,查看代码找不到注销广播的语句,所以导致出现leaked IntentReceiver的异常。以下是改正之后的代码。
private PendingIntent getSentIntent(final Uri uri) {
BroadcastReceiver receiver = new BroadcastReceiver(){
@Override
public void onReceive(Context context, Intent intent) {
mContext.unregisterReceiver(this);
int resultCode = getResultCode();
if(resultCode == RESULT_OK) {
Toast.makeText(context, "send message success.", Toast.LENGTH_SHORT).show();
//Log.i("txrjsms", "send message success.");
updateMsgType(uri, Sms.MESSAGE_TYPE_SENT);
} else if(resultCode == SmsManager.RESULT_ERROR_GENERIC_FAILURE) {
Toast.makeText(context, "Generic failure.", Toast.LENGTH_SHORT).show();
updateMsgType(uri, Sms.MESSAGE_TYPE_FAILED);
} else if(resultCode == SmsManager.RESULT_ERROR_NO_SERVICE) {
Toast.makeText(context, "service is currently unavailable.", Toast.LENGTH_SHORT).show();
updateMsgType(uri, Sms.MESSAGE_TYPE_FAILED);
} else if(resultCode == SmsManager.RESULT_ERROR_NULL_PDU) {
Toast.makeText(context, "no pdu provided.", Toast.LENGTH_SHORT).show();
updateMsgType(uri, Sms.MESSAGE_TYPE_FAILED);
} else if(resultCode == SmsManager.RESULT_ERROR_RADIO_OFF) {
Toast.makeText(context, "radio was explicitly turned off.", Toast.LENGTH_SHORT).show();
updateMsgType(uri, Sms.MESSAGE_TYPE_FAILED);
}
}
};
IntentFilter filter = new IntentFilter(TxrjConstant.ACTION_SEND_SMS);
mContext.registerReceiver(receiver, filter);
Intent sentIntent = new Intent(TxrjConstant.ACTION_SEND_SMS);
return PendingIntent.getBroadcast(mContext, 0, sentIntent, 0);
}
private PendingIntent getDeliveryIntent() {
BroadcastReceiver receiver = new BroadcastReceiver(){
@Override
public void onReceive(Context context, Intent intent) {
mContext.unregisterReceiver(this);
Toast.makeText(context, "message arrived.", Toast.LENGTH_SHORT).show();
//Log.i("txrjsms", "message arrived.");
}
};
IntentFilter filter = new IntentFilter(TxrjConstant.ACTION_DELIVERY_SMS);
mContext.registerReceiver(receiver, filter);
Intent deliveryIntent = new Intent(TxrjConstant.ACTION_DELIVERY_SMS);
return PendingIntent.getBroadcast(mContext, 0, deliveryIntent, 0);
}
3. 添加了注销广播接收器的语句之后还会有问题。比如,发出信息之后,对方有信息回复时,在MessageListActivity界面中无法将刚收到的信息显示出来。优化注册和注销Receiver的代码。以下是优化过的代码,将receiver作为实例变量来定义,在第一次获取sentIntert和deliveryIntent时创建并注册它们。在onDestroy方法中注销它们。
在sentReceiver中需要传递表示发送信息的uri。在调用getBroadcast时通过intent.putExtra放入Uri。在onReceive方法中通过intent.getParcelableExtra取出Uri。
private void createAndRegisterSentReceiver() {
sentReceiver = new BroadcastReceiver(){
@Override
public void onReceive(Context context, Intent intent) {
Uri uri = intent.getParcelableExtra(TxrjConstant.EXTRA_SENT_URI);
int resultCode = getResultCode();
if(resultCode == RESULT_OK) {
Toast.makeText(context, "send message success.", Toast.LENGTH_SHORT).show();
updateMsgType(uri, Sms.MESSAGE_TYPE_SENT);
} else if(resultCode == SmsManager.RESULT_ERROR_GENERIC_FAILURE) {
Toast.makeText(context, "Generic failure.", Toast.LENGTH_SHORT).show();
updateMsgType(uri, Sms.MESSAGE_TYPE_FAILED);
} else if(resultCode == SmsManager.RESULT_ERROR_NO_SERVICE) {
Toast.makeText(context, "service is currently unavailable.", Toast.LENGTH_SHORT).show();
updateMsgType(uri, Sms.MESSAGE_TYPE_FAILED);
} else if(resultCode == SmsManager.RESULT_ERROR_NULL_PDU) {
Toast.makeText(context, "no pdu provided.", Toast.LENGTH_SHORT).show();
updateMsgType(uri, Sms.MESSAGE_TYPE_FAILED);
} else if(resultCode == SmsManager.RESULT_ERROR_RADIO_OFF) {
Toast.makeText(context, "radio was explicitly turned off.", Toast.LENGTH_SHORT).show();
updateMsgType(uri, Sms.MESSAGE_TYPE_FAILED);
}
}
};
IntentFilter filter = new IntentFilter(TxrjConstant.ACTION_SEND_SMS);
mContext.registerReceiver(sentReceiver, filter);
}
private PendingIntent getSentIntent(final Uri uri) {
if(sentReceiver == null) {
createAndRegisterSentReceiver();
}
Intent sentIntent = new Intent(TxrjConstant.ACTION_SEND_SMS);
sentIntent.putExtra(TxrjConstant.EXTRA_SENT_URI, uri);
return PendingIntent.getBroadcast(mContext, 0, sentIntent, 0);
}
deliveryReceiver使用类似的处理方法。代码如下所示。
private void createAndRegisterDeliveryIntent() {
deliveryReceiver = new BroadcastReceiver(){
@Override
public void onReceive(Context context, Intent intent) {
mContext.unregisterReceiver(this);
Toast.makeText(context, "message arrived.", Toast.LENGTH_SHORT).show();
}
};
IntentFilter filter = new IntentFilter(TxrjConstant.ACTION_DELIVERY_SMS);
mContext.registerReceiver(deliveryReceiver, filter);
}
private PendingIntent getDeliveryIntent() {
if(deliveryReceiver == null) {
createAndRegisterDeliveryIntent();
}
Intent deliveryIntent = new Intent(TxrjConstant.ACTION_DELIVERY_SMS);
return PendingIntent.getBroadcast(mContext, 0, deliveryIntent, 0);
}
在onDestroy()方法中注销sentReceiver和deliveryReceiver。
protected void onDestroy() {
super.onDestroy();
getContentResolver().unregisterContentObserver(mInboxObserver);
if(sentReceiver != null) {
unregisterReceiver(sentReceiver);
}
if(deliveryReceiver != null) {
unregisterReceiver(deliveryReceiver);
}
}
MessageListActivity has leaked IntentReceiver的更多相关文章
- Android学习系列(37)--App调试内存泄露之Context篇(下)
接着<Android学习系列(36)--App调试内存泄露之Context篇(上)>继续分析. 5. AsyncTask对象 我N年前去盛大面过一次试,当时面试官极力推荐我使用AsyncT ...
- [Android Memory] App调试内存泄露之Context篇(下)
转载地址:http://www.cnblogs.com/qianxudetianxia/p/3655475.html 5. AsyncTask对象 我N年前去盛大面过一次试,当时面试官极力推荐我使用A ...
- Android 之 WebView
1:在AndroidManifest.xml中添加允许android访问网络权限. <uses-permission android:name="android.permission. ...
- WebView全面学习(二)-- Native与js双方通信
WebView全面学习(二)-- Native与js双方通信 Native与js通信的本质 Native与js通信的核心在于WebView. 两端的通信主要还是单向的.假如要完成js->Nati ...
- android中常见的内存泄漏和解决的方法
android中的内存溢出预计大多数人在写代码的时候都出现过,事实上突然认为工作一年和工作三年的差别是什么呢.事实上干的工作或许都一样,产品汪看到的结果也都一样,那差别就是速度和质量了. 写在前面的一 ...
- StrictMode 详解
StrictMode类是Android 2.3 (API 9)引入的一个工具类,可以用来帮助开发者发现代码中的一些不规范的问题.比如,如果你在UI线程中进行了网络或者磁盘操作,StrictMode就会 ...
- Activity has leaked window that was originally added
错误: E/WindowManager: android.view.WindowLeaked: Activity com.x.x.x has leaked window com.android.int ...
- 数据库内存泄漏——A SQLiteConnection object for database '/data/data/.../databases/....db' was leaked!
详细异常: A SQLiteConnection object for database '/data/data/.../database/....db' was leaked! Please ...
- has leaked ServiceConnection com.baidu.location.LocationClient
02-06 05:01:52.806: E/ActivityThread(1120): Activity com.project.xxxActivity $1@45fc5af8 that was or ...
随机推荐
- 我对android 软件栈了解
android 软件栈如图所示: Android平台的核心是Linux内核,它负责设备驱动程序.资源访问.电源管理和完成其他操作系统的职责.提供的设备驱动程序包括显示器.照相机,键盘.WiFi.闪存. ...
- Android基于RecyclerView实现高亮搜索列表
这篇应该是RecycleView的第四篇了,RecycleView真是新生代的宠儿能做这么多的事情. 转载请注明作者AndroidMsky及原文链接 http://blog.csdn.net/Andr ...
- bash: php: command not found
bash: php: command not found 解决:export PATH=$PATH:/usr/local/php/bin
- ubuntu忘记root密码 的解决方法
alt+f2,在弹出的运行窗口中输入:gnome-terminal,回车. 即进入终端 输入:sudo passwd root,回车后会提示你输入你当前用户的密码 之后 按提示输入两次root的密码( ...
- Tensorflow-3-使用RNN生成中文小说
https://blog.csdn.net/heisejiuhuche/article/details/73010638 这篇文章不涉及RNN的基本原理,只是从选择数据集开始,到最后生成文本,展示一个 ...
- 'Lock wait timeout exceeded; try restarting transaction'问题
OperationalError: (1205, 'Lock wait timeout exceeded; try restarting transaction') 原因很简单,太多错误,意外处理没有 ...
- css 下边框
float: left; width: 1200px; height: 42px; background-color: #fff0; /* background-color: #4f4aff; */ ...
- Asp.NET websocket,Asp.NET MVC 使用 SignalR 实现推送功能一(Hubs 在线聊天室)
ASP .NET SignalR 是一个ASP .NET 下的类库,可以在ASP .NET 的Web项目中实现实时通信.什么是实时通信的Web呢?就是让客户端(Web页面)和服务器端可以互相通知消息及 ...
- 结构体指针之 段错误 具体解释(segmentation fault)
一个网友问了我一个问题.一个C程序执行出现了段错误,这个问题非常好.非常多刚開始学习的人都easy犯这个错误,详细代码例如以下: watermark/2/text/aHR0cDovL2Jsb2cuY3 ...
- iOS开发点滴 - 关闭键盘
有时候系统显示的键盘会挡住视图中某些重要的控件,这个时候当用户按下换行键,就应该取消UITextField对象的第一响应(First Responder)状态而关闭键盘. 1. 首先,视图控制器必须遵 ...