ThreadLocal使用实例 超简单使用

ThreadLocal使用实例

ThreadLocal为解决多线程程序的并发问题提供了一种新的思路。使用这个工具类可以很简洁地编写出优美的多线程程序,

  • ThreadLocal并不是一个Thread,而是Thread的局部变量。
    ThreadLocal是解决线程安全问题一个很好的思路,
  • 它通过为每个线程提供一个独立的变量副本解决了变量并发访问的冲突问题。
  • 在很多情况下,ThreadLocal比直接使用synchronized同步机制解决线程安全问题更简单,更方便,且结果程序拥有更高的并发性。

ThreadLocal类接口很简单,只有4个方法,我们先来了解一下:

  • Object get():获取该线程局部变量的值。
  • void set(Object value):给该线程局部变量赋值。
  • protected Object initialValue():返回该线程局部变量的初始值,该方法是一个protected的方法,
    • 显然是为了让子类覆盖而设计的。
  • public void remove():将当前线程局部变量的值删除。

示例1

public class test2 {
    // ①通过匿名内部类覆盖ThreadLocal的initialValue()方法,指定初始值
    private static ThreadLocal<Object> seqNum = new ThreadLocal<Object>() {
        public Integer initialValue() {
            return 0;
        }
    };

    // ②获取下一个序列值
    public int getNextNum() {
        seqNum.set((Integer) seqNum.get() + 1);
        return (Integer) seqNum.get();
    }

    public static void main(String[] args) {
        test2 sn = new test2();
        // ③ 3个线程共享sn,各自产生序列号
        TestClient t1 = new TestClient(sn);
        TestClient t2 = new TestClient(sn);
        TestClient t3 = new TestClient(sn);
        new Thread(t1).start();
        new Thread(t2).start();
        new Thread(t3).start();
    }
}

class TestClient implements Runnable {
    private test2 sn;

    public TestClient(test2 sn) {
        super();
        this.sn = sn;
    }

    @Override
    public void run() {
        for (int i = 0; i < 3; i++) {
            // ④每个线程打出3个序列值
            System.out.println("thread[" + Thread.currentThread().getName() + "] sn[" + sn.getNextNum() + "]");
        }
    }
}
thread[Thread-0] sn[1]
thread[Thread-1] sn[1]
thread[Thread-2] sn[1]
thread[Thread-1] sn[2]
thread[Thread-2] sn[2]
thread[Thread-0] sn[2]
thread[Thread-2] sn[3]
thread[Thread-1] sn[3]
thread[Thread-0] sn[3]

分析:
考察输出的结果信息,我们发现每个线程所产生的序号虽然都

  • 共享同一个test2实例,
  • 但它们并没有发生相互干扰的情况,而是各自产生独立的序列号,
  • 这是因为我们通过ThreadLocal为每一个线程提供了单独的副本

示例2:

// 代码清单3 TopicDao:非线程安全
public class TopicDao {
    private Connection conn;// ①一个非线程安全的变量
    public void addTopic(){
        Statement stat = conn.createStatement();// ②引用非线程安全变量
    }
}

/**
由于①处的conn是成员变量,因为addTopic()方法是非线程安全的,必须在使用时创建一个新TopicDao实例(非singleton)。下面使用ThreadLocal对conn这个非线程安全的“状态”进行改造: 
**/

// 代码清单4 TopicDao:线程安全
import java.sql.Connection;
import java.sql.Statement;
public class TopicDao {
    // ①使用ThreadLocal保存Connection变量
    private static ThreadLocal<Connection> connThreadLocal = new ThreadLocal<Connection>();
    public static Connection getConnection(){
    // ②如果connThreadLocal没有本线程对应的Connection创建一个新的Connection,并将其保存到线程本地变量中。
        return connThreadLocal.get();
    }
    public void addTopic() {
        // ④从ThreadLocal中获取线程对应的Connection
        Statement stat = getConnection().createStatement();
    }
}
  • 如果connThreadLocal没有 本线程对应的Connection创建一个新的Connection,
  • 并将其保存到线程本地变量中。
  • 5
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: ThreadLocalJava类库中的一个类,它用于支持线程本地变量,即使在多线程环境中,每个线程也可以维护自己的一个变量副本。它是一种线程隔离的存储机制。下面是一个ThreadLocal使用实例代码:public class TestThreadLocal { private static ThreadLocal<Integer> num = new ThreadLocal<Integer>(){ @Override protected Integer initialValue(){ return 0; } }; public static void main(String[] args) { for (int i = 0; i < 5; i++) { new Thread(() -> { int n = num.get(); n += 5; num.set(n); System.out.println(Thread.currentThread().getName() + ":" + n); }).start(); } } }ThreadLocalJava类库中的一个类,它用于支持线程本地变量,可以在多线程环境中,每个线程维护自己的变量副本,实现线程隔离的存储机制,以上是一个使用ThreadLocal实例代码。 ### 回答2: ThreadLocal是一个用于存储线程局部变量的类。它在多线程程序中非常有用,可以实现每个线程都拥有自己独立的变量副本,互不干扰。 以下是一个ThreadLocal使用实例代码: ``` public class ThreadLocalExample { public static void main(String[] args) { // 创建一个ThreadLocal对象 ThreadLocal<String> threadLocal = new ThreadLocal<>(); // 创建并启动多个线程 for (int i = 0; i < 5; i++) { final int threadNum = i; Thread thread = new Thread(() -> { // 向ThreadLocal中存储线程局部变量 threadLocal.set("Thread " + threadNum + " 的变量"); // 获取并打印线程局部变量的值 System.out.println(threadLocal.get()); // 清除线程局部变量 threadLocal.remove(); }); thread.start(); } } } ``` 上述代码中,首先创建了一个`ThreadLocal`对象`threadLocal`用于存储线程局部变量。然后通过循环创建了5个线程,并通过Lambda表达式定义了每个线程的逻辑。在每个线程中,首先使用`threadLocal.set()`方法将线程局部变量存储到`threadLocal`对象中。接着使用`threadLocal.get()`方法获取该线程局部变量的值,并打印输出。最后使用`threadLocal.remove()`方法清除该线程的局部变量。 运行上述代码,可以观察到每个线程都打印出自己独立的变量副本,互不干扰。这就是通过`ThreadLocal`实现线程局部变量的效果。 ### 回答3: ThreadLocal是一种Java多线程技术,用于在每个线程中创建独立的变量副本。下面是一个使用ThreadLocal实例代码: ```java public class ThreadLocalExample { private static final ThreadLocal<Integer> threadLocal = new ThreadLocal<>(); public static void main(String[] args) throws InterruptedException { Thread t1 = new Thread(() -> { threadLocal.set(10); System.out.println("Thread " + Thread.currentThread().getId() + " value: " + threadLocal.get()); }); Thread t2 = new Thread(() -> { threadLocal.set(20); System.out.println("Thread " + Thread.currentThread().getId() + " value: " + threadLocal.get()); }); t1.start(); t2.start(); t1.join(); t2.join(); System.out.println("Main thread value: " + threadLocal.get()); } } ``` 输出结果可能是: ``` Thread 11 value: 10 Thread 12 value: 20 Main thread value: null ``` 这段代码首先创建了一个ThreadLocal对象,在每个线程中维护一个独立的Integer类型的值。在主线程中,我们将使用threadLocal.get()方法来获取ThreadLocal变量的值。在t1和t2两个子线程中,我们将使用threadLocal.set()方法来设置ThreadLocal变量的值,并使用threadLocal.get()方法来获取它们。最后,我们输出主线程中的ThreadLocal变量的值。 由于ThreadLocal变量是线程私有的,所以每个线程都可以独立设置和获取它们的值,这样就避免了多线程环境下的共享变量冲突。在上面的例子中,每个线程在设置和获取ThreadLocal变量的值时都是独立的,因此线程之间不会相互干扰。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值