当前位置: 首页 > 编程笔记 >

Android 中ThreadLocal的深入理解

胡景澄
2023-03-14
本文向大家介绍Android 中ThreadLocal的深入理解,包括了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总结下的,但总是忘记弄,后续得多鞭策下自己,写点东西相当于自己做个笔记,把知识框架化,不对的地方请大神们多多指教!

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

 类似资料:
  • 安卓(Android)是一种基于 Linux 内核(不包含 GNU 组件)的自由及开放源代码的操作系统。主要使用于移动设备,如智能手机和平板电脑,由美国 Google 公司和开放手机联盟领导及开发。

  • 本文向大家介绍深入理解Android 5.0中的Toolbar,包括了深入理解Android 5.0中的Toolbar的使用技巧和注意事项,需要的朋友参考一下 环境说明: Android Studio 2.0 V7包版本:com.android.support:appcompat-v7:23.4.0 compileSdkVersion 23 buildToolsVersion "24.0.0" T

  • 本文向大家介绍深入理解Android中的建造者模式,包括了深入理解Android中的建造者模式的使用技巧和注意事项,需要的朋友参考一下 前言 在Android开发过程中,我发现很多安卓源代码里应用了设计模式,比较常用的有适配器模式(各种adapter),建造者模式(Alert Dialog的构建)等等。虽然我们对大多数设计模式都有所了解,但是在应用设计模式的这个方面,感觉很多人在这方面有所不足。所

  • 本文向大家介绍深入理解vue中的$set,包括了深入理解vue中的$set的使用技巧和注意事项,需要的朋友参考一下 在我们使用vue进行开发的过程中,可能会遇到一种情况:当生成vue实例后,当再次给数据赋值时,有时候并不会自动更新到视图上去; 当我们去看vue文档的时候,会发现有这么一句话:如果在实例创建之后添加新的属性到实例上,它不会触发视图更新。如下代码: 运行结果: 为什么会这样呢?当去查对

  • 本文向大家介绍深入理解Android手势识别,包括了深入理解Android手势识别的使用技巧和注意事项,需要的朋友参考一下 对于触摸屏,其原生的消息无非按下、抬起、移动这几种,我们只需要简单重载onTouch或者设置触摸侦听器setOnTouchListener即可进行处理。不过,为了提高我们的APP的用户体验,有时候我们需要识别用户的手势,Android给我们提供的手势识别工具GestureDe

  • 本文向大家介绍深入理解Android中View绘制的三大流程,包括了深入理解Android中View绘制的三大流程的使用技巧和注意事项,需要的朋友参考一下 前言 最近对Android中View的绘制机制有了一些新的认识,所以想记录下来并分享给大家。View的工作流程主要是指measure、layout、draw这三大流程,即测量、布局和绘制,其中measure确定View的测量宽高,layout根

  • 本文向大家介绍android中Context深入详解,包括了android中Context深入详解的使用技巧和注意事项,需要的朋友参考一下 以下分别通过Context认知角度,继承关系,对象创建等方面android中Context做了深入的解释,一起学习下。 1、Context认知。 Context译为场景,一个应用程序可以认为是一个工作环境,在这个工作环境中可以存在许多场景,coding代码的场

  • 本文向大家介绍深入理解Java中的接口,包括了深入理解Java中的接口的使用技巧和注意事项,需要的朋友参考一下 一. 为什么要使用接口 假如有一个需求:要求实现防盗门的功能。门有"开"和"关"的功能,锁有"上锁"和"开锁"的功能。 分析:首先防盗门是一个门,门有开门和关门的功能,还有一把锁,锁有开锁和上锁,按照面向对象的编程的思想,我们会将门和锁都作为一个类而单独存在,但是,不能让防盗门继承自门的