真实项目中 ThreadLocal 的妙用

  • 时间:
  • 浏览:1
  • 来源:幸运快3_快3注册邀请码_幸运快3注册邀请码

一、哪几个是 ThreadLocal

ThreadLocal 提供了程序运行运行的局部变量,每个程序运行运行都都都可不能否 通过 set() 和 get() 来对之类 局部变量进行操作,但不不和有些程序运行运行的局部变量冲突,实现了程序运行运行间的据隔离。

简单讲:有1个获取用户的请求程序运行运行 A,有些向 ThreadLocal 填充变量 AValue(都都可不能否 了被程序运行运行 A 操作),该变量对有些获取用户的请求程序运行运行 B、C...是隔离的.

最简单的使用土办法 :

之类 一次 HTTP 请求程序运行运行中,利用 ThreadLocal 存储 Cookie 对象,进行情况汇报管理。set Cookie:

private ThreadLocal httpThreadLocal = new ThreadLocal();

httpThreadLocal.set(“Cookie: sid=13420771402233”)

后边存储格式是 String ,实际场景存储的是具体的对象。在这次 HTTP 请求过程中,任何过后都都都可不能否 获取 Cookie 。获取土办法 很简单 get Cookie:

String cookieValue = (String) httpThreadLocal.get();

Thread 与 ThreadLocal 对象引用关系图

二、你熟悉的场景

2.1 数据库连接池

比如一次请求程序运行运行进来,业务 Dao 时需更新 user 表和 user-detail 表。有些是 new 出有1个数据库 Connection ,分别不同的 Connection 操作 user 表和 user-detail 表,就无法保证事务。没人数据库连接池是怎样保证的?

答案是:利用 ThreadLocal 存储唯一 Connection 对象。每次请求程序运行运行,pool.getConnection 获取连接的过过后会没人 操作:

  • 会从 ThreadLocal 获取 Connection 对象。有些有,则保证了后边多个数据库操作共用同有1个 Connection ,从而保证了事务。
  • 有些没人,往 ThreadLocal 新增Connection 对象,并返回到程序运行运行
错误的做法
public class XXXService {

    private Connection conn;
}

有些 conn 是程序运行运行不安全的。没人 会因为多个请求公用有1个连接。请求量很大的情况汇报下,延迟各种。你懂。

有些,使用 ThreadLocal 保证每个请求程序运行运行的 Connection 是唯一的。即每个程序运行运行有买车人的连接。

继续讲到 Spring 框架,在事务现在始于时,会给当前程序运行运行有1个Jdbc Connection,在整个事务过程,有的是使用该程序运行运行绑定的connection来执行数据库操作,实现了事务的隔离性。Spring框架后边只是 我用的ThreadLocal来实现之类 隔离

比如你访问百度、我访问百度,会有不同 Cookie 。有些你都都可不能否 了访问我的 Cookie,我只是 我能。顾名思义,使用 ThreadLocal 保证每个 HTTP 请求程序运行运行的 Cookie 是唯一的。

Cookie 没人 都都可不能否 做 Session 等情况汇报管理。

三、实战场景

总结一下只是 我:ThreadLocal 都都可不能否 让同有1个程序运行运行中上下文之间数据共享

在后边章节 二、你熟悉的场景 确实介绍了只是 我现有场景。没人我这边具体的实战场景是哪几个?

简单的例子:

适用满足之类 有1个条件的场景:1.每个程序运行运行独有的有些信息,2.哪几个信息又会在多个土办法 或类中用到。

  1. 有1个请求程序运行运行,后边有有1个异步小程序运行运行,各有有1个土办法 。分别防止 A 或 B 业务
  2. 三种土办法 是传递不可变的入参
  3. 另三种只是 我 ThreadLocal,插进 ThreadLocal 的入参,会被各个土办法 共享。有些多个请求程序运行运行互不影响
比较复杂的例子:

一次发货操作:会根据入参,进行组件化、流程编排话。没人入参会被各个地方用到,有些有些流程组件是异步的(之类 new thread 操作的)。这过后都都可不能否 定有1个 XXContext 上下文:

public class XXContext {
    
    private static ThreadLocal<Map<Class<?>, Object>> context = new InheritableThreadLocal<>();
    
    /**
     * 把参数设置到上下文的Map中
     */
    public static void put(Object obj) {
        Map<Class<?>, Object> map = context.get();
        if (map == null) {
            map = new HashMap<>();
            context.set(map);
        }
        if (obj instanceof Enum) {
            map.put(obj.getClass().getSuperclass(), obj);
        } else {
            map.put(obj.getClass(), obj);
        }
    }
    
    /**
     * 从上下文中,根据类名取出参数
     */
    @SuppressWarnings("unchecked")
    public static <T> T get(Class<T> c) {
        Map<Class<?>, Object> map = context.get();
        if (map == null) {
            return null;
        }
        return (T) map.get(c);
    }
    
    /**
     * 清空ThreadLocal的数据
     */
    public static void clean() {
        context.remove();
    }
}

代码解析:

  • 有的是 static 操作,之类 DateUtil 玩法
  • 记得每次请求程序运行运行后清理。都都可不能否 AOP 去清理,加个注解就行。有些同有1个请求程序运行运行有些被业务方公用。

(完)