android学习日记25--ANR和Hander消息机制

1、ANR(Application Not Responding)定义
  在Android上,如果你的应用程序有一段时间响应不够灵敏,系统会向用户显示一个对话框,
这个对话框称作应用程序无响应(ANR:Application Not Responding)对话框。用户可以选择“等待”而让程序继续运行,
也可以选择“强制关闭”。所以一个流畅的合理的应用程序中不能出现anr,而让用户每次都要处理这个对话框。
因此,在程序里对响应性能的设计很重要,这样系统不会显示ANR给用户。默认情况下,在android中Activity的最长执行时间是5秒,
BroadcastReceiver的最长执行时间则是10秒。

2、如何避免ANR
  Android应用程序通常是运行在一个单独的UI主线程里,因此,运行在主线程里的任何方法都尽可能少做事情。
特别是,Activity应该在它的关键生命周期方法(如onCreate()和onResume())里尽可能少的去做创建操作。
潜在的耗时操作,例如网络或数据库操作,或者高耗时的计算如改变位图尺寸,应该在子线程里
(或者以数据库操作为例,通过异步请求的方式)来完成。然而,不是说你的主线程阻塞在那里等待子线程的完成——也不是调用
Thread.wait()或是Thread.sleep()。替代的方法是,主线程应该为子线程提供一个Handler,以便完成时能够提交给主线程。

3、Hander 定义
  主要接受子线程发送的数据, 并用此数据配合主线程更新UI.
直接在UI线程中开启子线程来更新TextView显示的内容,运行程序我们会发现,如下错误:
android.view.ViewRoot$CalledFromWrongThreadException: Only the original thread that created a view hierarchy can touch its views.
翻译过来就是:只有创建这个控件的线程才能去更新该控件的内容。
所有的UI线程要去负责View的创建并且维护它,例如更新冒个TextView的显示,都必须在主线程中去做,我们不能直接在UI线程中去创建子线程,
要利用消息机制:handler,如下就是handler的简单工作原理图:

  Android系统中的Looper负责管理线程的消息队列和消息循环。创建的工作线程默认是没有消息队列和消息循环的,如果想让工作线程具有消息队列和消息循环,

就需要在线程中先调用Looper.prepare()来创建消息队列,然后调用Looper.loop()进入消息循环。下面是我们创建的工作线程:

 1     class WorkThread extends Thread {
 2           public Handler mHandler;
 3 
 4           public void run() {
 5               Looper.prepare();
 6 
 7               mHandler = new Handler() {
 8                   public void handleMessage(Message msg) {
 9                       // 处理收到的消息
10                   }
11               };
12 
13               Looper.loop();
14           }
15       }

这样一来,我们创建的工作线程就具有了消息处理机制了。运行在主线程中,Android系统会在Activity启动时为其创建一个消息队列和消息循环,就不需要

Looper.prepare()和Looper.loop();

4、Hander 用法
  继承或实现Hendler类,并重写handleMessage(Message msg) 方法, 用于接受线程数据
如下实例:

启动后,更新后的值一直在增加


主要代码:

 1 public class MainActivity extends Activity {
 2 
 3     private TextView tv;
 4     private static final int UPDATE = 0;
 5     private Handler handler = new Handler() {
 6 
 7         @Override
 8         public void handleMessage(Message msg) {
 9             // TODO 接收消息并且去更新UI线程上的控件内容
10             if (msg.what == UPDATE) {
11                 // Bundle b = msg.getData();
12                 // tv.setText(b.getString("num"));
13                 tv.setText(String.valueOf(msg.obj));
14             }
15             super.handleMessage(msg);
16         }
17     };
18 
19     /** Called when the activity is first created. */
20     @Override
21     public void onCreate(Bundle savedInstanceState) {
22         super.onCreate(savedInstanceState);
23         setContentView(R.layout.activity_main);
24         tv = (TextView) findViewById(R.id.tv);
25 
26         new Thread() {
27             @Override
28             public void run() {
29                 // TODO 子线程中通过handler发送消息给handler接收,由handler去更新TextView的值
30                 try {
31                     for (int i = 0; i < 100; i++) {
32                         Thread.sleep(500);
33                         Message msg = new Message();
34                         msg.what = UPDATE;
35                         // Bundle b = new Bundle();
36                         // b.putString("num", "更新后的值:" + i);
37                         // msg.setData(b);
38                         msg.obj = "更新后的值:" + i;
39                         handler.sendMessage(msg);
40                     }
41                 } catch (InterruptedException e) {
42                     e.printStackTrace();
43                 }
44             }
45         }.start();
46     }
47 
48 }

  关于message,指对于Android中Handler可以传递一些内容,通过Bundle对象可以封装String、Integer以及Blob二进制对象,
我们通过在线程中使用Handler对象的sendEmptyMessage或sendMessage方法来传递一个Bundle对象到Handler处理器。
对于Handler类提供了重写方法handleMessage(Message msg) 来判断,通过msg.what来区分每条信息。
将Bundle解包来实现Handler类更新UI线程中的内容实现控件的刷新操作。相关的Handler对象有关消息发送sendXXXX相关方法如下,
同时还有postXXXX相关方法,这些和Win32中的道理基本一致,一个为发送后直接返回,一个为处理后才返回。

比如我们可以在message里封装Bundle对象。

两边传递的代码如下:

1             // 其他Thread里发送
2             Message msg = new Message();
3             Bundle b = new Bundle();// 存放数据
4             b.putString("color", "我的");
5             msg.setData(b);
6             
7             // handleMessage里接收
8             Bundle b = msg.getData();
9             String color = b.getString("color");

5、单线程的消息机制

  Hander属于多线程的消息机制,Android 单线程事件处理机制 有两种

1、基于监听器的机制

  绑定特定的事件监听器,如Button的OnClickListener监听器, 监听器模型:包含事件源(EventSource)、事件(Event)、事件监听器(EventListener)。 一般内部类作为事件监听器类,主要因为 内部类可以在当前类复用;内部类可以调用外部类的组件。当然如果多个 GUI公用一个监听器类,还是使用外部类好一点,只需传入当前的context。当然大部分时间处理没什么复用价值, 更多的时候使用内部匿名类来实现。

2、基于回调函数的机制

  重写回调方法,如View的OnKeyDown方法,基于回调函数的机制的事件源和事件监听器是统一的, Android为所有GUI提供一套事件处理方法,当重写回调函数处理完后必需返回true,事件才不会继续向上层调用。

优劣:

  基于回调函数 使得更具通用型、代码更加简洁, 基于回调函数 事件模型分工明确、具有更好的维护性 对于特定事件,无法使用回调函数只能使用监听器。

原文地址:https://www.cnblogs.com/aiguozhe/p/3672058.html