1.Android进程

一个应用程序被启动时,系统默认创建执行一个叫做"main"的线程。这个线程也是你的应用与界面工具包(android.widget和android.view包中的组件)交互的地方。于是main线程也被称为界面线程。

这种单线程的模式会带来低性能,除非你能正确的优化你的程序。

打 个比方:用户触摸屏幕上的一个按钮时的点击事件即向线程中派发事件,比如每一个事件为一辆车。而每一条线程都好像是一条单行线的单车道。这条车道上的车量 都排成队行驶至收费口进行事件处理。当处理的事件繁琐,比如在响应用户交互时需执行大量运算,或者像是执行网络连接、数据库请 求这样耗时的操作。就会造成 拥堵,将会阻止整个界面的响应。当线程被阻塞时,就不能派发事件了。从用户的角度看,程序反应太慢了。甚至更糟的是,如果界面线程被阻塞几秒钟(大5秒钟 吧),用户就户抱怨说程序没反应了,用户可能因而退出并删掉你的应用。 此外,Andoid界面不是线程安全的。所以你绝不能在一个工作线程中操作你的界面—你只能在界面线程中管理的你的界面。所以,对于单线程模式有两个简单 的规则

1不要阻塞界面线程

2不要在界面线程之外操作界面。

2. 工作线程

由于上述的单线程模式,不要阻塞你的界面线程以使你的应用的界面保持响应是非常重要的,那么如果你有不能很快完成的任务,你应把它们放在另一个线程中执行(后台线程或工作线程)。

例如,下面是的代码是响应click事件,在另外一个线程中从网络获取资源文字并以TextView来显示。

 1     mHandle.setOnClickListener(new OnClickListener() {
2
3 @Override
4 public void onClick(View v) {
5 new Thread(new Runnable() {
6
7 @Override
8 public void run() {
9 // 耗时操作
10 loadNetWork();
11 mTextView.setText(来自网络的文字);
12 }
13 });
14
15 }
16 });

第一眼,这看起来能很好的工作,因为它创建了一个新线程来进行网络操作。然而它违反了第二条规则:不要在界面线程之外操作界面—它简单的在工作线程中修改了mTextView。这会导至未定义的异常出现,并且难以调试追踪。

为了能改正这个问题,Android提供了很多从其它线程来操作界面的方法。下面是可用的方法们:

1 Activity.runOnUiThread(Runnable)

2 View.post(Runnable)

3 View.postDelayed(Runnable,long)

4 Handler

5 AsyncTask

现在我们就依次使用这几个方法:

1、Activity.runOnUiThread(Runnable)

 1     mHandle.setOnClickListener(new OnClickListener() {
2
3 @Override
4 public void onClick(View v) {
5 new Thread(new Runnable() {
6
7 @Override
8 public void run() {
9 MainActivity.this.runOnUiThread(new Runnable() {
10 // 耗时操作
11 loadNetWork();
12 @Override
13 public void run() {
14 mTextView.setText(来自网络的文字);
15 }
16 });
17
18 }
19 });
20
21 }
22 });

2、 View.post(Runnable)

 1 mHandle.setOnClickListener(new OnClickListener() {
2
3 @Override
4 public void onClick(View v) {
5 new Thread(new Runnable() {
6
7 @Override
8 public void run() {
9 // 耗时操作
10 loadNetWork();
11 mTextView.post(new Runnable() {
12
13 @Override
14 public void run() {
15 mTextView.setText(来自网络的文字);
16 }
17 });
18
19 }
20 });
21 }
22 });

3、View.postDelayed(Runnable,long)

 1     mHandle.setOnClickListener(new OnClickListener() {
2
3 @Override
4 public void onClick(View v) {
5 new Thread(new Runnable() {
6
7 @Override
8 public void run() {
9 // 耗时操作
10 loadNetWork();
11 mTextView.postDelayed(new Runnable() {
12
13 @Override
14 public void run() {
15 mTextView.setText(来自网络的文字);
16 }
17 }, 10);
18
19 }
20 });
21
22
23 }
24 });

4、Handler(子线程调用Handler的handle.sendMessage(msg);

 1     Handler handle = new Handler() {
2
3 @Override
4 public void handleMessage(Message msg) {
5 super.handleMessage(msg);
6 mTextView.setText(来自网络的文字);
7 }
8
9 };
10
11 class MyThread extends Thread {
12
13 @Override
14 public void run() {
15 // 耗时操作
16 loadNetWork();
17
18 Message msg = new Message();
19 handle.sendMessage(msg);
20 super.run();
21 }
22
23
24 }

5、AsyncTask

主线程中:aTask ak = new aTask();

ak.execute();

然后:

 1         private class aTask extends AsyncTask {
2
3 //后台线程执行时
4 @Override
5 protected Object doInBackground(Object... params) {
6 // 耗时操作
7 return loadNetWork();
8 }
9 //后台线程执行结束后的操作,其中参数result为doInBackground返回的结果
10 @Override
11 protected void onPostExecute(Object result) {
12 super.onPostExecute(result);
13 mTextView.setText(result);
14 }
15 }

子线程中如何修改ui界面的更多相关文章

  1. 子线程中刷新了UI

    This application is modifying the autolayout engine from a background thread, which can lead to engi ...

  2. 使用Handler在子线程中更新UI

    Android规定仅仅能在主线程中更新UI.假设在子线程中更新UI 的话会提演示样例如以下错误:Only the original thread that created a view hierach ...

  3. Android 在子线程中更新UI

    今天在做练习时,在一个新开启的线程中调用“Toast.makeText(MainActivity.this, "登陆成功",Toast.LENGTH_SHORT).show();” ...

  4. C#子线程中更新ui

    本文实例总结了C#子线程更新UI控件的方法,对于桌面应用程序设计的UI界面控制来说非常有实用价值.分享给大家供大家参考之用.具体分析如下: 一般在winform C/S程序中经常会在子线程中更新控件的 ...

  5. iOS 报错:(子线程中更新UI)This application is modifying the autolayout engine from a background thread after the engine was accessed from the main thread. This can lead to engine corruption and weird crashes.

    今天在写程序的时候,使用Xcode 运行工程时报出下面的错误错信息,我还以为是什么呢,好久没遇到过这样的错误了. **ProjectName[1512:778965] This application ...

  6. Android多线程之(一)View.post()源码分析——在子线程中更新UI

    提起View.post(),相信不少童鞋一点都不陌生,它用得最多的有两个功能,使用简便而且实用: 1)在子线程中更新UI.从子线程中切换到主线程更新UI,不需要额外new一个Handler实例来实现. ...

  7. C#-WinForm跨线程修改UI界面

    待解决的问题 在我做WinForm开发的过程中,经常会遇到耗时操作或阻塞操作.他们会引发软件的卡顿甚至假死,严重影响软件的使用. 因此,这类耗时或阻塞的操作一般都会使用异步的方式去执行,不影响主线程( ...

  8. 老问题:Android子线程中更新UI的3种方法

    在Android项目中经常有碰到这样的问题,在子线程中完成耗时操作之后要更新UI,下面就自己经历的一些项目总结一下更新的方法: 方法一:用Handler 1.主线程中定义Handler: Handle ...

  9. android 不能在子线程中更新ui的讨论和分析

    问题描写叙述 做过android开发基本都遇见过 ViewRootImpl$CalledFromWrongThreadException,上网一查,得到结果基本都是仅仅能在主线程中更改 ui.子线程要 ...

随机推荐

  1. Sublime Text 3 build 3103 注册码

    分享几个ST3的注册码,第一个我注册到自己电脑上,亲测可用,剩余几个没有测试.→原文链接 -– BEGIN LICENSE -–Michael BarnesSingle User LicenseEA7 ...

  2. 写出Zend 框架的目录结构,简单说明目录作用?

    application/ – 存放应用程序的目录,包括MVC 系统.配置文件.服务以及引导程序(Bootstrap.php)configs/ –配置文件目录.application/modules – ...

  3. ArcMap操作随记(2)

    1.空间校正 变换-仿射 仅发生偏移 橡皮页变化 形状改变 变换-投影 旋转,改变角度 变换-相似 改变大小.形状不变 2.计算行列号 Int(([POINT_Y]-1273.143242)/30)+ ...

  4. linux内存(一) 内核空间与用户空间

    来自如下网站 https://www.cnblogs.com/sparkdev/p/8410350.html 内核空间和用户空间 对 32 位操作系统而言,它的寻址空间(虚拟地址空间,或叫线性地址空间 ...

  5. [SPDK/NVMe存储技术分析]012 - 用户态ibv_post_send()源码分析

    OFA定义了一组标准的Verbs,并提供了一个标准库libibvers.在用户态实现NVMe over RDMA的Host(i.e. Initiator)和Target, 少不了要跟OFA定义的Ver ...

  6. 后门及持久化访问3----进程注入之AppInit_DLLs注册表项

    进程注入之AppInit_DLLs注册表项 User32.dll被加载到进程时,会获取AppInit_DLLs注册表项,若有值,则调用LoadLibrary() API加载用户DLL.只会影响加载了u ...

  7. Mysql学习笔记-InnoDB深度解析

    前言 我们在上一篇博客聊了Mysql的整体架构分布,连接层.核心层.存储引擎层和文件系统层,其中存储引擎层作为Mysql Server中最重要的一部分,为我们sql交互提供了数据基础支持.存储引擎和文 ...

  8. Linux下面有7个运行等级  run level

    Linux下面有7个运行等级  run level run level 0  系统停机状态,系统默认运行级别不能设为0,否则不能正常启动 run level 1  单用户工作状态,root权限,用于系 ...

  9. Windows 7/8 64位系统 不能注册32位dll 文件的解决方案

    这几天碰到一个问题,运行一个易语言开发的软件出现以下错误.我的系统是 Windows7 64 位 专业版.在系统盘 windows/system32  下查找 dm.dll.但是没有这个文件.于是我到 ...

  10. CRI容器运行时

    CRI容器运行时 我们知道 Kubernetes 提供了一个 CRI 的容器运行时接口,那么这个 CRI 到底是什么呢?这个其实也和 Docker 的发展密切相关的. 在 Kubernetes 早期的 ...