Handler 基本用法--线程间传值
Handler 作用:主要接受子线程发送的数据, 并用此数据配合主线程更新UI.
解释:
当应用程序启动时,Android首先会开启一个主线程 (也就是UI线程) , 主线程为管理界面中的UI控件,进行事件分发, 比如说, 你要是点击一个 Button ,Android会分发事件到Button上,来响应你的操作。 如果此时需要一个耗时的操作,例如: 联网读取数据, 或者读取本地较大的一个文件的时候,你不能把这些操作放在主线程中,,如果你放在主线程中的话,界面会出现假死现象, 如果5秒钟还没有完成的话,会收到Android系统的一个错误提示 "强制关闭".
这个时候我们需要把这些耗时的操作,放在一个子线程中,因为子线程涉及到UI更新,Android主线程是线程不安全的,也就是说,更新UI只能在主线程中更新,子线程中操作是危险的. 会出现如下异常:
即:只有创建该控件的线程才可以操作该控件
这个时候,Handler就出现了 。由于Handler运行在主线程中(UI线程中), 它与子线程可以通过Message对象来传递数据, 这个时候,Handler就承担着接受子线程传过来的Message对象,(里面包含数据) , 把这些消息放入主线程队列中,配合主线程进行更新UI。
例子如下:
1》布局文件:activity_main.xml
<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"android:paddingBottom="@dimen/activity_vertical_margin"android:paddingLeft="@dimen/activity_horizontal_margin"android:paddingRight="@dimen/activity_horizontal_margin"android:paddingTop="@dimen/activity_vertical_margin"tools:context="com.cxc.threadandhandler.MainActivity" ><TextViewandroid:layout_width="match_parent"android:layout_height="wrap_content"android:text="手动创建线程并与GUI同步 " /><LinearLayoutandroid:layout_width="match_parent"android:layout_height="wrap_content"android:orientation="horizontal" ><TextViewandroid:id="@+id/show_info_tv"android:layout_width="0dp"android:layout_height="wrap_content"android:layout_weight="1"android:text="显示后台任务情况" /><Buttonandroid:id="@+id/start_task_bt"android:layout_width="wrap_content"android:layout_height="wrap_content"android:text="Start Task" /></LinearLayout></LinearLayout>
效果如下:
2》MainActivity.java代码
代码中有很详细的备注
package com.cxc.handlerdemo;import android.app.Activity;
import android.os.Bundle;
import android.view.Menu;
import android.view.MenuItem;public class MainActivity extends Activity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);}@Overridepublic boolean onCreateOptionsMenu(Menu menu) {// Inflate the menu; this adds items to the action bar if it is present.getMenuInflater().inflate(R.menu.main, menu);return true;}@Overridepublic boolean onOptionsItemSelected(MenuItem item) {// Handle action bar item clicks here. The action bar will// automatically handle clicks on the Home/Up button, so long// as you specify a parent activity in AndroidManifest.xml.int id = item.getItemId();if (id == R.id.action_settings) {return true;}return super.onOptionsItemSelected(item);}
}
运行的结果为:
package com.cxc.threadandhandler;import android.app.Activity;
import android.os.Bundle;
import android.os.Handler;
import android.os.HandlerThread;
import android.os.Looper;
import android.os.Message;
import android.util.Log;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.TextView;public class MainActivity extends Activity {private static final String TAG = "com.cxc.threadandhandler.mainactivity";private static final int MESSAGE_KEY = 1001;private TextView show_info_tv;private Button start_task_bt;private Handler myHandler = new Handler() {@Overridepublic void handleMessage(Message msg) {// TODO Auto-generated method stubsuper.handleMessage(msg);if (msg.what == MESSAGE_KEY) {// 接收消息并更新UILog.d(TAG, "--thread:" + Thread.currentThread().getId()+"--reciver:"+msg.obj.toString());show_info_tv.setText(msg.obj.toString());}}};@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);Log.d(TAG, "---main---" + Thread.currentThread().getId());Log.d(TAG, "---onCreate---");initViews();}private void initViews() {Log.d(TAG, "---initViews---");show_info_tv = (TextView) findViewById(R.id.show_info_tv);start_task_bt = (Button) findViewById(R.id.start_task_bt);start_task_bt.setOnClickListener(myButtonClickListener);}OnClickListener myButtonClickListener = new OnClickListener() {@Overridepublic void onClick(View v) {// TODO Auto-generated method stubswitch (v.getId()) {case R.id.start_task_bt:// to-do// 将耗时的操作移到子线程中Log.d(TAG, "---Start Task Button Click---");BackgroundThread backgroundThread = new BackgroundThread();new Thread(backgroundThread).start();// 启动后台线程,以处理耗时任务break;default:break;}}};class BackgroundThread implements Runnable {public BackgroundThread() {// TODO Auto-generated constructor stub}@Overridepublic void run() {// TODO Auto-generated method stubLog.d(TAG, "-----work thread ---" + Thread.currentThread().getId());backgroundThreadProcessing();}// 在后台执行一些处理的方法---耗时操作private void backgroundThreadProcessing() {// to-do 耗时操作Log.d(TAG, "-----work thread -backgroundThreadProcessing ---"+ Thread.currentThread().getId());for (int i = 0; i < 10; i++) {String text = show_info_tv.getText().toString();text += ("#" + i);Message msg = new Message();msg.what = MESSAGE_KEY;//用于区分不同的消息msg.obj = text;//消息内容Log.d(TAG, "--thread:" + Thread.currentThread().getId()+"--send:"+text);myHandler.sendMessage(msg);//发送消息// 当前线程(后台子线程)休眠1000mstry {Thread.sleep(1000);} catch (InterruptedException e) {// TODO Auto-generated catch blocke.printStackTrace();}}}}
}
运行结果:
可以看到主线程(UI线程)已经接收到了子线程发送过来的数据。
这一点也可以从程序的Log看出来
06-02 16:25:19.299: D/com.cxc.threadandhandler.mainactivity(10304): ---main---1
06-02 16:25:19.299: D/com.cxc.threadandhandler.mainactivity(10304): ---onCreate---
06-02 16:25:19.299: D/com.cxc.threadandhandler.mainactivity(10304): ---initViews---
06-02 16:25:30.259: D/com.cxc.threadandhandler.mainactivity(10304): ---Start Task Button Click---
06-02 16:25:30.259: D/com.cxc.threadandhandler.mainactivity(10304): -----work thread ---6346
06-02 16:25:30.259: D/com.cxc.threadandhandler.mainactivity(10304): -----work thread -backgroundThreadProcessing ---6346
06-02 16:25:30.259: D/com.cxc.threadandhandler.mainactivity(10304): --thread:6346--send:显示后台任务情况#0
06-02 16:25:30.259: D/com.cxc.threadandhandler.mainactivity(10304): --thread:1--reciver:显示后台任务情况#0
06-02 16:25:31.259: D/com.cxc.threadandhandler.mainactivity(10304): --thread:6346--send:显示后台任务情况#0#1
06-02 16:25:31.259: D/com.cxc.threadandhandler.mainactivity(10304): --thread:1--reciver:显示后台任务情况#0#1
06-02 16:25:32.264: D/com.cxc.threadandhandler.mainactivity(10304): --thread:6346--send:显示后台任务情况#0#1#2
06-02 16:25:32.264: D/com.cxc.threadandhandler.mainactivity(10304): --thread:1--reciver:显示后台任务情况#0#1#2
06-02 16:25:33.264: D/com.cxc.threadandhandler.mainactivity(10304): --thread:6346--send:显示后台任务情况#0#1#2#3
06-02 16:25:33.264: D/com.cxc.threadandhandler.mainactivity(10304): --thread:1--reciver:显示后台任务情况#0#1#2#3
06-02 16:25:34.264: D/com.cxc.threadandhandler.mainactivity(10304): --thread:6346--send:显示后台任务情况#0#1#2#3#4
06-02 16:25:34.264: D/com.cxc.threadandhandler.mainactivity(10304): --thread:1--reciver:显示后台任务情况#0#1#2#3#4
06-02 16:25:35.264: D/com.cxc.threadandhandler.mainactivity(10304): --thread:6346--send:显示后台任务情况#0#1#2#3#4#5
06-02 16:25:35.264: D/com.cxc.threadandhandler.mainactivity(10304): --thread:1--reciver:显示后台任务情况#0#1#2#3#4#5
06-02 16:25:36.264: D/com.cxc.threadandhandler.mainactivity(10304): --thread:6346--send:显示后台任务情况#0#1#2#3#4#5#6
06-02 16:25:36.264: D/com.cxc.threadandhandler.mainactivity(10304): --thread:1--reciver:显示后台任务情况#0#1#2#3#4#5#6
06-02 16:25:37.269: D/com.cxc.threadandhandler.mainactivity(10304): --thread:6346--send:显示后台任务情况#0#1#2#3#4#5#6#7
06-02 16:25:37.269: D/com.cxc.threadandhandler.mainactivity(10304): --thread:1--reciver:显示后台任务情况#0#1#2#3#4#5#6#7
06-02 16:25:38.269: D/com.cxc.threadandhandler.mainactivity(10304): --thread:6346--send:显示后台任务情况#0#1#2#3#4#5#6#7#8
06-02 16:25:38.269: D/com.cxc.threadandhandler.mainactivity(10304): --thread:1--reciver:显示后台任务情况#0#1#2#3#4#5#6#7#8
06-02 16:25:39.269: D/com.cxc.threadandhandler.mainactivity(10304): --thread:6346--send:显示后台任务情况#0#1#2#3#4#5#6#7#8#9
06-02 16:25:39.269: D/com.cxc.threadandhandler.mainactivity(10304): --thread:1--reciver:显示后台任务情况#0#1#2#3#4#5#6#7#8#9
可以看出,主线程(UI线程)的线程ID为1,而我们自己启动的线程ID为6346,可以说明它们确实不在一个线程中。
关于线程的相关知识可以参考:Java线程之两种方法Runnable和Thread的区别
Handler 基本用法--线程间传值相关推荐
- handler回调主线程_Android使用Handler实现子线程与子线程、子线程与主线程之间通信...
转载:https://blog.csdn.net/shaoenxiao/article/details/54561753 今天这篇文章只讲一下怎么使用Handler实现子线程与子线程之间.子线程与主线 ...
- 线程间通讯机制(基础篇)——Handler、Runnable、HandlerThread、AsyncTask的使用
前言: android线程通讯机制是android应用开发的基础课程,对于很多初学android的朋友可能还没有完全理解,所以,今天我就做一下知识小结吧. 一.线程安全 可能有java基本的朋友都知道 ...
- 线程间通信: Handler , Looper, MessageQueue, Message (完结)
概述: 为了 线程间 通信方便, Handler 机制 通过 Handler 和 Looper, MessageQueue, Message 这些 类 之间的协作, 简化 多线程的开发. 线程 ...
- 源码分析Android Handler是如何实现线程间通信的
源码分析Android Handler是如何实现线程间通信的 Handler作为Android消息通信的基础,它的使用是每一个开发者都必须掌握的.开发者从一开始就被告知必须在主线程中进行UI操作.但H ...
- Java中condition的用法_java5 Condition用法--实现线程间的通信
Condition的功能类似在传统线程技术中的Object.wait()和Object.natify()的功能,传统线程技术实现的互斥只能一个线程单独干,不能说这个线程干完了通知另一个线程来干,Con ...
- 【Android】线程间通信——Handler消息机制
文章目录 引言 Java层 永动机跑起来 示例 Looper Handler MessageQueue 永动机停下 Native层 nativeInit() nativePollOnce() nati ...
- Handler线程间通信
异步更新之Handler.post解惑 在Android中使用Handler和Thread线程执行后台操作 对于线程的控制,我们将介绍一个 Handler类,使用该类可以对运行在不同线程中的多个任务进 ...
- Android线程间通信之handler
本文来一起讨论下Android的handler机制. 相信写过android的童鞋,一定对handler很熟悉.因为使用频率实在太高了.尤其是在非ui线程,想要刷新ui控件的时候.因为ui控件的刷新只 ...
- android线程间通信的几种方法_Android 技能图谱学习路线
Java基础 Java Object类方法 HashMap原理,Hash冲突,并发集合,线程安全集合及实现原理 HashMap 和 HashTable 区别 HashCode 作用,如何重载hashC ...
最新文章
- python编程零基础-编程零基础应当如何开始学习 Python?
- MariaDB的二进制包安装方法
- QT的QAssociativeIterable类的使用
- 《精通Windows Sockets网络开发--基于Visual C++实现》.(孙海民).[PDF]ckook
- NoteRenderer.render logic
- jap和java有关系吗_hibernate与jpa有什么区别和联系?
- 陶哲轩实分析 定理 13.1.5
- 大数据新手之路四:联合使用Flume和Kafka
- 如何对CentOS FTP服务配置 - 51CTO.COM
- 水晶报表切换语言_BI软件一键做报表,用过才知道有多快
- 使用内网穿透实现外网访问内网地址
- 亚信安全助手、杀毒软件卸载
- 沁恒MCU串口使用指南
- 研究生留学资助项目突遭暂停,美国两大科研机构设限,上千访问学者或受影响...
- 简易的安卓天气app(四)——搜索城市、完善页面
- VC++6.0安装、编译NTL类库
- PS练习3——渐变色
- IO系统性能之二:缓存和RAID如何提高磁盘IO性能
- word中段落里面的选项“如果定义了文档网格,则对齐到网格”起什么作用?
- IIS是什么?有什么用?怎么用?