Android 中ThreadLocal的深入理解

ThreadLocal

前言:

ThreadLocal很容易让人望文生义,想当然地认为是一个“本地线程”。其实,ThreadLocal并不是一个Thread,ThreadLocal是一个线程内部的数据存储类,通过它可以在指定的线程中存储数据,数据存储以后,只有在指定线程中可以获取到存储的数据,对于其它线程来说无法获取到数据。设计初衷就是:提供线程内部的局部变量,在本线程内随时可取,而隔离了其他线程。

private static void prepare(boolean quitAllowed) {
  if (sThreadLocal.get() != null) {
    throw new RuntimeException("Only one Looper may be created per thread");
  }
  sThreadLocal.set(new Looper(quitAllowed));
}

这段代码就是在初始化Looper的时候会执行到的方法,这里也可以看出,一个looper只能对应一个thread。

public void set(T value) {
  Thread currentThread = Thread.currentThread();
  Values values = values(currentThread);
  if (values == null) {
    values = initializeValues(currentThread);
  }
  values.put(this, value);
}

looper创建时调用了ThreadLocal类中的set方法,这里,首先获取到当前的线程,然后,将线程通过values的方法得到当前线程的Values,而Values类是ThreadLocal中的一个嵌套类,用来存储不同thread的信息。

/**
 * Gets Values instance for this thread and variable type.
 */
Values values(Thread current) {
  return current.localValues;
}

在Thread类中有这么一段:

/**
  * Normal thread local values.
  */
  ThreadLocal.Values localValues;

所以从上面我们了解到set方法把当前thread中的localValues获取到,然后用得到的values将当前的this和传进来的Looper进行put操作:

/**
 * Sets entry for given ThreadLocal to given value, creating an
 * entry if necessary.
 */
void put(ThreadLocal<?> key, Object value) {
  cleanUp(); 

  // Keep track of first tombstone. That's where we want to go back
  // and add an entry if necessary.
  int firstTombstone = -1; 

  for (int index = key.hash & mask;; index = next(index)) {
    Object k = table[index]; 

    if (k == key.reference) {
      // Replace existing entry.
      table[index + 1] = value;
      return;
    } 

    if (k == null) {
      if (firstTombstone == -1) {
        // Fill in null slot.
        table[index] = key.reference;
        table[index + 1] = value;
        size++;
        return;
      } 

      // Go back and replace first tombstone.
      table[firstTombstone] = key.reference;
      table[firstTombstone + 1] = value;
      tombstones--;
      size++;
      return;
    } 

    // Remember first tombstone.
    if (firstTombstone == -1 && k == TOMBSTONE) {
      firstTombstone = index;
    }
  }
}

这段代码的意思就是将传进来的looper对象保存在了Values类中的table成员变量中,保存的下标是在[index+1]里,table是一个Object[]的数组。最后看看对应的get方法:

public T get() {
    // Optimized for the fast path.
    Thread currentThread = Thread.currentThread();
    Values values = values(currentThread);
    if (values != null) {
      Object[] table = values.table;
      int index = hash & values.mask;
      if (this.reference == table[index]) {
        return (T) table[index + 1];
      }
    } else {
      values = initializeValues(currentThread);
    } 

    return (T) values.getAfterMiss(this);
  }

首先获取到当前线程,然后去取当前线程的Values值,如果值不空,先拿table数组,再得到此values的下标,最后返回此下标对应的table[]值。所以ThreadLocal我自己的理解是:不同的线程拥有不同的Values值,这个值统一在ThreadLocal类的table数组中,也就是说每个线程有自己的副本,在自己的副本里面读写信息互补干扰!

时间过得好快,转眼一年了。整整快了一年没怎么写东西,说多了都是借口,没有及时整理和沉淀,今年间是有点想法把自己平日写的小demo总结下的,但总是忘记弄,后续得多鞭策下自己,写点东西相当于自己做个笔记,把知识框架化,不对的地方请大神们多多指教!

如有疑问请留言或者到本站社区交流讨论,感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!

(0)

相关推荐

  • Android 中 ThreadLocal使用示例

    Android 中 ThreadLocal使用示例 概要: Demo描述:  ThreadLocal使用示例. 关于ThreadLocal的官方文档描述 Implements a thread-local storage, that is, a variable for which each thread has its own value. All threads share the same ThreadLocal object, but each sees a different valu

  • Android 详解ThreadLocal及InheritableThreadLocal

     Android  详解ThreadLocal及InheritableThreadLocal 概要: 因为在android中经常用到handler来处理异步任务,通常用于接收消息,来操作UIThread,其中提到涉及到的looper对象就是保存在Threadlocal中的,因此研究下Threadlocal的源码. 分析都是基于android sdk 23 源码进行的,ThreadLocal在android和jdk中的实现可能并不一致. 在最初使用Threadlocal的时候,很容易会产生的误解就

  • Android 中ThreadLocal的深入理解

    ThreadLocal 前言: ThreadLocal很容易让人望文生义,想当然地认为是一个"本地线程".其实,ThreadLocal并不是一个Thread,ThreadLocal是一个线程内部的数据存储类,通过它可以在指定的线程中存储数据,数据存储以后,只有在指定线程中可以获取到存储的数据,对于其它线程来说无法获取到数据.设计初衷就是:提供线程内部的局部变量,在本线程内随时可取,而隔离了其他线程. private static void prepare(boolean quitAll

  • Java中ThreadLocal的一些理解

    前言 面试的时候被问到ThreadLocal的相关知识,没有回答好(奶奶的,现在感觉问啥都能被问倒),所以我决定先解决这几次面试中都遇到的高频问题,把这几个硬骨头都能理解的透彻的说出来了,感觉最起码不能总是一轮游. ThreadLocal介绍 ThreadLocal是JDK1.2开始就提供的一个用来存储线程本地变量的类.ThreadLocal中的变量是在每个线程中独立存在的,当多个线程访问ThreadLocal中的变量的时候,其实都是访问的自己当前线程的内存中的变量,从而保证的变量的线程安全.

  • 深入理解Android中的建造者模式

    前言 在Android开发过程中,我发现很多安卓源代码里应用了设计模式,比较常用的有适配器模式(各种adapter),建造者模式(Alert Dialog的构建)等等.虽然我们对大多数设计模式都有所了解,但是在应用设计模式的这个方面,感觉很多人在这方面有所不足.所以这篇文章我们一起深入的理解Android中的建造者模式. 建造者模式(Builder Pattern)也叫生成器模式,其定义如下: separate the construction of a complex object from

  • 深入理解Android中Scroller的滚动原理

    View的平滑滚动效果 什么是实现View的平滑滚动效果呢,举个简单的例子,一个View从在我们指定的时间内从一个位置滚动到另外一个位置,我们利用Scroller类可以实现匀速滚动,可以先加速后减速,可以先减速后加速等等效果,而不是瞬间的移动的效果,所以Scroller可以帮我们实现很多滑动的效果. 首先我们先来看一下Scroller的用法,基本可概括为"三部曲": 1.创建一个Scroller对象,一般在View的构造器中创建: public ScrollViewGroup(Cont

  • 深入理解Android中View绘制的三大流程

    前言 最近对Android中View的绘制机制有了一些新的认识,所以想记录下来并分享给大家.View的工作流程主要是指measure.layout.draw这三大流程,即测量.布局和绘制,其中measure确定View的测量宽高,layout根据测量的宽高确定View在其父View中的四个顶点的位置,而draw则将View绘制到屏幕上,这样通过ViewGroup的递归遍历,一个View树就展现在屏幕上了. 说的简单,下面带大家一步一步从源码中分析: Android的View是树形结构的: 基本概

  • 理解Android中Activity的方法回调

    为什么需要方法回调? 方法回调是功能定义和功能分离的一种手段,是一种松耦合的设计思想.在JAVA中回调是通过接口来实现的.作为一种系统架构,必须要有自己的运行环境,并且要提供用户的实现接口. 下面通过实例来模拟一下Android中Activity的方法回调思想. Activity接口 复制代码 代码如下: package com.xujing.test  //定义接口  public interface Activity{      //创建时调用的方法      public void onC

  • 理解Android中的自定义属性

    本文实例讲解了Android中的自定义属性,具体内容如下 1.引言 对于自定义属性,大家肯定都不陌生,遵循以下几步,就可以实现: 自定义一个CustomView(extends View )类 编写values/attrs.xml,在其中编写styleable和item等标签元素 在布局文件中CustomView使用自定义的属性(注意namespace) 在CustomView的构造方法中通过TypedArray获取 ps:如果你对上述几个步骤不熟悉,建议先熟悉下,再继续~ 那么,我有几个问题:

  • 深入理解Android中的Window和WindowManager

    Window表示一个窗口的概念,Window是一个抽象类,它的具体实现是PhoneWindow.创建一个Window,需要通过WindowManager即可完成,WindowManager是外界访问Window的入口,Window具体实现位于WindowManagerService中,WindowManager和WindowManagerService的交互是一个IPC的过程.Android中,所有的视图都是通过Window来呈现,不管是Activity.Dialog.还是Toast,它们的视图

  • Android中BaseAdapter的用法分析与理解

    本文实例分析了Android中BaseAdapter的用法.分享给大家供大家参考,具体如下: 最近做一个项目,项目中用到了ListView,ListView最重要的就是绑定数据,这个数据由Adapter来提供,这里我重写了BaseAdapter这个类来实现自己的menuAdapter代码如下: package org.leepood.lanorder; import java.io.InputStream; import java.util.ArrayList; import java.util

随机推荐