Android Handler 事件

发布于 2024-06-22 10:30:22 字数 5740 浏览 14 评论 0

Handler 的定义

Handler 主要接受子线程发送的数据, 并用此数据配合主线程更新 UI。 当应用程序启动时,Android 首先会开启一个主线程(也就是 UI 线程) ,主线程为管理界面中的 UI 控件,进行事件分发, 比如说,你要是点击一个 Button,Android 会分发事件到 Button 上,来响应你的操作。如果此时需要一个耗时的操作,例如: 联网读取数据,或者读取本地较大的一个文件的时候,你不能把这些操作放在主线程中,如果你放在主线程中的话,界面会出现假死现象, 如果 5 秒钟还没有完成的话,会收到 Android 系统的一个错误提示 强制关闭。

这个时候我们需要把这些耗时的操作,放在一个子线程中,因为子线程涉及到 UI 更新,Android 主线程不是线程安全的,也就是说,更新 UI 只能在主线程中更新,子线程中操作是危险的。这个时候,Handler 就出现了,来解决这个复杂的问题。Handler 运行在主线程中(UI 线程中),它与子线程可以通过 Message 对象来传递数据,这个时候,Handler 就承担着接受子线程传过来的(子线程用 sedMessage() 方法传递)Message 对象(里面包含数据),把这些消息放入主线程队列中,配合主线程进行更新 UI。

Handler 的一些特点

Handler 为 Android 操作系统中的线程通信工具,包为 android.os.Handler

与 Handler 绑定的有两个队列,一个为消息队列,另一个为线程队列。Handler 可以通过这两个队列来分别:

  1. 发送、接受、处理消息–消息队列;
  2. 启动、结束、休眠线程–线程队列;

Android OS 中,一个进程被创建之后,主线程(可理解为当前 Activity) 创建一个消息队列,这个消息队列维护所有顶层应用对象(Activities, Broadcast receivers 等) 以及主线程创建的窗口。你可以在主线程中创建新的线程,这些新的线程都通过 Handler 与主线程进行通信。通信通过新线程调用 Handler 的 post() 方法和 sendMessage() 方法实现,分别对应功能:

  1. post() 将一个线程加入线程队列;
  2. sendMessage() 发送一个消息对象到消息队列;

当然,post() 方法还有一些变体,比如

post(Runnable)
postAtTime(Runnable,long)
postDelayed(Runnable long)
sendEmptyMessage(int)
sendMessage(Message)
sendMessageAtTime(Message,long)
sendMessageDelayed(Message,long)

Handler 实例

public class MyHandlerActivity extends Activity {
    Button button;
    MyHandler myHandler;

    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.handlertest);

        button = (Button) findViewById(R.id.button);
        myHandler = new MyHandler();
        // 当创建一个新的 Handler 实例时, 它会绑定到当前线程和消息的队列中,开始分发数据
        // Handler 有两个作用, (1) : 定时执行 Message 和 Runnalbe 对象
        // (2): 让一个动作,在不同的线程中执行.

        // 它安排消息,用以下方法
        // post(Runnable)
        // postAtTime(Runnable,long)
        // postDelayed(Runnable,long)
        // sendEmptyMessage(int)
        // sendMessage(Message);
        // sendMessageAtTime(Message,long)
        // sendMessageDelayed(Message,long)
      
        // 以上方法以 post 开头的允许你处理 Runnable 对象
        //sendMessage() 允许你处理 Message 对象(Message 里可以包含数据,)

        MyThread m = new MyThread();
        new Thread(m).start();
    }

    /**
    * 接受消息,处理消息 ,此 Handler 会与当前主线程一块运行
    * */

    class MyHandler extends Handler {
        public MyHandler() {
        }

        public MyHandler(Looper L) {
            super(L);
        }

        // 子类必须重写此方法,接受数据
        @Override
        public void handleMessage(Message msg) {
            // TODO Auto-generated method stub
            Log.d("MyHandler", "handleMessage......");
            super.handleMessage(msg);
            // 此处可以更新 UI
            Bundle b = msg.getData();
            String color = b.getString("color");
            MyHandlerActivity.this.button.append(color);

        }
    }

    class MyThread implements Runnable {
        public void run() {

            try {
                Thread.sleep(10000);
            } catch (InterruptedException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }

            Log.d("thread.......", "mThread........");
            Message msg = new Message();
            Bundle b = new Bundle();// 存放数据
            b.putString("color", "我的");
            msg.setData(b);

            MyHandlerActivity.this.myHandler.sendMessage(msg); // 向 Handler 发送消息,更新 UI

        }
    }

} 

简单的说,Activity 的 onCreate 方法里启动一个线程,在这个线程的 run 方法里使用一个 Message 对象并使用 Handler 的 sendMessage 方法发送到队列中,最后在 Activity 里 new 一个 Handler 的内部类实现 handMessage 方法,使用这个方法把队列中的 Message 对象取出来以实现异步操作。

然后是 post 的例子,这里稍微说一下,直接使用 new Handler().post(Runnable) 这样的写法并没有新开线程,也就是说依然是在主线程中执行,相当于简单调用了线程的 run 方法而不是 start 方法。这个有人说是 android 的 bug,解决方案是这样使用:

HandlerThread handlerThread = new HandlerThread("myHandlerThread");  
handlerThread.start();  
handler = new Handler(handlerThread.getLooper()); 

来看一个完整的 post 例子:

public class MyThread extends Activity {    
    private Handler handler = null;    
    @Override    
    public void onCreate(Bundle savedInstanceState) {    
        super.onCreate(savedInstanceState);    
        HandlerThread handlerThread = new HandlerThread("myHandlerThread");    
        handlerThread.start();    
        handler = new Handler(handlerThread.getLooper());    
        handler.post(new MyRunnable());    
        System.out.println("Oncreate---The Thread id is :"    
                + Thread.currentThread().getId());    
        setContentView(R.layout.main);    
    }    
    private class MyRunnable implements Runnable {    
        public void run() {    
            System.out.println("Runnable---The Thread is running");    
            System.out.println("Runnable---The Thread id is :"    
                    + Thread.currentThread().getId());    
            try {    
                Thread.sleep(6000);    
            } catch (InterruptedException e) {    
                // TODO Auto-generated catch block    
                e.printStackTrace();    
            }    
        }    
    }    
}   

在这个 demo 中,用到了 HandlerThread,在 HandlerThread 对象中可以通过 getLooper 方法获取一个 Looper 对象控制句柄,我们可以将其这个 Looper 对象映射到一个 Handler 中去来实现一个线程同步机制。于是就有以下结果;

1:控制台的输出:

Oncreate---The Thread id is :1
Runnable---The Thread is running
Runnable---The Thread id is :10

2:程序启动后,我们立刻看到 main.xml 中的内容。这样就达到了多线程的结果。

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据

关于作者

0 文章
0 评论
23 人气
更多

推荐作者

我们的影子

文章 0 评论 0

素年丶

文章 0 评论 0

南笙

文章 0 评论 0

18215568913

文章 0 评论 0

qq_xk7Ean

文章 0 评论 0

    我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
    原文