热门标签 | HotTags
当前位置:  开发笔记 > Android > 正文

Android子线程与更新UI问题的深入讲解

首先和其他许多的GUI库一样,Android的UI线程是不安全的。所以下面这篇文章主要给大家介绍了关于Android子线程与更新UI问题的相关资料,需要的朋友可以参考借鉴,下面随着小编来一起学习学习吧

前言

在Android项目中经常有碰到这样的问题,在子线程中完成耗时操作之后要更新UI,下面就自己经历的一些项目总结一下更新的方法。话不多说了,来一起看看详细的介绍吧

引子:

情形1

 @Override
 protected void onCreate(Bundle savedInstanceState) {
 super.onCreate(savedInstanceState);
 setContentView(R.layout.activity_main);

 TextView textView = findViewById(R.id.home_tv);
 ImageView imageView = findViewById(R.id.home_img);

 new Thread(new Runnable() {
  @Override
  public void run() {
  textView.setText("更新TextView");
  imageView.setImageResource(R.drawable.img);
  }
 }).start();
 }

运行结果:正常运行!!!

情形二

 @Override
 protected void onCreate(Bundle savedInstanceState) {
 super.onCreate(savedInstanceState);
 setContentView(R.layout.activity_main);

 TextView textView = findViewById(R.id.home_tv);
 ImageView imageView = findViewById(R.id.home_img);

 new Thread(new Runnable() {
  @Override
  public void run() {
  try {
   Thread.sleep(5000);
  } catch (InterruptedException e) {
   e.printStackTrace();
  }
  textView.setText("更新TextView");
  imageView.setImageResource(R.drawable.img);
  }
 }).start();
 }

运行结果:异常

    android.view.ViewRootImpl$CalledFromWrongThreadException: Only the original thread that created a view hierarchy can touch its views.
        at android.view.ViewRootImpl.checkThread(ViewRootImpl.java:6357)
        at android.view.ViewRootImpl.requestLayout(ViewRootImpl.java:874)
        at android.view.View.requestLayout(View.java:17476)
        at android.view.View.requestLayout(View.java:17476)
        at android.view.View.requestLayout(View.java:17476)
        at android.view.View.requestLayout(View.java:17476)
        at android.view.View.requestLayout(View.java:17476)
        at android.view.View.requestLayout(View.java:17476)
        at android.widget.RelativeLayout.requestLayout(RelativeLayout.java:360)
        at android.view.View.requestLayout(View.java:17476)
        at android.widget.TextView.checkForRelayout(TextView.java:6871)
        at android.widget.TextView.setText(TextView.java:4057)
        at android.widget.TextView.setText(TextView.java:3915)
        at android.widget.TextView.setText(TextView.java:3890)
        at com.dong.demo.MainActivity$1.run(MainActivity.java:44)
        at java.lang.Thread.run(Thread.java:818)

不是说,子线程不能更新UI吗,为什么情形一可以正常运行,情形二不能正常运行呢;

子线程修改UI出现异常,与什么方法有关

首先从出现异常的log日志入手,发现出现异常的方法调用顺序如下:

TextView.setText(TextView.java:4057)

TextView.checkForRelayout(TextView.java:6871)

View.requestLayout(View.java:17476)

RelativeLayout.requestLayout(RelativeLayout.java:360)

View.requestLayout(View.java:17476)

ViewRootImpl.requestLayout(ViewRootImpl.java:874)

ViewRootImpl.checkThread(ViewRootImpl.java:6357)

更改ImageView时,出现的异常类似;

首先看TextView.setText()方法的源码

 private void setText(CharSequence text, BufferType type,
    boolean notifyBefore, int oldlen) {
 
 //省略其他代码

 if (mLayout != null) {
  checkForRelayout();
 }

 sendOnTextChanged(text, 0, oldlen, textLength);
 onTextChanged(text, 0, oldlen, textLength);

 //省略其他代码

然后,查看以下checkForRelayout()方法的与源码。

 private void checkForRelayout() {
 // If we have a fixed width, we can just swap in a new text layout
 // if the text height stays the same or if the view height is fixed.

 if ((mLayoutParams.width != LayoutParams.WRAP_CONTENT

  //省略代码

  // We lose: the height has changed and we have a dynamic height.
  // Request a new view layout using our new text layout.
  requestLayout();
  invalidate();
 } else {
  // Dynamic width, so we have no choice but to request a new
  // view layout with a new text layout.
  nullLayouts();
  requestLayout();
  invalidate();
 }
 }

checkForReLayout方法,首先会调用需要改变的View的requestLayout方法,然后执行invalidate()重绘操作;

TextView没有重写requestLayout方法,requestLayout方法由View实现;

查看RequestLayout方法的源码:

 public void requestLayout() {
 //省略其他代码
 if (mParent != null && !mParent.isLayoutRequested()) {
  mParent.requestLayout();
 }
 if (mAttachInfo != null && mAttachInfo.mViewRequestingLayout == this) {
  mAttachInfo.mViewRequestingLayout = null;
 }
 }

View获取到父View(类型是ViewParent,ViewPaerent是个接口,requestLayout由子类来具体实现),mParent,然后调用父View的requestLayout方法,比如示例中的父View就是xml文件的根布局就是RelativeLayout。

 @Override
 public void requestLayout() {
 super.requestLayout();
 mDirtyHierarchy = true;
 }

继续跟踪super.requestLayout()方法,即ViewGroup没有重新,即调用的是View的requestLayout方法。

经过一系列的调用ViewParent的requestLayout方法,最终调用到ViewRootImp的requestLayout方法。ViewRootImp实现了ViewParent接口,继续查看ViewRootImp的requestLayout方法源码。

 @Override
 public void requestLayout() {
  if (!mHandlingLayoutInLayoutRequest) {
   checkThread();
   mLayoutRequested = true;
   scheduleTraversals();
  }
 }

ViewRootImp的requestLayout方法中有两个方法:

一、checkThread,检查线程,源码如下

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

判断当前线程,是否是创建ViewRootImp的线程,而创建ViewRootImp的线程就是主线程,当前线程不是主线程的时候,就抛出异常。

二、scheduleTraversals(),查看源码:

 void scheduleTraversals() {
  if (!mTraversalScheduled) {
   mTraversalScheduled = true;
   mTraversalBarrier = mHandler.getLooper().getQueue().postSyncBarrier();
   mChoreographer.postCallback(
     Choreographer.CALLBACK_TRAVERSAL, mTraversalRunnable, null);
   if (!mUnbufferedInputDispatch) {
    scheduleConsumeBatchedInput();
   }
   notifyRendererOfFramePending();
   pokeDrawLockIfNeeded();
  }
 }

查看mTraversalRunnable中run()方法的具体操作

 final class TraversalRunnable implements Runnable {
  @Override
  public void run() {
   doTraversal();
  }
 }

继续追踪doTraversal()方法

 void doTraversal() {
  if (mTraversalScheduled) {
   mTraversalScheduled = false;
   mHandler.getLooper().getQueue().removeSyncBarrier(mTraversalBarrier);

   if (mProfile) {
    Debug.startMethodTracing("ViewAncestor");
   }

   performTraversals();

   if (mProfile) {
    Debug.stopMethodTracing();
    mProfile = false;
   }
  }
 }

查看到performTraversals()方法,熟悉了吧,这是View绘制的起点。

总结一下:

1.Android更新UI会调用View的requestLayout()方法,在requestLayout方法中,获取ViewParent,然后调用ViewParent的requestLayout()方法,一直调用下去,直到调用到ViewRootImp的requestLayout方法;

2.ViewRootImp的requetLayout方法,主要有两部操作一个是checkThread()方法,检测线程,一个是scheduleTraversals,执行绘制相关工作;

情形3

 @Override
 protected void onCreate(Bundle savedInstanceState) {
  Log.i("Dong", "Activity: onCreate");
  super.onCreate(savedInstanceState);
  setContentView(R.layout.activity_main);

  new Thread(new Runnable() {
   @Override
   public void run() {

    Looper.prepare();

    try {
     Thread.sleep(5000);
    } catch (InterruptedException e) {
     e.printStackTrace();
    }

    Toast.makeText(MainActivity.this, "显示Toast", Toast.LENGTH_LONG).show();

    Looper.loop();
   }
  }).start();
 }

运行结果:正常

分析

下面从Toast源码进行分析:

 public static Toast makeText(Context context, CharSequence text, @Duration int duration) {
  return makeText(context, null, text, duration);
 }

makeText方法调用了他的重载方法,继续追踪

 public static Toast makeText(@NonNull Context context, @Nullable Looper looper,
   @NonNull CharSequence text, @Duration int duration) {
  Toast result = new Toast(context, looper);

  LayoutInflater inflate = (LayoutInflater)
    context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
  View v = inflate.inflate(com.android.internal.R.layout.transient_notification, null);
  TextView tv = (TextView)v.findViewById(com.android.internal.R.id.message);
  tv.setText(text);

  result.mNextView = v;
  result.mDuration = duration;

  return result;
 }

新建了一个Toast对象,然后对显示的布局、内容、时长进行了设置,并返回Toast对象。

继续查看new Toast()的源码

 public Toast(@NonNull Context context, @Nullable Looper looper) {
  mCOntext= context;
  mTN = new TN(context.getPackageName(), looper);
  mTN.mY = context.getResources().getDimensionPixelSize(
    com.android.internal.R.dimen.toast_y_offset);
  mTN.mGravity = context.getResources().getInteger(
    com.android.internal.R.integer.config_toastDefaultGravity);
 }

继续查看核心代码 mTN = new TN(context.getPackageName(), looper);

TN初始化的源码为:

  TN(String packageName, @Nullable Looper looper) {
   //省略部分不相关代码
   if (looper == null) {
    // 没有传入Looper对象的话,使用当前线程对应的Looper对象
    looper = Looper.myLooper();
    if (looper == null) {
     throw new RuntimeException(
       "Can't toast on a thread that has not called Looper.prepare()");
    }
   }
   //初始化了Handler对象
   mHandler = new Handler(looper, null) {
    @Override
    public void handleMessage(Message msg) {
     switch (msg.what) {
      case SHOW: {
       IBinder token = (IBinder) msg.obj;
       handleShow(token);
       break;
      }
      case HIDE: {
       handleHide();
       // Don't do this in handleHide() because it is also invoked by
       // handleShow()
       mNextView = null;
       break;
      }
      case CANCEL: {
       handleHide();
       // Don't do this in handleHide() because it is also invoked by
       // handleShow()
       mNextView = null;
       try {
        getService().cancelToast(mPackageName, TN.this);
       } catch (RemoteException e) {
       }
       break;
      }
     }
    }
   };
  }

继续追踪handleShow(token)方法:

  public void handleShow(IBinder windowToken) {
   //省略部分代码
   if (mView != mNextView) {
    // remove the old view if necessary
    handleHide();
    mView = mNextView;
    Context cOntext= mView.getContext().getApplicationContext();
    String packageName = mView.getContext().getOpPackageName();
    if (cOntext== null) {
     cOntext= mView.getContext();
    }
    mWM = (WindowManager)context.getSystemService(Context.WINDOW_SERVICE);
    /*
    ·*省略设置显示属性的代码
    ·*/
    if (mView.getParent() != null) {
     if (localLOGV) Log.v(TAG, "REMOVE! " + mView + " in " + this);
     mWM.removeView(mView);
    }
=    try {
     mWM.addView(mView, mParams);
     trySendAccessibilityEvent();
    } catch (WindowManager.BadTokenException e) {
     /* ignore */
    }
   }
  }

通过源码可以看出,Toast显示内容是通过mWM(WindowManager类型)的直接添加的,更正:mWm.addView 时,对应的ViewRootImp初始化发生在子线程,checkThread方法中的mThread != Thread.currentThread()判断为true,所以不会抛出只能在主线程更新UI的异常。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对的支持。


推荐阅读
  • 本文介绍如何使用 Android 的 Canvas 和 View 组件创建一个简单的绘图板应用程序,支持触摸绘画和保存图片功能。 ... [详细]
  • 本文探讨了Java编程的核心要素,特别是其面向对象的特性,并详细介绍了Java虚拟机、类装载器体系结构、Java类文件和Java API等关键技术。这些技术使得Java成为一种功能强大且易于使用的编程语言。 ... [详细]
  • 在使用STM32Cube进行定时器配置时,有时会遇到延时不准的问题。本文探讨了可能导致延时不准确的原因,并提供了解决方法和预防措施。 ... [详细]
  • 阿里云ecs怎么配置php环境,阿里云ecs配置选择 ... [详细]
  • Python处理Word文档的高效技巧
    本文详细介绍了如何使用Python处理Word文档,涵盖从基础操作到高级功能的各种技巧。我们将探讨如何生成文档、定义样式、提取表格数据以及处理超链接和图片等内容。 ... [详细]
  • 本文介绍了如何利用 Spring Boot 和 Groovy 构建一个灵活且可扩展的动态计算引擎,以满足钱包应用中类似余额宝功能的推广需求。我们将探讨不同的设计方案,并最终选择最适合的技术栈来实现这一目标。 ... [详细]
  • 采用IKE方式建立IPsec安全隧道
    一、【组网和实验环境】按如上的接口ip先作配置,再作ipsec的相关配置,配置文本见文章最后本文实验采用的交换机是H3C模拟器,下载地址如 ... [详细]
  • 在进行QT交叉编译时,可能会遇到与目标架构不匹配的宏定义问题。例如,当为ARM或MIPS架构编译时,需要确保使用正确的宏(如QT_ARCH_ARM或QT_ARCH_MIPS),而不是默认的QT_ARCH_I386。本文将详细介绍如何正确配置编译环境以避免此类错误。 ... [详细]
  • 基于机器学习的人脸识别系统实现
    本文介绍了一种使用机器学习技术构建人脸识别系统的实践案例。通过结合Python编程语言和深度学习框架,详细展示了从数据预处理到模型训练的完整流程,并提供了代码示例。 ... [详细]
  • 本文详细介绍了如何使用 PHP 接收并处理微信支付的回调结果,确保支付通知能够被正确接收和响应。 ... [详细]
  • 我有一个SpringRestController,它处理API调用的版本1。继承在SpringRestControllerpackagerest.v1;RestCon ... [详细]
  • 简化报表生成:EasyReport工具的全面解析
    本文详细介绍了EasyReport,一个易于使用的开源Web报表工具。该工具支持Hadoop、HBase及多种关系型数据库,能够将SQL查询结果转换为HTML表格,并提供Excel导出、图表显示和表头冻结等功能。 ... [详细]
  • 理解UML的重要性及其应用
    探讨为什么大多数开发人员难以成为架构师,介绍从现实世界到业务模型的抽象过程,并详细解释UML在软件设计中的关键作用。 ... [详细]
  • ABBYY FineReader:高效PDF转换、精准OCR识别与文档对比工具
    在处理PDF转换和OCR识别时,您是否遇到过格式混乱、识别率低或图表无法正常识别的问题?ABBYY FineReader以其强大的功能和高精度的识别技术,完美解决这些问题,帮助您轻松找到最终版文档。 ... [详细]
  • 深入解析SpringMVC核心组件:DispatcherServlet的工作原理
    本文详细探讨了SpringMVC的核心组件——DispatcherServlet的运作机制,旨在帮助有一定Java和Spring基础的开发人员理解HTTP请求是如何被映射到Controller并执行的。文章将解答以下问题:1. HTTP请求如何映射到Controller;2. Controller是如何被执行的。 ... [详细]
author-avatar
手机用户2702932415_836
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有