ThreadLocal

本文深入解析了ThreadLocal的工作原理及其在Java中的应用。介绍了如何利用ThreadLocal实现线程安全,探讨了其在多线程环境中的优势及潜在的风险,如内存泄漏等问题。

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

ThreadLocal的核心是:每个线程都可以通过ThreadLocal对象的 get 或 set 方法来访问属于自己的,独立初始化的变量拷贝。

ThreadLocal 介绍

  • 我们想要在多线程当中拥有某个类的多个独立的私有化实例对象而不造成冲突。每个对象唯一对应一个线程。这其中(指ThreadLocal)便是实现了线程安全。

  • 有一点需要注意的是,ThreadLocal变量是全局可访问的。我们可以从线程中的任何一个地方来访问它。同时注意,TheadLocal 变量是被声明为static 和final的。

什么是线程安全?

一个线程相当于一个单一的流水线执行过程。当我们谈及多线程应用时,指的是多个流水线执行着相同的代码(感觉这里说得不太对,但原文就是:we mean that there multiple(sequential flow of control) line of process that runs through the same lines of code.)。这种情况下,就有可能出现某个线程访问或修改另一个线程的数据。如果我们不允许这种情况出现,就得使它能够线程安全。以下是几种实现线程安全的操作:
- 重入(Re-entrancy, java 里有个重入锁类: ReentrantLock , 实现的是可被中断的同步)
- 同步执行 Mutual exclusion(synchronization)
- Thread-local
- 原子操作 (Atomic operation)

根据上面说的可,ThreadLocal也是实现线程安全的一种选择,现在我们看看ThreadLocal如何实现。

ThreadLocal 的用法

这里我不禁引用 Joshua Bloch 的话(对于这一节我还有什么更好的人选),以下是他的原话(这几句原话都不太理解,估计翻译不对,都附上原话。。):

  • 单纯每个线程的上下文, 例如 用户id 或者 交易id(transaction id)。都能很好工作。在线程退出的时候能够轻松地进行资源释放和清除工作。不会产生内存泄漏。( Genuine per-thread context, such as user id or transaction id. Works great. Easy to clean up when the thread exits the scope. No leaks.)

  • 每个线程的性能(Per-thread instances for performance.)

  • 当你不注意控制好回调的时候,脏数据将产生(“Sleazing” values through callbacks that you don’t control:):有时候,你必须调用一个回调你的函数的库函数(sometimes you must call a library method that calls back into your package)。这个时候,由于库函数的缺点,你无法自己将自己需要的上下文信息传递给自己(you need some context that you were unable to pass to yourself)。这种罕见的情况,thread locals将会成为你的救命恩人。

以上几个要点,用我自己的话来讲就是:我们拥有一个非线程安全的对象,当我们想要安全地使用它。我们可以 通过将该对象包裹于 synchronized 代码块中来实现同步。另一个方法就是使用ThreadLocal,它通过为每个线程持有分开的独立的对象拷贝来使其线程安全。

ThreadLocal实例

package com.javapapers;

import java.text.SimpleDateFormat;
import java.util.Date;

public class ThreadLocalExample {
    private static final ThreadLocal formatter = new ThreadLocal() {

        protected SimpleDateFormat initialValue() {
            return new SimpleDateFormat("yyyyMMdd HHmm");
        }
    };

    public String formatIt(Date date) {
        return formatter.get().format(date);
    }
}

在上面的例子中, get() 方法是理解的关键。它返回了当前线程对于这个thread-local变量的一份拷贝。如果该变量对于当前线程没有值,则会先进行初始化为 initialValue 方法的返回值。

java文档中的例子

下面的类为每个线程生成唯一的本地识别码(即id)。一个线程在它第一次调用ThreadId.get()的时候便获得了它的id,并且在后续调用过程中保持不变。

import java.util.concurrent.atomic.AtomicInteger;

 public class ThreadId {
     // Atomic integer containing the next thread ID to be assigned
     private static final AtomicInteger nextId = new AtomicInteger(0);

     // Thread local variable containing each thread's ID
     private static final ThreadLocal threadId =
         new ThreadLocal() {
             @Override protected Integer initialValue() {
                 return nextId.getAndIncrement();
         }
     };

     // Returns the current thread's unique ID, assigning it if necessary
     public static int get() {
         return threadId.get();
     }
 }
ThreadLocal在Java API中的使用

在JDK1.7中有了一个新的类叫ThreadLocalRandom。它可以用于为并行的线程产生随机数。随机数种子对于每个线程都是唯一的。这真是个很酷的工具。

以下是该类中对于ThreadLocal的使用:

    private static final ThreadLocal localRandom =
        new ThreadLocal() {
            protected ThreadLocalRandom initialValue() {
                return new ThreadLocalRandom();
            }
    };
ThreadLocalRandom 的使用例子
package com.javapapers;

import java.util.concurrent.ThreadLocalRandom;

public class ThreadLocalRandomExample {

    public static void main(String args[]) throws InterruptedException {

        //tossing 3 coins
        for (int i = 0; i < 3; i++) {
            final Thread thread = new Thread() {

                public void run() {
                    System.out.print(Thread.currentThread().getName() + ":");

                    // generating 3 random numbers - random for every thread
                    for (int j = 0; j < 3; j++) {
                        final int random = ThreadLocalRandom.current().nextInt(
                                1, 3);
                        System.out.print(random + ",");
                    }
                    System.out.println();
                }
            };
            thread.start();
            thread.join();
        }
    }
}

ThreadLocal与内存泄漏

ThreadLocal本身并不坏,它是个很有用的工具API。当着但这完全取决于我们如何使用它。我们应该学着在恰当的情况下选择合适的工具。我们不能用炮弹来攻击蚊子, 却责怪炮弹本身。

网上弥漫着对于ThreadLocal的强烈抱怨,说它会导致内存泄漏。大部分情况下,不是这样的!

我又想引用 Joshua Bloch 的话:
“…thead local本身并没有错,它们并没有导致内存泄漏,也并不慢。它们比非thread-local的副本更加本地化(local, 或者翻译为私有吧)(例如,它们有更好的信息隐藏属性)。他们有可能被误用,当然,大多数其他的编程工具也是如此…”

原文:http://javapapers.com/core-java/threadlocal/

资源下载链接为: https://pan.quark.cn/s/22ca96b7bd39 在当今的软件开发领域,自动化构建与发布是提升开发效率和项目质量的关键环节。Jenkins Pipeline作为一种强大的自动化工具,能够有效助力Java项目的快速构建、测试及部署。本文将详细介绍如何利用Jenkins Pipeline实现Java项目的自动化构建与发布。 Jenkins Pipeline简介 Jenkins Pipeline是运行在Jenkins上的一套工作流框架,它将原本分散在单个或多个节点上独立运行的任务串联起来,实现复杂流程的编排与可视化。它是Jenkins 2.X的核心特性之一,推动了Jenkins从持续集成(CI)向持续交付(CD)及DevOps的转变。 创建Pipeline项目 要使用Jenkins Pipeline自动化构建发布Java项目,首先需要创建Pipeline项目。具体步骤如下: 登录Jenkins,点击“新建项”,选择“Pipeline”。 输入项目名称和描述,点击“确定”。 在Pipeline脚本中定义项目字典、发版脚本和预发布脚本。 编写Pipeline脚本 Pipeline脚本是Jenkins Pipeline的核心,用于定义自动化构建和发布的流程。以下是一个简单的Pipeline脚本示例: 在上述脚本中,定义了四个阶段:Checkout、Build、Push package和Deploy/Rollback。每个阶段都可以根据实际需求进行配置和调整。 通过Jenkins Pipeline自动化构建发布Java项目,可以显著提升开发效率和项目质量。借助Pipeline,我们能够轻松实现自动化构建、测试和部署,从而提高项目的整体质量和可靠性。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值