• 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,事件才不会继续向上层调用。

    优劣:

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

  • 相关阅读:
    基于C语言的高斯曲线拟合原理以及实现【转】
    高斯曲线拟合原理及实现
    软件测试Lab2————用selenium进行自动化测试
    软件测试作业四 —— 课后习题七及扩展
    软件测试作业三—— 上机实验判断三角形的类型
    软件项目管理作业一 —— 描述项目
    软件测试作业二 —— 对于Faults,Errors,Failures的认识的习题
    软件测试作业一 —— 记录我遇到的项目error
    The introduction about my project.
    课程表WPF制作 学习步骤三
  • 原文地址:https://www.cnblogs.com/aiguozhe/p/3672058.html
Copyright © 2020-2023  润新知