7.1 基础知识Android消息处理机制
1. Android消息处理机制: Handler, MessageQueue, Looper, Thread
线程概念 : 一个应用程序运行时它的主体被称为进程,
一个进程内部可以有多个线程,
线程共享进程的资源
线程间通信
在android系统中是怎么封装通讯的,假如存在两个线程A和B,如果A线程要告诉B线程一些消息,怎么实现?
A进程怎么发发消息?(1)构造消息,消息里面有数据信息和处理函数(2)发消息;这两个步骤在android源码中被封装成Handler
消息在android中被封装为Message,A发送Message给B,可能B处理不过来,因此A把消息放到B进程里面的消息队列MessageQueue类,在B进程的循环体中做什么事?
(1)、从队列中取出消息;(2)处理消息,执行消息的处理函数;这两个步骤在android源码中被封装成Looper
Looper源码在frameworks/base/core/java/android/os/Looper.java
Handler源码在frameworks/base/core/java/android/os/Handler.java //ctrl+shift+n在输入handler.java来打开
a. 创建MessageQueue: Looper.prepare()
b. 使用Handler构造、发送Message
b.1 new Handler //创建handler的时候可以指定Looper即消息接受者和Callback回调函数即消息的处理函数,如果不知道肯定有默认的Looper和消息处理函数
b.2 Handler.sendMessage, sendEmptyMessageAtTime, sendMessageDelayed
c. 使用Looper循环处理消息:在loop函数中有个for死循环
c.1 从MessageQueue中取出Message,//Message msg = queue.next();
c.2 执行它的处理函数: msg.target.dispatchMessage(msg)//target就是Handler,dispatchMessage会调用处理函数
应用程序编写:
功能说明:app中创建一个button,创建子线程,添加消息处理功能,并且给Button添加处理函数,并且主线程会监测button,当button按下时给子线程发送消息,子线程收到消息后打印出来
import android.view.View;
import android.util.Log;//打印导入的库
public class MainActivity extends AppCompatActivity{
private Button mButton;
private final String TAG = "MessageTest";
private int ButtonCount = 0;
private Thread myThread;
private MyThread myThread2;
private Handler mHandler;
private int mMessageCount = 0;
class MyRunnable:implements Runable{
int count=0;
public void run(){
for(;;){
Log.d(TAG,"MyThread"+(count++));
try{
Thread.sleep(3000);
}catch(INterrupteException e){
e.printStackTrace();
}
}
}
}
class Mythread extends Thread{ //android只带的消息处理是在HandlerThread.java中,其功能同我们怎么的Mythread,如果不自己创建Mythread,可以直接使用HandlerThread类
public void run(){
super.run();
Looper.prepare();//创建消息队列
Looper.loop();//从消息队列中取出消息,调用消息的处理函数
}
public Looper getLooper(){
return Looper.myLooper();
}
}
pretected void onCreate(Bundle savedInstanceState){
........
........
mButton = (Button)findViewById(R.id.button);//双击Button,按下shift+F1,可以查看类的帮助文档
mButton.setOnClickListener(new View.OnClickListener(){
public void onClick(View v){
Log.d(TAG,"SendMessage"+(ButtonCount++) );
Message msg = new Message();
mHandler.sendMessage(msg );//发送消息的会从把消息放到消息队列中,消息队列是从handler创建的时候从Looper中获得
}
});
myThread = new Thread(new MyRunnable,"MessageTestThread");//(runnable接口的方法run是线程的主体函数)
myThread.start();
myThread2 = new MyThread();
myThread2 .start();//功能和myThread一样,start会导致MyThread的run函数被执行,run函数会执行Looper.prepare()去创建消息队列,但这个函数并不一定马上就执行,如果住进程中创建Handler的时候没有队列,会存在风险,所以修改class MyThread extends Thread{}
////我们现在的代码里面有三个线程了,主线程、myThread和myThread2,,要确定消息发给谁Looper
mHandler = new Handler(myThread2.getLooper(),new Handler.Callback(){
public boolean handleMessage(Message msg){
Log.d(TAG,"getMessage"+(mMessageCount++))
return false;
}
});
}
}
针对上面所说的风险修改class MyThread extends Thread{},如下:
class MyThread extends Thread{
private Looper mLooper;
public void run(){
super.run();
Looper.prepare();//创建消息队列
mLooper = Looper.myLooper();
notifyAll();//当主线程调用getLooper,在mLooper 为空的时候会休眠,所以这里需要唤醒休眠的线程
Looper.loop();//从消息队列中取出消息,调用消息的处理函数
}
public Looper getLooper(){
if(!isAlive()){
return null;
}
synchronized(this){
while(isAlive() && mLooper == null){
try{
wait();
}catch (InterruptedException e){
}
}
}
return mLooper;
}
}
修改代码使用系统只带的消息处理线程:
1、导入包:import android.os.HandlerThread;
2、在public class MainActivity extends AppCompatActivity中新增
private HandlerThread myThread3;
private Handler mHandler3;
myThread3 = new HandlerThread("MessageTestThread3");
myThread3 .start();
mHandler3 = new Handler(myThread3.getLooper());
在button的onClick函数中添加:
mHandler3.post(new Runnable(){
public void run(){
Log.d(TAG,"getMessage for Thread3"+(mMessageCount++));
}
});
7.1 基础知识Android消息处理机制的更多相关文章
- 【Android 开发】: Android 消息处理机制之一: Handler 与 Message
最近几讲内容,我们学习了Android中关于多线程的一些知识,上一讲我们讲解了异步任务 AsyncTask 的操作,Android中还提供了其他的线程操作,如Handler Message Messa ...
- (转)Android消息处理机制(Handler、Looper、MessageQueue与Message)
转自 http://www.cnblogs.com/angeldevil/p/3340644.html Android消息处理机制(Handler.Looper.MessageQueue与Messag ...
- Android消息处理机制
Android消息处理机制 Android应用程序消息处理机制(深入到native,实际由管道实现-pipe&epoll)
- 解析Android消息处理机制:Handler/Thread/Looper & MessageQueue
解析Android消息处理机制 ——Handler/Thread/Looper & MessageQueue Keywords: Android Message HandlerThread L ...
- Android架构分析之Android消息处理机制(二)
作者:刘昊昱 博客:http://blog.csdn.net/liuhaoyutz Android版本号:4.4.2 在上一篇文章中我们看了一个使用Handler处理Message消息的样例,本文我们 ...
- Android架构分析之Android消息处理机制(一)
作者:刘昊昱 博客:http://blog.csdn.net/liuhaoyutz Android版本号:4.4.2 在这个系列文章中我们将来分析Android消息处理机制. 本文介绍了一个使用Han ...
- Android消息处理机制(Handler 与Message)---01
一.handler的使用场景为么会有handler?(部分内容图片摘自http://www.runoob.com/w3cnote/android-tutorial-handler-message.ht ...
- [旧][Android] 消息处理机制
备注 原发表于2016.06.06,资料已过时,仅作备份,谨慎参考 概述 Android 的消息处理机制主要是指 Handler 的运行机制以及 Handler 所附带的 MessageQueue 和 ...
- 【Android】Android消息处理机制
三大核心类 android的消息处理有三个核心类:Looper,Handler和Message. 其实还有一个Message Queue(消息队列),但是MQ被封装到Looper里面了 Looper ...
随机推荐
- Android 学习笔记进阶十二之裁截图片
package xiaosi.cut; import java.io.File; import android.app.Activity; import android.content.Intent; ...
- js---对象 和 函数this
一:对象创建的方法 //普通 字面量形式 var obj = { name:'名字', fn:function(){ console.log(this.name); } } //new 实例 var ...
- 71.lambda表达式的递归
#include <iostream> #include <functional> using namespace std; void main() { //&调用外部 ...
- Kinect 开发 —— 用户交互设计的若干思考
Metro 风格 windows 8 Kinect Hub 手势原型设计 悬停选择 翻页控制 关节点重叠的处理方法 将箭靶设置在画面的边缘,这样玩家持弓的角度与屏幕保持一个大约45度的锐角,这 ...
- 搭建Spark源码研读和代码调试的开发环境
转载自https://github.com/linbojin/spark-notes/blob/master/ide-setup.md 搭建Spark源码研读和代码调试的开发环境 工欲善其事,必先利其 ...
- C_数组详解
数组: 一 一维数组 1.1 一维数组的定义: 类型符 数组名[常量表达式]; int a[10]; 说明: 1.数组的命名规则遵循标识符命名规则. 2.定义时需要指定元素的个数.方括号里的常量表达式 ...
- vue 使用同一组件,切换时不触发created、mounted钩子
两个页面参数不同使用同一组件,默认情况下当这两个页面切换时并不会触发created或者mounted钩子. 方法一:通过watch $route的变化来做处理 watch: { $route() { ...
- linux系统常用日志
系统日志记录着系统运行中的记录信息,在服务或者系统发生故障的时候,通过查询系统日志,可以帮助我们诊断.系统日志可以预警安全问题,系统日志一般都存放在/var/log目录下 /var/log/dmesg ...
- Python 3 与"Hello World!"
Python 3 版本 Python的3.0版本,常被称为Python 3000,或简称Py3k.相对于Python的早期版本,这是一个较大的升级.为了不带入过多的累赘,Python 3.0在设计的时 ...
- 【Hello 2018 B】Christmas Spruce
[链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 写个dfs看看是不是每个节点都有3个叶子节点就可以了. [代码] #include <bits/stdc++.h> us ...