关于ThreadLocal的一些认识

本文探讨了ThreadLocal在Android中的应用,通过示例展示了如何使用ThreadLocal保持线程数据的私有性,避免线程间的数据干扰。文章还深入解析了ThreadLocal的内部实现,指出每个线程都有一个Values对象,用于存储ThreadLocal变量和对应的值,其数据结构基于一个Object数组。ThreadLocal的set和get方法确保了在线程局部操作的安全性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

在前面讲关于Handler的Looper的时候,我们讲到过Looper的获取和设置,是通过ThreadLocal来操作的,如下:

static final ThreadLocal<Looper> sThreadLocal = new ThreadLocal<Looper>();

在准备Looper的时候,

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

在获取Looper的时候,

   /**
     * Return the Looper object associated with the current thread.  Returns
     * null if the calling thread is not associated with a Looper.
     */
    public static Looper myLooper() {
        return sThreadLocal.get();
    }

那么ThreadLocal的作用到底什么呢?为什么会有ThreadLocal的出现呢?

首先,我们先来看下面一个例子吧,如下:

public class HelperTesting {

    private String text = "empty";

    private static HelperTesting helperTesting = new HelperTesting();

    private void printText() {
        System.out.println(text);
    }

    private void changeText(String str) {
        text = str;
    }
    
    private void printWithLine(String str){
        System.out.println("##################### " + str +" #############################");
    }

    public static void main(String[] args) {

        helperTesting.printWithLine(Thread.currentThread().getName() + " start");
        helperTesting.printText();
        
        Thread thread = new Thread(new Runnable() {
            @Override
            public void run() {                      
                helperTesting.printWithLine(Thread.currentThread().getName() + " start");
                helperTesting.printText();
                helperTesting.changeText("thread1");
                helperTesting.printText();
                helperTesting.printWithLine(Thread.currentThread().getName() + " end");
            }
        });
        
        thread.start();
        try {
            thread.join();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        
        helperTesting.printText();  
        helperTesting.printWithLine(Thread.currentThread().getName() + " end");
    }
}


在这个小Demo里面,我们做了几件事情:

1)创建了一个HelperString对象,其有一个 text 字段,初始化其值为 “empty”。

2)创建了一个新的线程,在线程中改变helperString对象的 text 的值为"thread1",运行线程,并打印其值。

3)最后我们在原来的线程中打印出text的值。

运行结果如下:

##################### main start #############################
empty
##################### Thread-0 start #############################
empty
thread1
##################### Thread-0 end #############################
thread1
##################### main end #############################

我们可以看到,在主线程中的 text 值被改变了。

在多线程的编程过程当中,因为thread共享同一段进程空间,所以A线程的运行是完全有可能在不经意间就改变了某个值,而当B线程也去拿这个值的时候,可能就已经不是其想要获取的数值了。

于是有人就想,能不能让线程也拥有其私有的一些空间呢,让存放在这段空间里面的值并不会被其他线程改变,而这就是ThreadLocal的出现。

注意,其出现只是为了保证线程数据的私有性,并不是为了解决线程间的同步问题,我估计这也是其为什么叫Thread Local 的原因。

接下来,我们再通过一个Demo来看看,怎么使用ThreadLocal,如下:

public class HelperTestingSec {

    private ThreadLocal<String> sThreadLocal = new ThreadLocal<String>();

    private static HelperTestingSec helperTesting = new HelperTestingSec();

    private HelperTestingSec(){
        sThreadLocal.set("empty");
    }
    
    private void printText() {
        System.out.println(sThreadLocal.get());
    }

    private void changeText(String str) {
       sThreadLocal.set(str);
    }
    
    private void printWithLine(String str){
        System.out.println("##################### " + str +" #############################");
    }        
    public static void main(String[] args) {

        helperTesting.printWithLine(Thread.currentThread().getName() + " start");
        helperTesting.printText();      
        
        Thread thread = new Thread(new Runnable() {
            @Override
            public void run() {
                helperTesting.printWithLine(Thread.currentThread().getName() + " start");
                helperTesting.printText();
                helperTesting.changeText("thread1");
                helperTesting.printText();
                helperTesting.printWithLine(Thread.currentThread().getName() + " end");
            }
        });
                        
        thread.start();
        try {
            thread.join();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        
        helperTesting.printText();
        helperTesting.printWithLine(Thread.currentThread().getName() + " end");
    }
}

这个Demo中做的事情,其实跟上一个Demo类似,唯一不同的是,在这个Demo中,我们利用一个ThreadLocal对象来存放这个String的值,运行结果如下:

##################### main start #############################
empty
##################### Thread-0 start #############################
null
thread1
##################### Thread-0 end #############################
empty
##################### main end #############################

从上面的结果,我们可以看到,同一个helperTestingSec对象,在 main线程中,其 sThreadLocal中存放的对象一直是empty。

而在新建的线程当中呢,当在调用changeText进行设值之前,其value是null的,也就是说在新线程中,虽然是同一个对象,但是其值是不一样的,即使我们在新线程中改变了其值,也不会影响其在Main线程中的值。

可见,通过 ThreadLocal,我们就做到了线程间的数据独立了。

那么,ThreadLocal的实现原理是什么,为什么其能做到这个事情呢,我们还是想了解一下其内部的构造的,看一下能不能学到点什么东西,对吧。

按照顺序,一般情况下,我们总是先放东西进去,再去拿东西出来的,所以我们先看一下ThreadLocal 的 set 方法。

有一点要注意的是,这里讲解的ThreadLocal的实现是Android里面的实现,JDK中的实现方式有点不一样,但是原理是一样的。

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

在Demo2中,我们调用sThreadLocal的 set方法的时候,我们就进入了上面的逻辑。

可以看到,第一步,就是先利用Thread.currentThread拿到当前的线程,再根据这个线程去获取一个叫做Values的对象。再看一下values函数,我们可看到其是从哪里去获取这个Values对象的,如下:

   /**
     * Creates Values instance for this thread and variable type.
     */
    Values initializeValues(Thread current) {
        return current.localValues = new Values();
    }

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

可以看到,其是从current这个线程里去获取Values对象的,而如果不存在这个对象,其就会调用initializeValues方法,为线程的localValues创建一个新的Values对象。

最近再调用values的put方法,以当前ThreadLocal对象为key值,将我们的valule给存放到这个Values对象中,我们会马上意识到,Values实现的应该是一个类似Map的键值对的数据结构。

从这里,我们就可以意识到:

1)在每个线程中都存在一个Values对象。

2)无论我们在哪个线程中调用ThreadLocal对象,都会先去当前线程中取得一个Values对象(如果不存在,就创建一个)再对这个Values对象进行键值操作,比如ThreadLocal的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);
    }

3)一个ThreadLocal对象其实是可以给所有线程共享的,因为其只是作为一个入口和一个WeakRefreence的Key值,真正的数据都是存在Values中,而这个Values对象是线程独有,且在当前线程中是唯一的。

所以这也是为什么在Looper类中,sThreadLocal对象是一个静态的对象就行了。

而Values的内部实现,其实是用一个Object数组来做为数据结构的,偶数位(0,2,4....)等作为Key值,其存放是指向ThreadLocal对象本身的一个WeakReference, 奇数位(1,3,5....)等作为的Value值,存放的是我们设置的值,从上面 get方法中, 我们也可以略窥一二。

在这里,推荐两边文章,相信对大家了解ThreadLocal有一定的帮助。

When and how to use a ThreadLocal
How is ThreadLocal implemented?



使用nvm安装后,如果npmnode失效,可能有多种原因导致。一种可能性是在安装nvm的过程中出现问题,导致无法正确设置环境变量。另一种可能性是nvm安装后没有正确安装配置npm。 要解决这个问题,可以尝试以下方法: 1. 确认nvm安装成功:首先,检查nvm是否成功安装设置了正确的环境变量。可以在命令行中运行`nvm --version`来验证nvm安装情况。如果没有显示版本号,说明nvm没有正确安装,请重新安装nvm。 2. 检查环境变量:如果nvm已经安装,但npmnode仍然失效,可能是由于环境变量配置错误所致。请确保在你的环境变量中添加了正确的路径引用。在Windows操作系统上,你可以在系统环境变量中添加以下两个变量: - 变量名:`NVM_HOME`,变量值:nvm安装路径(例如:C:\Users\YourUsername\AppData\Roaming\nvm) - 变量名:`NVM_SYMLINK`,变量值:node.js安装路径(例如:C:\Program Files\nodejs) 在这些变量添加后,重新打开一个新的命令行窗口,然后尝试运行`node -v``npm -v`来检查是否恢复正常。 3. 配置国内镜像:在某些情况下,无法自动下载对应的npm包可能导致nodenpm指令失效。你可以尝试配置国内镜像来解决这个问题。根据你提供的引用,你需要在nvm安装目录下的setting.txt文件中添加以下两句配置: ``` node_mirror: https://npm.taobao.org/mirrors/node/ npm_mirror: https://npm.taobao.org/mirrors/npm/ ``` 4. 手动安装node版本:如果上述方法都没有解决问题,你可以尝试手动安装特定版本的node。首先,使用`nvm ls-remote`命令列出可用的node版本。然后,选择一个与你的nvm版本兼容的node版本,使用`nvm install <version>`命令手动安装该版本。 5. 卸载nvm:如果以上方法都无法解决问题,你可以考虑卸载nvm使用其他方式安装nodenpm。可以按照官方文档提供的方法卸载nvm通过其他途径安装nodenpm。 希望这些方法能够帮助你恢复npmnode的正常使用。如果问题仍然存在,请提供更多详细信息,以便我们能够进一步帮助你解决问题。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值