android处理耗时任务_ANR


ANR

有些应用在使用中会弹出一个对话框,一般这个对话框叫做应用程序无响应对话框(ANR:Application Not Responding)。虽然这个提示框有等待和关闭应用程序两种选择,但是它的弹出就已经影响了用户使用app过程中的体验,所以一般来说,Android开发的过程中都会严格控制ANR的出现。

ANR表面原因:应用在一定时间内(一般是5秒)没有响应输入操作(例如输入,按钮,手势触摸等)。

ANR根本原因:Android主线程中进行耗时操作,造成主线程阻塞;BroadcastReceiver生命周期结束前没有完成相应的耗时任务。
Android应用都是运行在主线程中的,包括activity、service的生命周期,ui的刷新(所以主线程又叫ui线程),输入响应操作等等。也就是说任何在主线程中运行的函数都不能进行耗时的动作,否则会阻塞主线程,导致输入无法响应,从而弹出ANR。
BroadcastReceiver的生命周期很短(一般为10秒),如果在它生命周期结束前耗时任务没有完成,Android认为该程序没有响应。

这些耗时的操主要包括网络,数据库,大量的计算操作(如bitmap的处理)等。

简单的解决方案

1、为了防止主线程被阻塞,通常需要启动子线程来处理耗时任务,子线程的任务完成时通过handler通知主线程,让主线程刷新ui等。下面是一个简单的事例:
import android.app.Activity;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.TextView;

public class MainActivity extends Activity {

	private TextView text;
	private Button button;

	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.activity_main);

		text = (TextView) findViewById(R.id.text);
		button = (Button) findViewById(R.id.button);

		button.setOnClickListener(new OnClickListener() {

			@Override
			public void onClick(View arg0) {
				System.out.println("-->开启子线程");
				new TaskThread().start();
			}
		});

	}

	Handler handler = new Handler() {
		public void handleMessage(Message msg) {
			switch (msg.what) {
			case 0:
			{
				System.out.println("-->回到主线程刷新ui任务");
				text.setText("任务完成");
			}
				break;

			default:
				break;
			}
		};
	};

	class TaskThread extends Thread {
		public void run() {
			System.out.println("-->做一些耗时的任务");
			try {
				sleep(500);
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
			handler.sendEmptyMessage(0);
		};
	};
}
运行后打印结果如下:
05-26 05:30:16.490: I/System.out(381): -->开启子线程
05-26 05:30:16.510: I/System.out(381): -->做一些耗时的任务
05-26 05:30:17.011: I/System.out(381): -->回到主线程刷新ui任务

2、BroadcastReceiver的函数也是由主线程调用的,但是不应该启动子线程来处理耗时任务。因为BroadcastReceiver属于一个空进程,在BroadcastReceiver有限的生命周期结束后他的优先级就会降低,系统很容易就把它回收了,这样,子线程的宿主已经被回收了,那么它的子线程也会被杀死。所以建议在BroadcastReceiver中启动service,在service中启动线程处理耗时任务。





郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。