Android 中 Activity 的 onCreate 方法里面子线程为何能设置UI界面

2024-05-11 00:48

本文主要是介绍Android 中 Activity 的 onCreate 方法里面子线程为何能设置UI界面,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

如果之前你没有尝试过 onCreate 方法里面用子线程的 run 方法去设置 UI (比如对 Textview 进行 setText 操作),相信你看到这个标题,也会感到困惑和好奇吧。

废话不多说,先来个 Demo 。

protected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);tv = (TextView) findViewById(R.id.tv);iv = (ImageView) findViewById(R.id.iv);bt = (Button) findViewById(R.id.bt);Thread thread = new Thread("Thread1") {public void run() {tv.setText(System.currentTimeMillis() + "33333333");tv.append(Thread.currentThread().getName());iv.setImageResource(R.drawable.ic_launcher);};};thread.start();
}

代码很简单,就是在 onCreate 方法里面 new 了一个 Thread,然后在 run 方法里面进行 UI 操作。

运行结果如下:

没错,你要相信这正是运行结果。确实 TextView 的文本设置为当前时间的 long 型,还有子线程的名字。ImageView 也设置了默认图标

那,不是说 android 不允许子线程更新UI吗?

在这里,我们不妨再加一个 Button,在 Button 点击事件里面通过子线程来更新UI

onClick 方法代码如下:

public void onClick(View v) {// TODO 自动生成的方法存根Thread thread = new Thread("Thread1") {public void run() {tv.setText(System.currentTimeMillis() + "\n");tv.append("thread name:"+ Thread.currentThread().getName());iv.setImageResource(R.drawable.ic_launcher);};;thread.start();
}

编译后运行结果来看:

编译后程序可以启动,并且没有挂掉,界面上 TextView 和 ImageView 显示与上图一致。

但是我们点击按钮后发现程序出现 “ Unforunately,Study has stopped.” 的字样,程序挂掉了。

不难看出,onClick 里面子线程更新UI与我们的预期结果一样,但是为什么 onCreate 方法里面却没有挂掉呢。

 

在此,我选择查看一下 setText 方法,发现 setText 里面调用了 checkForRelayout();,而在 checkForRelayout 方法里面调用了invalidate()这个方法

再追踪下去,是 invalidate 方法调用了 ViewGroup.invalidateChild,最终调用 ViewRootImpl.checkThread()。

ViewRootImpl 是一个隐藏类,我们只能去看 framework 的源码,源码如下:

void checkThread() {if (mThread != Thread.currentThread()) {throw new CalledFromWrongThreadException("Only the original thread that created a view hierarchy can touch its views.");}
}

mThread是 UI 线程,这里会检查当前线程是不是 UI 线程。那么为什么 onCreate 里面没有进行这个检查呢。

 

这个问题原因出现在 Activity 的生命周期中,在 onCreate 方法中,UI处于创建过程,对用户来说界面还不可视,直到onStart方法后界面可视了,再到onResume方法后界面可以交互,。

从某种程度来讲,在onCreate方法中进行setText不能算是更新UI,只能说是配置 UI,或者是设置 UI 的属性。这个时候并不会调用 invalidate,也就是不会调用到 ViewRootImpl.checkThread() 。而在 onResume 方法后,ViewRootImpl 才被创建。这个时候去交互界面以及算是更新 UI 了,这个时候 ViewRootImpl.checkThread() 就会报错了。

不信,我们把子线程休眠200ms,代码如下:

Thread thread = new Thread("Thread1") {public void run() {try {Thread.sleep(200);} catch (InterruptedException e) {// TODO 自动生成的 catch 块e.printStackTrace();}tv.append( System.currentTimeMillis() + "\n");tv.append("thread name:" + Thread.currentThread().getName());iv.setImageResource(R.drawable.ic_launcher);};
};
thread.start();

运行结果是程序出现“Unforunately,Study has stopped

 

简单的来说,就是多线程下的问题,在 activity 的生命周期里各个方法的执行时间都有影响,而UI线程检查是由 ViewRootImpl 类调用的,而 ViewRootImpl 只有在 onRsume 方法后才会被创建。

这篇关于Android 中 Activity 的 onCreate 方法里面子线程为何能设置UI界面的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/978042

相关文章

Oracle 通过 ROWID 批量更新表的方法

《Oracle通过ROWID批量更新表的方法》在Oracle数据库中,使用ROWID进行批量更新是一种高效的更新方法,因为它直接定位到物理行位置,避免了通过索引查找的开销,下面给大家介绍Orac... 目录oracle 通过 ROWID 批量更新表ROWID 基本概念性能优化建议性能UoTrFPH优化建议注

Pandas进行周期与时间戳转换的方法

《Pandas进行周期与时间戳转换的方法》本教程将深入讲解如何在pandas中使用to_period()和to_timestamp()方法,完成时间戳与周期之间的转换,并结合实际应用场景展示这些方法的... 目录to_period() 时间戳转周期基本操作应用示例to_timestamp() 周期转时间戳基

在 PyQt 加载 UI 三种常见方法

《在PyQt加载UI三种常见方法》在PyQt中,加载UI文件通常指的是使用QtDesigner设计的.ui文件,并将其转换为Python代码,以便在PyQt应用程序中使用,这篇文章给大家介绍在... 目录方法一:使用 uic 模块动态加载 (不推荐用于大型项目)方法二:将 UI 文件编译为 python 模

Python将字库文件打包成可执行文件的常见方法

《Python将字库文件打包成可执行文件的常见方法》在Python打包时,如果你想将字库文件一起打包成一个可执行文件,有几种常见的方法,具体取决于你使用的打包工具,下面就跟随小编一起了解下具体的实现方... 目录使用 PyInstaller基本方法 - 使用 --add-data 参数使用 spec 文件(

Python的pip在命令行无法使用问题的解决方法

《Python的pip在命令行无法使用问题的解决方法》PIP是通用的Python包管理工具,提供了对Python包的查找、下载、安装、卸载、更新等功能,安装诸如Pygame、Pymysql等Pyt... 目录前言一. pip是什么?二. 为什么无法使用?1. 当我们在命令行输入指令并回车时,一般主要是出现以

通过C#获取Excel单元格的数据类型的方法详解

《通过C#获取Excel单元格的数据类型的方法详解》在处理Excel文件时,了解单元格的数据类型有助于我们正确地解析和处理数据,本文将详细介绍如何使用FreeSpire.XLS来获取Excel单元格的... 目录引言环境配置6种常见数据类型C# 读取单元格数据类型引言在处理 Excel 文件时,了解单元格

Android NDK版本迭代与FFmpeg交叉编译完全指南

《AndroidNDK版本迭代与FFmpeg交叉编译完全指南》在Android开发中,使用NDK进行原生代码开发是一项常见需求,特别是当我们需要集成FFmpeg这样的多媒体处理库时,本文将深入分析A... 目录一、android NDK版本迭代分界线二、FFmpeg交叉编译关键注意事项三、完整编译脚本示例四

MySQL连接池(Pool)常用方法详解

《MySQL连接池(Pool)常用方法详解》本文详细介绍了MySQL连接池的常用方法,包括创建连接池、核心方法连接对象的方法、连接池管理方法以及事务处理,同时,还提供了最佳实践和性能提示,帮助开发者构... 目录mysql 连接池 (Pool) 常用方法详解1. 创建连接池2. 核心方法2.1 pool.q

Spring Boot Controller处理HTTP请求体的方法

《SpringBootController处理HTTP请求体的方法》SpringBoot提供了强大的机制来处理不同Content-Type​的HTTP请求体,这主要依赖于HttpMessageCo... 目录一、核心机制:HttpMessageConverter​二、按Content-Type​处理详解1.

Android与iOS设备MAC地址生成原理及Java实现详解

《Android与iOS设备MAC地址生成原理及Java实现详解》在无线网络通信中,MAC(MediaAccessControl)地址是设备的唯一网络标识符,本文主要介绍了Android与iOS设备M... 目录引言1. MAC地址基础1.1 MAC地址的组成1.2 MAC地址的分类2. android与I