当前位置: 首页 > news >正文

Android组件通信——消息机制(二十六)

1. 消息机制

1.1 知识点

(1)掌握Message、Handler、Looper类的使用以及消息的传递;

(2)可以通过消息机制动态取得信息;

1.2 具体内容

对于android的消息机制,我们主要要使用Java中线程的一些知识:

线程:线程是进程一个细的划分,一个进程可以存在多个线程。Java中实现多线程的手段有两种:

       ·继承Thread类

       ·实现Runnable接口

在开发中,使用第二种方式实现多线程是优先选择的,主要继承Thread实现的多线程有一下两个问题:

       ·Java单继承的局限

       ·使用Runnable可以实现数据的共享

但是使用第二种方式实现的多线程,在线程启动的时候也必须使用Thread进行线程的启动。

主线程一般在android中成为UI线程,就是一个界面显示,那么这种就是主线程,而子线程就是利用那些实现了Runnable接口的线程的操作类。

对于Message和Handler类的操作,都会比较不太理解,现在完成一个更新的操作(子线程向主线程发送消息)。

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"android:layout_width="match_parent"android:layout_height="match_parent"android:orientation="vertical"><TextViewandroid:id="@+id/info"android:layout_width="wrap_content"android:layout_height="wrap_content"/>
</LinearLayout>

下面就是希望完成文本自动更新的操作,使用任务管理器完成(TimerTask)。

package com.example.messageproject;import java.util.Timer;
import java.util.TimerTask;import android.app.Activity;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.widget.TextView;public class MainActivity extends Activity {private TextView info = null;private static int count = 0;public static final int SET = 1;//定义消息的标记private Handler myHandler = new Handler(){@Overridepublic void handleMessage(Message msg) {switch(msg.what){case SET:MainActivity.this.info.setText("Wanczy-" + count++);break;}}};@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);super.setContentView(R.layout.activity_main);this.info = (TextView) super.findViewById(R.id.info);Timer timer = new Timer();//定义调度器timer.schedule(new MyTimerTask(), 0, 1000);//启动定时调度}/*** 定义了一个子线程* @author Administrator**/private class MyTimerTask extends TimerTask{@Overridepublic void run() {Message msg = new Message();//定义消息msg.what = SET;//定义操作标记MainActivity.this.myHandler.sendMessage(msg);//发送消息}}
}

对于这个程序而言,发现是在Handler中处理组件(TextView)的,为什么不在子线程中完成呢?

01-26 08:24:47.374: ERROR/AndroidRuntime(3533): android.view.ViewRoot$CalledFromWrongThreadException: Only the original thread that created a view hierarchy can touch its views.

这个错误指:子线程不能更新主线程中各个组件的状态。表示只要是子线程就无法去更新组件,那么现在只能采用之前的方法,在子线程中返回要操作的信息,而后主线程中利用Handler处理这些消息,从而实现线程的操作。

在正常的开发之中,不需要开发者去手动处理Looper,Activity类中会自动的启动好。

范例:Looper进行通讯操作:

package com.example.messageproject;import android.app.Activity;
import android.os.Bundle;
import android.os.Handler;
import android.os.Looper;
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 info = null;private Button but = null;public static final int SET = 1;//定义消息的标记@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);super.setContentView(R.layout.activity_main);this.info = (TextView) super.findViewById(R.id.info);this.but = (Button) super.findViewById(R.id.but);this.but.setOnClickListener(new OnClickListenerImpl());} private class OnClickListenerImpl implements OnClickListener{@Overridepublic void onClick(View v) {Looper looper = Looper.myLooper();//取得Looper对象MyHandler handler = new MyHandler(looper);handler.removeMessages(0);//清空队列中所有消息String data = "厦门万策智业科技有限公司(Wanczy)";Message msg = handler.obtainMessage(SET, data);handler.sendMessage(msg);//发送消息}}private class MyHandler extends Handler{public MyHandler(Looper looper){//用来接收Loopersuper(looper);}@Overridepublic void handleMessage(Message msg) {switch(msg.what){case SET:MainActivity.this.info.setText(msg.obj.toString());//取得消息的内容break;}}}
}

在程序中,去掉Looper之后,发现程序的运行也是一样的,说明Activity程序会自动启动Looper,很多时候不需要开发者去定义Looper。现在我们程序里面这3个关键类都有使用了,对于操作而言,以后就只需要用到Message 和Handler,下面我们来完成一个子线程与主线程的数据交互。

对于子线程,不能更新组件,所以接受到消息之后也只能进行后台的输出。

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"android:layout_width="match_parent"android:layout_height="match_parent"android:orientation="vertical"><TextViewandroid:id="@+id/info"android:layout_width="match_parent"android:layout_height="wrap_content"/><Buttonandroid:id="@+id/but"android:layout_width="match_parent"android:layout_height="wrap_content"android:text="交互"/>
</LinearLayout>
package com.example.messageproject;import android.app.Activity;
import android.os.Bundle;
import android.os.Handler;
import android.os.Looper;
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 {public static final int SETMAIN = 1; 			// 设置一个what标记public static final int SETCHILD = 2; 			// 设置一个what标记private Handler mainHandler, childHandler;		// 定义Handler对象private TextView msg;					// 文本显示组件private Button but;	class ChildThread implements Runnable {			// 子线程类@Overridepublic void run() {Looper.prepare(); 				// 初始化LooperMainActivity.this.childHandler = new Handler() {public void handleMessage(Message msg) {switch (msg.what) { 	// 判断what操作case SETCHILD: 		// 主线程发送给子线程的信息System.out.println("*** Main Child Message : "+ msg.obj);	// 打印消息Message toMain = MainActivity.this.mainHandler.obtainMessage(); 	// 创建MessagetoMain.obj = "\n\n[B] 这是子线程发给主线程的信息:"; // 设置显示文字toMain.what = SETMAIN; //设置主线程操作的状态码MainActivity.this.mainHandler.sendMessage(toMain); 					break;}}};Looper.loop();			// 启动该线程的消息队列}}@Overridepublic void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);super.setContentView(R.layout.activity_main); 		// 调用布局文件this.msg = (TextView) super.findViewById(R.id.info); 	// 取得组件this.but = (Button) super.findViewById(R.id.but); 	// 取得按钮this.mainHandler = new Handler() { 		// 主线程的Handler对象public void handleMessage(Message msg) {	// 消息处理switch (msg.what) { 	// 判断Message类型case SETMAIN: 		// 设置主线程的操作类MainActivity.this.msg.setText("主线程接收数据:"+ msg.obj.toString()); 	// 设置文本内容break;}}};new Thread(new ChildThread(), "Child Thread").start(); // 启动子线程this.but.setOnClickListener(new OnClickListenerImpl()) ; 	// 单击事件操作}private class OnClickListenerImpl implements OnClickListener {@Overridepublic void onClick(View view) {if (MainActivity.this.childHandler != null) { 	// 已实例化子线程HandlerMessage childMsg = MainActivity.this.childHandler.obtainMessage(); 		// 创建一个消息childMsg.obj = MainActivity.this.mainHandler.getLooper().getThread().getName()+ " --> Hello MLDN .";// 设置消息内容childMsg.what = SETCHILD; 	// 操作码MainActivity.this.childHandler.sendMessage(childMsg); // 向子线程发送}}}@Overrideprotected void onDestroy() {super.onDestroy();MainActivity.this.childHandler.getLooper().quit(); 	// 结束队列}}

范例:时钟显示

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"android:layout_width="match_parent"android:layout_height="match_parent"android:orientation="vertical"><AnalogClock android:id="@+id/myAnalogClock"android:layout_width="match_parent"android:layout_height="wrap_content"/><TextViewandroid:id="@+id/info"android:layout_width="match_parent"android:layout_height="wrap_content"/>
</LinearLayout>
package com.example.messageproject;import java.text.SimpleDateFormat;
import java.util.Date;import android.app.Activity;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.widget.TextView;public class MainActivity extends Activity {private TextView info = null;public static final int SET = 1;private Handler myHandler = new Handler(){@Overridepublic void handleMessage(Message msg) {switch (msg.what){case SET:MainActivity.this.info.setText("当前时间为:"+msg.obj.toString());break;}}};protected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);super.setContentView(R.layout.activity_main);this.info = (TextView) super.findViewById(R.id.info);new Thread(new ChildThread()).start();//启动子线程} private class ChildThread implements Runnable{public void run(){while(true){Message msg = MainActivity.this.myHandler.obtainMessage();msg.what = SET;msg.obj = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(new Date());MainActivity.this.myHandler.sendMessage(msg);//发送消息}}}
}
package com.example.messageproject;import java.text.SimpleDateFormat;
import java.util.Date;import android.app.Activity;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.widget.TextView;public class MainActivity extends Activity {private TextView info = null;public static final int SET = 1;private Handler myHandler = new Handler(){@Overridepublic void handleMessage(Message msg) {switch (msg.what){case SET:MainActivity.this.info.setText("当前时间为:"+msg.obj.toString());break;}}};protected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);super.setContentView(R.layout.activity_main);this.info = (TextView) super.findViewById(R.id.info);new Thread().start();//启动子线程} private class ChildThread implements Runnable{public void run(){while(true){Message msg = MainActivity.this.myHandler.obtainMessage();msg.what = SET;msg.obj = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(new Date());MainActivity.this.myHandler.sendMessage(msg);//发送消息}}}
}

1.3 小结

(1)在Android之中子线程不能直接对主线程的组件进行更新;

http://www.lryc.cn/news/191468.html

相关文章:

  • 《进化优化》第4章 遗传算法的数学模型
  • spring:详解spring MVC
  • 【Leetcode】207.课程表
  • Ubuntu18.04中QT安装下载安装pcl和vtk以及使用过程中踩过的坑
  • C++学习——对象数组、成员对象与封闭类
  • 解锁机器学习-梯度下降:从技术到实战的全面指南
  • day62:ARMday9,I2c总线通信
  • 【Python学习笔记】类型/运算/变量/注释
  • 国内常用源开发环境换源(flutter换源,python换源,Linux换源,npm换源)
  • 关于一篇什么是JWT的原理与实际应用
  • 【Method】把 arXiv论文 转换为 HTML5 网页
  • 每日一题AC
  • 后端:推荐 2 个 .NET 操作的 Redis 客户端类库
  • 华泰证券:京东营收增长或短期承压
  • Java从resources文件下载文档,文档没有后缀名
  • 【动手学深度学习-Pytorch版】BERT预测系列——BERTModel
  • Python之元组、字典和集合练习
  • 【数据结构】归并排序和计数排序(排序的总结)
  • 某医疗机构:建立S-SDLC安全开发流程,保障医疗前沿科技应用高质量发展
  • 验证二叉搜索树的后序遍历序列
  • 第三章 内存管理 一、内存的基础知识
  • 【Java学习之道】Java常用集合框架
  • logicFlow 流程图编辑工具使用及开源地址
  • ATF(TF-A)/OPTEE之动态代码分析汇总
  • 10-11 周三 shell xargs tr curl 做大事情
  • 1.1 向量与线性组合
  • django: You may need to add ‘localhost‘ to ALLOWED_HOSTS
  • 网络安全(黑客技术)—自学手册
  • 【Vue】之Vuex的入门使用,取值,修改值,同异步请求处理---保姆级别教学
  • ubuntu20.04 nerf Instant-ngp (下) 复现,自建数据集,导出mesh