1. Android进程

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

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

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

1不要阻塞界面线程

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

2. 工作线程

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

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

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

第一眼,这看起来能很好的工作,因为它创建了一个新线程来进行网络操作。然而它违反了第二条规则:不要在界面线程之外操作界面—它简单的在工作线程中修改了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. @Override
  3. public void onClick(View v) {
  4. new Thread(new Runnable() {
  5. @Override
  6. public void run() {
  7. MainActivity.this.runOnUiThread(new Runnable() {
  8. // 耗时操作
  9. loadNetWork();
  10. @Override
  11. public void run() {
  12. mTextView.setText(来自网络的文字);
  13. }
  14. });
  15. }
  16. });
  17. }
  18. });

2、View.post(Runnable)

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

3、View.postDelayed(Runnable,long)

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

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

  1. Handler handle = new Handler() {
  2. @Override
  3. public void handleMessage(Message msg) {
  4. super.handleMessage(msg);
  5. mTextView.setText(来自网络的文字);
  6. }
  7. };
  1. class MyThread extends Thread {
  2. @Override
  3. public void run() {
  4. // 耗时操作
  5. loadNetWork();
  6. Message msg = new Message();
  7. handle.sendMessage(msg);
  8. super.run();
  9. }
  10. }

5、AsyncTask

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

然后:

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

Android中进程与线程及如何在子线程中操作UI线程的更多相关文章

  1. Android 关于操作UI线程

    在非UI线程里访问 Android UI toolkit—这个在一个worker线程修改了 View .这会导致不可预期的结果,而且还难以调试. 为了修复这个问题,Android提供了几个方法从非UI ...

  2. Android 操作UI线程的一些方法

    我们经常会在后台线程中去做一些耗时的操作,比如去网络取数据.但是当数据取回来,需要显示到页面上的时候,会遇到一些小麻烦,因为我们都知道,android的UI页面是不允许在其他线程直接操作的.下面总结4 ...

  3. 在子jsp页面中调用父jsp中的function或父jsp调用子页面中的function

    项目场景: A.jsp中有一个window,window里嵌入了一个<iframe>,通过<iframe>引入了另一个页面B.jsp.在B.jsp中的一个function中需要 ...

  4. jquery 中选择当前标签下众多相同子标签中的第n个

    可以用jquery选择器的:eq选择器或者jquery遍历的eq()方法,下面带那给出ul下第4个li的内容 $("ul li:eq(3)")  // 元素的index位置工0开始 ...

  5. Android 的进程和线程

    进程和线程 如果某个应用程序组件是第一次被启动,且这时应用程序也没有其他组件在运行,则android系统会为应用程序创建一个包含单个线程的linux进程.默认情况下,同一个应用程序的所有组件都运行在同 ...

  6. Android中UI线程与后台线程交互设计的5种方法

    我想关于这个话题已经有很多前辈讨论过了.今天算是一次学习总结吧. 在android的设计思想中,为了确保用户顺滑的操作体验.一 些耗时的任务不能够在UI线程中运行,像访问网络就属于这类任务.因此我们必 ...

  7. Android中后台线程如何与UI线程交互

    我想关于这个话题已经有很多前辈讨论过了.今天算是一次学习总结吧. 在android的设计思想中,为了确保用户顺滑的操作体验.一些耗时的任务不能够在UI线程中运行,像访问网络就属于这类任务.因此我们必须 ...

  8. Android UI线程和非UI线程

    Android UI线程和非UI线程 UI线程及Android的单线程模型原则 当应用启动,系统会创建一个主线程(main thread). 这个主线程负责向UI组件分发事件(包括绘制事件),也是在这 ...

  9. Android异步处理一:使用Thread+Handler实现非UI线程更新UI界面

    Android应用的开发过程中需要把繁重的任务(IO,网络连接等)放到其他线程中异步执行,达到不阻塞UI的效果. 下面将由浅入深介绍Android进行异步处理的实现方法和系统底层的实现原理. 本文介绍 ...

随机推荐

  1. HDU1495 非常可乐 —— BFS + 模拟

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1495 非常可乐 Time Limit: 2000/1000 MS (Java/Others)    M ...

  2. html5--6-10 CSS选择器7--伪类选择器

    html5--6-10 CSS选择器7--伪类选择器 实例 学习要点 掌握常用的CSS选择器 了解不太常用的CSS选择器 什么是选择器 当我们定义一条样式时候,这条样式会作用于网页当中的某些元素,所谓 ...

  3. RandomUtils

    package com.cc.hkjc.util; import java.util.Random; public class RandomUtils {    /**     * 获取count个随 ...

  4. org.apache.hadoop.hbase.NotServingRegionException: Region is not online 错误

    当遇到如下错误的时候 可能以为是regionserver 挂掉或者其他原因导致连接不上regionserver  但后面提示了Hbase 表statistic_login 具体信息 Thu Jan 1 ...

  5. BroadcastReceiver中调用Service

    首先是代码: package com.larry.msglighter; import android.content.BroadcastReceiver; import android.conten ...

  6. BZOJ1499 单调队列+DP

    1499: [NOI2005]瑰丽华尔兹 Time Limit: 3 Sec  Memory Limit: 64 MBSubmit: 1560  Solved: 949[Submit][Status] ...

  7. JAVA RTTI

    基础类可接收我们发给派生类的任何消息,因为两者拥有完全一致的接口.我们要做的全部事情就是从派生上溯造型,而且永远不需要回过头来检查对象的准确类型是什么.所有细节都已通过多态性获得了完美的控制. 但经过 ...

  8. B. Color the Fence

    time limit per test 2 seconds memory limit per test 256 megabytes input standard input output standa ...

  9. B. Chris and Magic Square

    time limit per test 2 seconds memory limit per test 256 megabytes input standard input output standa ...

  10. k8s认证及ServiceAccount-十五

    一.ServiceAccount (1)简介 https://www.kubernetes.org.cn/service-account Service account是为了方便Pod里面的进程调用K ...